Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
go
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
go
Commits
b880543f
Commit
b880543f
authored
Mar 29, 2010
by
Robert Griesemer
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
go/ast: generalized ast filtering
R=rsc CC=golang-dev
https://golang.org/cl/788041
parent
299cd38f
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
129 additions
and
26 deletions
+129
-26
src/pkg/go/ast/ast.go
src/pkg/go/ast/ast.go
+2
-0
src/pkg/go/ast/filter.go
src/pkg/go/ast/filter.go
+127
-26
No files found.
src/pkg/go/ast/ast.go
View file @
b880543f
...
...
@@ -358,6 +358,8 @@ func (x *ChanType) exprNode() {}
// ----------------------------------------------------------------------------
// Convenience functions for Idents
var
noPos
token
.
Position
// NewIdent creates a new Ident without position and minimal object
// information. Useful for ASTs generated by code other than the Go
// parser.
...
...
src/pkg/go/ast/filter.go
View file @
b880543f
...
...
@@ -6,8 +6,10 @@ package ast
import
"go/token"
// ----------------------------------------------------------------------------
// Export filtering
func
filterIdentList
(
list
[]
*
Ident
)
[]
*
Ident
{
func
identListExports
(
list
[]
*
Ident
)
[]
*
Ident
{
j
:=
0
for
_
,
x
:=
range
list
{
if
x
.
IsExported
()
{
...
...
@@ -36,7 +38,7 @@ func isExportedType(typ Expr) bool {
}
func
fi
lterFieldList
(
fields
*
FieldList
,
incomplete
*
bool
)
{
func
fi
eldListExports
(
fields
*
FieldList
,
incomplete
*
bool
)
{
if
fields
==
nil
{
return
}
...
...
@@ -54,14 +56,14 @@ func filterFieldList(fields *FieldList, incomplete *bool) {
exported
=
isExportedType
(
f
.
Type
)
}
else
{
n
:=
len
(
f
.
Names
)
f
.
Names
=
filterIdentList
(
f
.
Names
)
f
.
Names
=
identListExports
(
f
.
Names
)
if
len
(
f
.
Names
)
<
n
{
*
incomplete
=
true
}
exported
=
len
(
f
.
Names
)
>
0
}
if
exported
{
filterType
(
f
.
Type
)
typeExports
(
f
.
Type
)
list
[
j
]
=
f
j
++
}
...
...
@@ -73,49 +75,47 @@ func filterFieldList(fields *FieldList, incomplete *bool) {
}
func
filterParamList
(
fields
*
FieldList
)
{
func
paramListExports
(
fields
*
FieldList
)
{
if
fields
==
nil
{
return
}
for
_
,
f
:=
range
fields
.
List
{
filterType
(
f
.
Type
)
typeExports
(
f
.
Type
)
}
}
var
noPos
token
.
Position
func
filterType
(
typ
Expr
)
{
func
typeExports
(
typ
Expr
)
{
switch
t
:=
typ
.
(
type
)
{
case
*
ArrayType
:
filterType
(
t
.
Elt
)
typeExports
(
t
.
Elt
)
case
*
StructType
:
fi
lterFieldList
(
t
.
Fields
,
&
t
.
Incomplete
)
fi
eldListExports
(
t
.
Fields
,
&
t
.
Incomplete
)
case
*
FuncType
:
filterParamList
(
t
.
Params
)
filterParamList
(
t
.
Results
)
paramListExports
(
t
.
Params
)
paramListExports
(
t
.
Results
)
case
*
InterfaceType
:
fi
lterFieldList
(
t
.
Methods
,
&
t
.
Incomplete
)
fi
eldListExports
(
t
.
Methods
,
&
t
.
Incomplete
)
case
*
MapType
:
filterType
(
t
.
Key
)
filterType
(
t
.
Value
)
typeExports
(
t
.
Key
)
typeExports
(
t
.
Value
)
case
*
ChanType
:
filterType
(
t
.
Value
)
typeExports
(
t
.
Value
)
}
}
func
filterSpec
(
spec
Spec
)
bool
{
func
specExports
(
spec
Spec
)
bool
{
switch
s
:=
spec
.
(
type
)
{
case
*
ValueSpec
:
s
.
Names
=
filterIdentList
(
s
.
Names
)
s
.
Names
=
identListExports
(
s
.
Names
)
if
len
(
s
.
Names
)
>
0
{
filterType
(
s
.
Type
)
typeExports
(
s
.
Type
)
return
true
}
case
*
TypeSpec
:
if
s
.
Name
.
IsExported
()
{
filterType
(
s
.
Type
)
typeExports
(
s
.
Type
)
return
true
}
}
...
...
@@ -123,10 +123,10 @@ func filterSpec(spec Spec) bool {
}
func
filterSpecList
(
list
[]
Spec
)
[]
Spec
{
func
specListExports
(
list
[]
Spec
)
[]
Spec
{
j
:=
0
for
_
,
s
:=
range
list
{
if
filterSpec
(
s
)
{
if
specExports
(
s
)
{
list
[
j
]
=
s
j
++
}
...
...
@@ -135,10 +135,10 @@ func filterSpecList(list []Spec) []Spec {
}
func
filterDecl
(
decl
Decl
)
bool
{
func
declExports
(
decl
Decl
)
bool
{
switch
d
:=
decl
.
(
type
)
{
case
*
GenDecl
:
d
.
Specs
=
filterSpecList
(
d
.
Specs
)
d
.
Specs
=
specListExports
(
d
.
Specs
)
return
len
(
d
.
Specs
)
>
0
case
*
FuncDecl
:
d
.
Body
=
nil
// strip body
...
...
@@ -161,7 +161,7 @@ func filterDecl(decl Decl) bool {
func
FileExports
(
src
*
File
)
bool
{
j
:=
0
for
_
,
d
:=
range
src
.
Decls
{
if
filterDecl
(
d
)
{
if
declExports
(
d
)
{
src
.
Decls
[
j
]
=
d
j
++
}
...
...
@@ -189,6 +189,107 @@ func PackageExports(pkg *Package) bool {
}
// ----------------------------------------------------------------------------
// General filtering
type
Filter
func
(
string
)
bool
func
filterIdentList
(
list
[]
*
Ident
,
f
Filter
)
[]
*
Ident
{
j
:=
0
for
_
,
x
:=
range
list
{
if
f
(
x
.
Name
())
{
list
[
j
]
=
x
j
++
}
}
return
list
[
0
:
j
]
}
func
filterSpec
(
spec
Spec
,
f
Filter
)
bool
{
switch
s
:=
spec
.
(
type
)
{
case
*
ValueSpec
:
s
.
Names
=
filterIdentList
(
s
.
Names
,
f
)
return
len
(
s
.
Names
)
>
0
case
*
TypeSpec
:
return
f
(
s
.
Name
.
Name
())
}
return
false
}
func
filterSpecList
(
list
[]
Spec
,
f
Filter
)
[]
Spec
{
j
:=
0
for
_
,
s
:=
range
list
{
if
filterSpec
(
s
,
f
)
{
list
[
j
]
=
s
j
++
}
}
return
list
[
0
:
j
]
}
func
filterDecl
(
decl
Decl
,
f
Filter
)
bool
{
switch
d
:=
decl
.
(
type
)
{
case
*
GenDecl
:
d
.
Specs
=
filterSpecList
(
d
.
Specs
,
f
)
return
len
(
d
.
Specs
)
>
0
case
*
FuncDecl
:
return
f
(
d
.
Name
.
Name
())
}
return
false
}
// FilterFile trims the AST for a Go file in place by removing all
// names from top-level declarations (but not from parameter lists
// or inside types) that don't pass through the filter f. If the
// declaration is empty afterwards, the declaration is removed from
// the AST.
// The File.comments list is not changed.
//
// FilterFile returns true if there are any top-level declarations
// left after filtering; it returns false otherwise.
//
func
FilterFile
(
src
*
File
,
f
Filter
)
bool
{
j
:=
0
for
_
,
d
:=
range
src
.
Decls
{
if
filterDecl
(
d
,
f
)
{
src
.
Decls
[
j
]
=
d
j
++
}
}
src
.
Decls
=
src
.
Decls
[
0
:
j
]
return
j
>
0
}
// FilterPackage trims the AST for a Go package in place by removing all
// names from top-level declarations (but not from parameter lists
// or inside types) that don't pass through the filter f. If the
// declaration is empty afterwards, the declaration is removed from
// the AST.
// The pkg.Files list is not changed, so that file names and top-level
// package comments don't get lost.
//
// FilterPackage returns true if there are any top-level declarations
// left after filtering; it returns false otherwise.
//
func
FilterPackage
(
pkg
*
Package
,
f
Filter
)
bool
{
hasDecls
:=
false
for
_
,
src
:=
range
pkg
.
Files
{
if
FilterFile
(
src
,
f
)
{
hasDecls
=
true
}
}
return
hasDecls
}
// ----------------------------------------------------------------------------
// Merging of package files
// separator is an empty //-style comment that is interspersed between
// different comment groups when they are concatenated into a single group
//
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment