// This Source Code Form is subject to the terms of the Mozilla Public // License, v. 2.0. If a copy of the MPL was not distributed with this // file, You can obtain one at http://mozilla.org/MPL/2.0/. // structprotogen is a tool to generate proto files from Go structs. package main //nolint:gci import ( "bytes" "fmt" "os" "path" "path/filepath" "github.com/spf13/cobra" "github.com/siderolabs/structprotogen/ast" "github.com/siderolabs/structprotogen/loader" "github.com/siderolabs/structprotogen/proto" "github.com/siderolabs/structprotogen/types" ) // rootCmd represents the base command when called without any subcommands. var rootCmd = &cobra.Command{ Use: "structprotogen path dest", Short: "This CLI is used to generate proto files from Go structs into one proto file", Example: "gotagsrewrite github.com/siderolabs/talos/pkg/machinery/resources/... ./api/resource/definitions", Args: cobra.ExactArgs(2), Version: "v1.0.0", RunE: func(cmd *cobra.Command, args []string) error { return run(args[0], args[1]) }, SilenceUsage: true, DisableAutoGenTag: true, } func main() { err := rootCmd.Execute() if err != nil { os.Exit(1) } } // TODO(DmitriyMV): get comments for fields //nolint:gocyclo func run(pkgPath, dst string) error { loadedPkgs, err := loader.LoadPackages(pkgPath) if err != nil { return err } taggedStructs := ast.FindAllTaggedStructs(loadedPkgs) printFoundStructs(taggedStructs) sortedPkgs, err := types.FindPkgDecls(taggedStructs, loadedPkgs) if err != nil { return fmt.Errorf("error finding path '%s' declarations: %w", pkgPath, err) } pkgsTypes, err := types.ParseDeclsData(sortedPkgs, taggedStructs) if err != nil { return fmt.Errorf("error parsing path '%s' declarations data: %w", pkgPath, err) } externalTypes := types.FindExternalTypes(pkgsTypes, taggedStructs) for i := 0; i < externalTypes.Len(); i++ { externalType := externalTypes.Get(i) if !proto.IsSupportedExternalType(externalType) { return fmt.Errorf("external type '%s.%s' is not supported", externalType.Pkg, externalType.Name) } } data := proto.PrepareProtoData(pkgsTypes) for i := 0; i < data.Len(); i++ { protoData := data.Get(i) fmt.Println("--------") protoData.WriteDebug(os.Stdout) } err = os.MkdirAll(dst, 0o755) if err != nil { return fmt.Errorf("failed to create directory for proto files: %w", err) } for i := 0; i < data.Len(); i++ { protoData := data.Get(i) dstDir, err := filepath.Abs(filepath.Join(dst, protoData.Name)) if err != nil { return fmt.Errorf("failed to get absolute path for pkg '%s': %w", protoData.GoPkg, err) } err = os.MkdirAll(dstDir, 0o755) if err != nil { return fmt.Errorf("failed to create directory for pkg '%s' proto files: %w", protoData.GoPkg, err) } dstFile, err := filepath.Abs(filepath.Join(dstDir, path.Base(protoData.GoPkg)+".proto")) if err != nil { return fmt.Errorf("failed to get absolute path for destination file: %w", err) } fmt.Println("writing file", dstFile) var buf bytes.Buffer protoData.Format(&buf) if err := os.WriteFile(dstFile, buf.Bytes(), os.ModePerm); err != nil { return fmt.Errorf("failed to write file '%s': %w", dstFile, err) } } return nil } func printFoundStructs(structs ast.TaggedStructs) { for decl := range structs { fmt.Printf("found tagged struct '%s' in pkg '%s'\n", decl.Name, decl.Pkg) } }