-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfind_struct_def.go
226 lines (191 loc) · 5.11 KB
/
find_struct_def.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
package quickclop
import (
"go/ast"
"go/token"
"log"
"path/filepath"
"strings"
"golang.org/x/tools/go/packages"
)
// 新增缓存:包路径 -> 包语法树
var pkgCache = make(map[string][]*ast.File)
// findStructDef 优化后的实现
func findStructDef(typeName string, file *ast.File, path string, subcommand string) *ast.StructType {
// 基本类型不需要查找结构体定义
if isBasicType(typeName) {
return nil
}
// 时间类型不需要查找结构体定义
if isTimeType(typeName) {
return nil
}
// 处理指针类型
if strings.HasPrefix(typeName, "*") {
typeName = typeName[1:]
}
// 处理切片类型
if strings.HasPrefix(typeName, "[]") {
return nil
}
// 处理映射类型
if strings.HasPrefix(typeName, "map[") {
return nil
}
// 如果提供了文件,则在该文件中查找
if file != nil {
for _, decl := range file.Decls {
genDecl, ok := decl.(*ast.GenDecl)
if !ok || genDecl.Tok != token.TYPE {
continue
}
for _, spec := range genDecl.Specs {
typeSpec, ok := spec.(*ast.TypeSpec)
if !ok {
continue
}
if typeSpec.Name.Name == typeName {
if structType, ok := typeSpec.Type.(*ast.StructType); ok {
return structType
}
}
}
}
}
return nil
}
// 1. 在当前文件查找
func findInCurrentFile(typeName string, file *ast.File) *ast.StructType {
for _, decl := range file.Decls {
genDecl, ok := decl.(*ast.GenDecl)
if !ok || genDecl.Tok != token.TYPE {
continue
}
for _, spec := range genDecl.Specs {
typeSpec, ok := spec.(*ast.TypeSpec)
if !ok || typeSpec.Name.Name != typeName {
continue
}
if structType, ok := typeSpec.Type.(*ast.StructType); ok {
log.Printf("Found struct in current file: %s", typeName)
return structType
}
}
}
return nil
}
// 2. 在导入的包中查找
func findInImportedPackage(pkgPath, typeName string, file *ast.File) *ast.StructType {
// 从文件的 imports 中查找完整的包路径
var fullPkgPath string
for _, imp := range file.Imports {
importedPath := strings.Trim(imp.Path.Value, `"`)
// 匹配别名或路径最后部分
if imp.Name != nil && imp.Name.Name == pkgPath {
fullPkgPath = importedPath
break
}
if filepath.Base(importedPath) == pkgPath {
fullPkgPath = importedPath
break
}
}
if fullPkgPath == "" {
log.Printf("Package not imported: %s", pkgPath)
return nil
}
return findStructDefInPackage(fullPkgPath, typeName)
}
// 3. 在同一包的其他文件中查找
func findInCurrentPackage(typeName string, file *ast.File, path string) *ast.StructType {
pkgPath := filepath.Dir(path)
// 检查缓存
files, ok := pkgCache[pkgPath]
if !ok {
// 未缓存,加载包信息
cfg := &packages.Config{
Mode: packages.NeedName | packages.NeedFiles | packages.NeedSyntax,
Dir: pkgPath,
}
pkgs, err := packages.Load(cfg, ".")
if err != nil || len(pkgs) == 0 {
log.Printf("Failed to load current package: %v", err)
return nil
}
// 提取语法树并缓存
var syntax []*ast.File
for _, pkg := range pkgs {
syntax = append(syntax, pkg.Syntax...)
}
pkgCache[pkgPath] = syntax
files = syntax
}
// 遍历所有文件查找结构体
for _, f := range files {
if structType := findInCurrentFile(typeName, f); structType != nil {
log.Printf("Found struct in package %s: %s", pkgPath, typeName)
return structType
}
}
return nil
}
// findStructDefInPackage 在包中查找结构体定义
func findStructDefInPackage(pkgPath, structName string) (structType *ast.StructType) {
// 配置 packages.Config
cfg := &packages.Config{
Mode: packages.NeedName |
packages.NeedSyntax |
packages.NeedTypes |
packages.NeedDeps,
}
// 加载包
pkgs, err := packages.Load(cfg, pkgPath)
if err != nil {
log.Printf("Failed to load package %s: %v", pkgPath, err)
return nil
}
if len(pkgs) == 0 {
log.Printf("No packages found for %s", pkgPath)
return nil
}
// 遍历包中的文件
for _, pkg := range pkgs {
log.Printf("Inspecting package: %s, syntax: %d\n", pkg.PkgPath, len(pkg.Syntax))
for _, file := range pkg.Syntax {
// 遍历 AST
ast.Inspect(file, func(n ast.Node) bool {
// 查找类型声明
ts, ok := n.(*ast.TypeSpec)
if !ok {
return true
}
log.Printf("Found type: %s", ts.Name.Name)
if ts.Name.Name != structName {
return true // 继续遍历
}
// 找到目标结构体
log.Printf("Found type: %s, struceName:%s\n", ts.Name.Name, structName)
if structType, ok = ts.Type.(*ast.StructType); ok {
log.Printf("Type %s is a struct with fields:", ts.Name.Name)
for _, field := range structType.Fields.List {
for _, fieldName := range field.Names {
log.Printf(" - %s", fieldName.Name)
}
}
return false // 退出遍历
}
return true
})
}
}
if structType == nil {
log.Printf("Struct definition not found: %s in package %s", structName, pkgPath)
}
return structType
}
func parsePkgType(typeStr string) (pkgPath, typeName string) {
parts := strings.Split(typeStr, ".")
if len(parts) > 1 {
return strings.Join(parts[:len(parts)-1], "."), parts[len(parts)-1]
}
return "", typeStr
}