diff --git a/main.go b/main.go index 43bc32e..8f75048 100644 --- a/main.go +++ b/main.go @@ -673,7 +673,7 @@ func transformCompile(args []string) ([]string, error) { } debugFilePath := filepath.Join(pkgDebugDir, name) - if err := os.WriteFile(debugFilePath, src, 0666); err != nil { + if err := os.WriteFile(debugFilePath, src, 0o666); err != nil { return nil, err } } @@ -972,7 +972,8 @@ func (tf *transformer) recordReflectArgs(files []*ast.File) { if fnType.Pkg().Path() == "reflect" && (fnType.Name() == "TypeOf" || fnType.Name() == "ValueOf") { for _, arg := range call.Args { - tf.recordIgnore(tf.info.TypeOf(arg), false) + argType := tf.info.TypeOf(arg) + tf.recordIgnore(argType, false) } } return true @@ -1301,6 +1302,13 @@ func (tf *transformer) recordIgnore(t types.Type, allPkgs bool) { for i := 0; i < t.NumFields(); i++ { field := t.Field(i) + // This check is similar to the one in *types.Named. + // It's necessary for unnamed struct types, + // as they aren't named but still have named fields. + if !allPkgs && field.Pkg() != tf.pkg { + return // not from the current package + } + // Record the field itself, too. tf.ignoreObjects[field] = true diff --git a/testdata/scripts/reflect.txt b/testdata/scripts/reflect.txt index bb50515..ecfbadf 100644 --- a/testdata/scripts/reflect.txt +++ b/testdata/scripts/reflect.txt @@ -51,7 +51,7 @@ func main() { // Use of a common library using reflect, encoding/json. // TODO: remove the TypeOf hint once we detect json.Marshal. - var _ = reflect.TypeOf(EncodingT{}) + _ = reflect.TypeOf(EncodingT{}) enc, _ := json.Marshal(EncodingT{Foo: 3}) println(string(enc)) @@ -60,10 +60,24 @@ func main() { outer.InnerField = 3 enc, _ = json.Marshal(outer) println(string(enc)) + + // An edge case; the struct type is defined in a different package. + // Note that the struct type is unnamed, but it still has named fields. + // We only use reflection on it here, not the declaring package. + // As such, we should obfuscate the field name. + // Simply using the field name here used to cause build failures. + _ = reflect.TypeOf(importedpkg.UnnamedWithDownstreamReflect{}) + fmt.Printf("%v\n", importedpkg.UnnamedWithDownstreamReflect{ + ExportedField: "foo", + }) } func printfWithoutPackage(format string, v interface{}) { - fmt.Print(strings.Split(fmt.Sprintf(format, v), ".")[1]) + s := fmt.Sprintf(format, v) + if i := strings.IndexByte(s, '.'); i > 0 { + s = s[i+1:] + } + fmt.Print(s) } type EncodingT struct { @@ -128,6 +142,10 @@ type EmbeddingInner struct { InnerField int } +type UnnamedWithDownstreamReflect = struct { + ExportedField string +} + -- main.stdout -- 9000 {5 0} @@ -135,3 +153,4 @@ ReflectTypeOf ReflectTypeOfIndirect ReflectValueOf{ExportedField:"abc", unexportedField:""} [method: abc] +{foo}