| 
									
										
										
										
											2023-01-27 13:37:20 -08:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Cloner is a tool to automate the creation of a Clone method. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The result of the Clone method aliases no memory that can be edited | 
					
						
							|  |  |  | // with the original. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This tool makes lots of implicit assumptions about the types you feed it. | 
					
						
							|  |  |  | // In particular, it can only write relatively "shallow" Clone methods. | 
					
						
							|  |  |  | // That is, if a type contains another named struct type, cloner assumes that | 
					
						
							|  |  |  | // named type will also have a Clone method. | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"flag" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"go/types" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 15:41:57 -07:00
										 |  |  | 	"tailscale.com/util/codegen" | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	flagTypes     = flag.String("type", "", "comma-separated list of types; required") | 
					
						
							|  |  |  | 	flagBuildTags = flag.String("tags", "", "compiler build tags to apply") | 
					
						
							| 
									
										
										
										
											2020-10-19 10:46:30 -07:00
										 |  |  | 	flagCloneFunc = flag.Bool("clonefunc", false, "add a top-level Clone func") | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							|  |  |  | 	log.SetFlags(0) | 
					
						
							|  |  |  | 	log.SetPrefix("cloner: ") | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 	if len(*flagTypes) == 0 { | 
					
						
							|  |  |  | 		flag.Usage() | 
					
						
							|  |  |  | 		os.Exit(2) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	typeNames := strings.Split(*flagTypes, ",") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 	pkg, namedTypes, err := codegen.LoadTypes(*flagBuildTags, ".") | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		log.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 	it := codegen.NewImportTracker(pkg.Types) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 	buf := new(bytes.Buffer) | 
					
						
							|  |  |  | 	for _, typeName := range typeNames { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:24:50 -07:00
										 |  |  | 		typ, ok := namedTypes[typeName] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 			log.Fatalf("could not find type %s", typeName) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 		gen(buf, it, typ) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-16 16:27:57 -07:00
										 |  |  | 	w := func(format string, args ...any) { | 
					
						
							| 
									
										
										
										
											2020-09-04 15:19:12 -07:00
										 |  |  | 		fmt.Fprintf(buf, format+"\n", args...) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-19 10:46:30 -07:00
										 |  |  | 	if *flagCloneFunc { | 
					
						
							|  |  |  | 		w("// Clone duplicates src into dst and reports whether it succeeded.") | 
					
						
							|  |  |  | 		w("// To succeed, <src, dst> must be of types <*T, *T> or <*T, **T>,") | 
					
						
							|  |  |  | 		w("// where T is one of %s.", *flagTypes) | 
					
						
							| 
									
										
										
										
											2022-03-16 16:27:57 -07:00
										 |  |  | 		w("func Clone(dst, src any) bool {") | 
					
						
							| 
									
										
										
										
											2020-10-19 10:46:30 -07:00
										 |  |  | 		w("	switch src := src.(type) {") | 
					
						
							|  |  |  | 		for _, typeName := range typeNames { | 
					
						
							|  |  |  | 			w("	case *%s:", typeName) | 
					
						
							|  |  |  | 			w("		switch dst := dst.(type) {") | 
					
						
							|  |  |  | 			w("		case *%s:", typeName) | 
					
						
							|  |  |  | 			w("			*dst = *src.Clone()") | 
					
						
							|  |  |  | 			w("			return true") | 
					
						
							|  |  |  | 			w("		case **%s:", typeName) | 
					
						
							|  |  |  | 			w("			*dst = src.Clone()") | 
					
						
							|  |  |  | 			w("			return true") | 
					
						
							|  |  |  | 			w("		}") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		w("	}") | 
					
						
							|  |  |  | 		w("	return false") | 
					
						
							|  |  |  | 		w("}") | 
					
						
							| 
									
										
										
										
											2020-09-04 15:19:12 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-07-12 14:59:35 -05:00
										 |  |  | 	cloneOutput := pkg.Name + "_clone" | 
					
						
							|  |  |  | 	if *flagBuildTags == "test" { | 
					
						
							|  |  |  | 		cloneOutput += "_test" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cloneOutput += ".go" | 
					
						
							| 
									
										
										
										
											2023-01-27 13:36:46 -08:00
										 |  |  | 	if err := codegen.WritePackageFile("tailscale.com/cmd/cloner", pkg, cloneOutput, it, buf); err != nil { | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 		log.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | func gen(buf *bytes.Buffer, it *codegen.ImportTracker, typ *types.Named) { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 	t, ok := typ.Underlying().(*types.Struct) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	name := typ.Obj().Name() | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 	typeParams := typ.Origin().TypeParams() | 
					
						
							|  |  |  | 	_, typeParamNames := codegen.FormatTypeParams(typeParams, it) | 
					
						
							|  |  |  | 	nameWithParams := name + typeParamNames | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 	fmt.Fprintf(buf, "// Clone makes a deep copy of %s.\n", name) | 
					
						
							|  |  |  | 	fmt.Fprintf(buf, "// The result aliases no memory with the original.\n") | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 	fmt.Fprintf(buf, "func (src *%s) Clone() *%s {\n", nameWithParams, nameWithParams) | 
					
						
							| 
									
										
										
										
											2022-03-16 16:27:57 -07:00
										 |  |  | 	writef := func(format string, args ...any) { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 		fmt.Fprintf(buf, "\t"+format+"\n", args...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	writef("if src == nil {") | 
					
						
							|  |  |  | 	writef("\treturn nil") | 
					
						
							|  |  |  | 	writef("}") | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 	writef("dst := new(%s)", nameWithParams) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 	writef("*dst = *src") | 
					
						
							| 
									
										
										
										
											2024-04-16 13:15:13 -07:00
										 |  |  | 	for i := range t.NumFields() { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 		fname := t.Field(i).Name() | 
					
						
							|  |  |  | 		ft := t.Field(i).Type() | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 		if !codegen.ContainsPointers(ft) || codegen.HasNoClone(t.Tag(i)) { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 			continue | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-02-15 08:19:44 -08:00
										 |  |  | 		if named, _ := ft.(*types.Named); named != nil { | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 			if codegen.IsViewType(ft) { | 
					
						
							| 
									
										
										
										
											2022-02-15 08:19:44 -08:00
										 |  |  | 				writef("dst.%s = src.%s", fname, fname) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if !hasBasicUnderlying(ft) { | 
					
						
							|  |  |  | 				writef("dst.%s = *src.%s.Clone()", fname, fname) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		switch ft := ft.Underlying().(type) { | 
					
						
							|  |  |  | 		case *types.Slice: | 
					
						
							| 
									
										
										
										
											2021-09-17 11:29:17 -07:00
										 |  |  | 			if codegen.ContainsPointers(ft.Elem()) { | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 				n := it.QualifiedName(ft.Elem()) | 
					
						
							| 
									
										
										
										
											2023-09-29 18:15:53 -07:00
										 |  |  | 				writef("if src.%s != nil {", fname) | 
					
						
							|  |  |  | 				writef("dst.%s = make([]%s, len(src.%s))", fname, n, fname) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("for i := range dst.%s {", fname) | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:20 -07:00
										 |  |  | 				if ptr, isPtr := ft.Elem().(*types.Pointer); isPtr { | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 					writef("if src.%s[i] == nil { dst.%s[i] = nil } else {", fname, fname) | 
					
						
							|  |  |  | 					if codegen.ContainsPointers(ptr.Elem()) { | 
					
						
							|  |  |  | 						if _, isIface := ptr.Elem().Underlying().(*types.Interface); isIface { | 
					
						
							|  |  |  | 							it.Import("tailscale.com/types/ptr") | 
					
						
							|  |  |  | 							writef("\tdst.%s[i] = ptr.To((*src.%s[i]).Clone())", fname, fname) | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							writef("\tdst.%s[i] = src.%s[i].Clone()", fname, fname) | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 						it.Import("tailscale.com/types/ptr") | 
					
						
							|  |  |  | 						writef("\tdst.%s[i] = ptr.To(*src.%s[i])", fname, fname) | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:20 -07:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 					writef("}") | 
					
						
							| 
									
										
										
										
											2023-07-24 21:07:00 -07:00
										 |  |  | 				} else if ft.Elem().String() == "encoding/json.RawMessage" { | 
					
						
							|  |  |  | 					writef("\tdst.%s[i] = append(src.%s[i][:0:0], src.%s[i]...)", fname, fname, fname) | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 				} else if _, isIface := ft.Elem().Underlying().(*types.Interface); isIface { | 
					
						
							|  |  |  | 					writef("\tdst.%s[i] = src.%s[i].Clone()", fname, fname) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 					writef("\tdst.%s[i] = *src.%s[i].Clone()", fname, fname) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				writef("}") | 
					
						
							| 
									
										
										
										
											2023-09-29 18:15:53 -07:00
										 |  |  | 				writef("}") | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				writef("dst.%s = append(src.%s[:0:0], src.%s...)", fname, fname, fname) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 		case *types.Pointer: | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 			base := ft.Elem() | 
					
						
							|  |  |  | 			hasPtrs := codegen.ContainsPointers(base) | 
					
						
							|  |  |  | 			if named, _ := base.(*types.Named); named != nil && hasPtrs { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("dst.%s = src.%s.Clone()", fname, fname) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 			it.Import("tailscale.com/types/ptr") | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 			writef("if dst.%s != nil {", fname) | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 			if _, isIface := base.Underlying().(*types.Interface); isIface && hasPtrs { | 
					
						
							|  |  |  | 				writef("\tdst.%s = ptr.To((*src.%s).Clone())", fname, fname) | 
					
						
							|  |  |  | 			} else if !hasPtrs { | 
					
						
							|  |  |  | 				writef("\tdst.%s = ptr.To(*src.%s)", fname, fname) | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("\t" + `panic("TODO pointers in pointers")`) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			writef("}") | 
					
						
							|  |  |  | 		case *types.Map: | 
					
						
							| 
									
										
										
										
											2022-07-12 23:52:44 -07:00
										 |  |  | 			elem := ft.Elem() | 
					
						
							|  |  |  | 			if sliceType, isSlice := elem.(*types.Slice); isSlice { | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | 				n := it.QualifiedName(sliceType.Elem()) | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 				writef("if dst.%s != nil {", fname) | 
					
						
							|  |  |  | 				writef("\tdst.%s = map[%s]%s{}", fname, it.QualifiedName(ft.Key()), it.QualifiedName(elem)) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("\tfor k := range src.%s {", fname) | 
					
						
							|  |  |  | 				// use zero-length slice instead of nil to ensure | 
					
						
							|  |  |  | 				// the key is always copied. | 
					
						
							|  |  |  | 				writef("\t\tdst.%s[k] = append([]%s{}, src.%s[k]...)", fname, n, fname) | 
					
						
							|  |  |  | 				writef("\t}") | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 				writef("}") | 
					
						
							| 
									
										
										
										
											2022-07-12 23:52:44 -07:00
										 |  |  | 			} else if codegen.ContainsPointers(elem) { | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 				writef("if dst.%s != nil {", fname) | 
					
						
							|  |  |  | 				writef("\tdst.%s = map[%s]%s{}", fname, it.QualifiedName(ft.Key()), it.QualifiedName(elem)) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("\tfor k, v := range src.%s {", fname) | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				switch elem := elem.Underlying().(type) { | 
					
						
							| 
									
										
										
										
											2022-07-12 23:52:44 -07:00
										 |  |  | 				case *types.Pointer: | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 					writef("\t\tif v == nil { dst.%s[k] = nil } else {", fname) | 
					
						
							|  |  |  | 					if base := elem.Elem().Underlying(); codegen.ContainsPointers(base) { | 
					
						
							|  |  |  | 						if _, isIface := base.(*types.Interface); isIface { | 
					
						
							|  |  |  | 							it.Import("tailscale.com/types/ptr") | 
					
						
							|  |  |  | 							writef("\t\t\tdst.%s[k] = ptr.To((*v).Clone())", fname) | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							writef("\t\t\tdst.%s[k] = v.Clone()", fname) | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						it.Import("tailscale.com/types/ptr") | 
					
						
							|  |  |  | 						writef("\t\t\tdst.%s[k] = ptr.To(*v)", fname) | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					writef("}") | 
					
						
							|  |  |  | 				case *types.Interface: | 
					
						
							|  |  |  | 					if cloneResultType := methodResultType(elem, "Clone"); cloneResultType != nil { | 
					
						
							|  |  |  | 						if _, isPtr := cloneResultType.(*types.Pointer); isPtr { | 
					
						
							|  |  |  | 							writef("\t\tdst.%s[k] = *(v.Clone())", fname) | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							writef("\t\tdst.%s[k] = v.Clone()", fname) | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						writef(`panic("%s (%v) does not have a Clone method")`, fname, elem) | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2022-07-12 23:52:44 -07:00
										 |  |  | 				default: | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 					writef("\t\tdst.%s[k] = *(v.Clone())", fname) | 
					
						
							| 
									
										
										
										
											2022-07-12 23:52:44 -07:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 				writef("\t}") | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 				writef("}") | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2023-08-20 13:16:06 -04:00
										 |  |  | 				it.Import("maps") | 
					
						
							|  |  |  | 				writef("\tdst.%s = maps.Clone(src.%s)", fname, fname) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 		case *types.Interface: | 
					
						
							|  |  |  | 			// If ft is an interface with a "Clone() ft" method, it can be used to clone the field. | 
					
						
							|  |  |  | 			// This includes scenarios where ft is a constrained type parameter. | 
					
						
							|  |  |  | 			if cloneResultType := methodResultType(ft, "Clone"); cloneResultType.Underlying() == ft { | 
					
						
							|  |  |  | 				writef("dst.%s = src.%s.Clone()", fname, fname) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			writef(`panic("%s (%v) does not have a compatible Clone method")`, fname, ft) | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2021-09-16 16:31:35 -07:00
										 |  |  | 			writef(`panic("TODO: %s (%T)")`, fname, ft) | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-16 16:00:34 -07:00
										 |  |  | 	writef("return dst") | 
					
						
							|  |  |  | 	fmt.Fprintf(buf, "}\n\n") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 	buf.Write(codegen.AssertStructUnchanged(t, name, typeParams, "Clone", it)) | 
					
						
							| 
									
										
										
										
											2022-02-15 08:19:44 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 16:15:14 -07:00
										 |  |  | // hasBasicUnderlying reports true when typ.Underlying() is a slice or a map. | 
					
						
							| 
									
										
										
										
											2020-07-24 18:00:02 +10:00
										 |  |  | func hasBasicUnderlying(typ types.Type) bool { | 
					
						
							|  |  |  | 	switch typ.Underlying().(type) { | 
					
						
							|  |  |  | 	case *types.Slice, *types.Map: | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-07-08 10:11:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | func methodResultType(typ types.Type, method string) types.Type { | 
					
						
							|  |  |  | 	viewMethod := codegen.LookupMethod(typ, method) | 
					
						
							|  |  |  | 	if viewMethod == nil { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sig, ok := viewMethod.Type().(*types.Signature) | 
					
						
							|  |  |  | 	if !ok || sig.Results().Len() != 1 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return sig.Results().At(0).Type() | 
					
						
							|  |  |  | } |