From 6436a44be0a26a1f884f94980b7e861638120e67 Mon Sep 17 00:00:00 2001 From: Taylor Sutton Date: Thu, 21 Apr 2022 13:41:39 -0700 Subject: [PATCH 1/2] Remove deep references to GOPATH With this change, GOPATH is still used, but only at the top level (main.go) where it's used to determine the absolute path for output. --- clients/go/gengo.go | 15 ++++++------ main.go | 55 +++++++++++++++++-------------------------- main_test.go | 4 ---- models/genmodels.go | 16 ++++++------- server/gendb/gendb.go | 4 ++-- server/genserver.go | 20 ++++++++-------- swagger/swagger.go | 8 +++---- 7 files changed, 53 insertions(+), 69 deletions(-) diff --git a/clients/go/gengo.go b/clients/go/gengo.go index 76936946..580be160 100644 --- a/clients/go/gengo.go +++ b/clients/go/gengo.go @@ -14,11 +14,11 @@ import ( ) // Generate generates a client -func Generate(packageName, packagePath string, s spec.Swagger) error { - if err := generateClient(packageName, packagePath, s); err != nil { +func Generate(packageName, basePath string, s spec.Swagger) error { + if err := generateClient(packageName, basePath, s); err != nil { return err } - return generateInterface(packageName, packagePath, &s, s.Info.InfoProps.Title, s.Paths) + return generateInterface(packageName, basePath, &s, s.Info.InfoProps.Title, s.Paths) } type clientCodeTemplate struct { @@ -197,8 +197,7 @@ func shortHash(s string) string { } ` -func generateClient(packageName, packagePath string, s spec.Swagger) error { - +func generateClient(packageName, basePath string, s spec.Swagger) error { codeTemplate := clientCodeTemplate{ PackageName: packageName, ServiceName: s.Info.InfoProps.Title, @@ -227,7 +226,7 @@ func generateClient(packageName, packagePath string, s spec.Swagger) error { return err } - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(clientCode) return g.WriteFile("client/client.go") } @@ -248,8 +247,8 @@ func IsBinaryParam(param spec.Parameter, definitions map[string]spec.Schema) boo return definitions[definitionName].Format == "binary" } -func generateInterface(packageName, packagePath string, s *spec.Swagger, serviceName string, paths *spec.Paths) error { - g := swagger.Generator{PackagePath: packagePath} +func generateInterface(packageName, basePath string, s *spec.Swagger, serviceName string, paths *spec.Paths) error { + g := swagger.Generator{BasePath: basePath} g.Printf("package client\n\n") g.Printf(swagger.ImportStatements([]string{"context", packageName + "/models"})) g.Printf("//go:generate mockgen -source=$GOFILE -destination=mock_client.go -package=client\n\n") diff --git a/main.go b/main.go index 7bb33e0e..eb94e47d 100644 --- a/main.go +++ b/main.go @@ -38,13 +38,9 @@ type config struct { goPackageName *string dynamoPath string - goAbsolutePackagePath string - goClientPath string goPackagePath string + goAbsolutePackagePath string jsClientPath string - modelsPath string - serverPath string - tracingPath string generateDynamo bool generateGoClient bool @@ -95,31 +91,31 @@ func main() { } if conf.generateGoModels { - if err := generateGoModels(conf.modelsPath, conf.goPackagePath, swaggerSpec); err != nil { + if err := generateGoModels(conf.goAbsolutePackagePath, swaggerSpec); err != nil { log.Fatal(err.Error()) } } if conf.generateServer { - if err := generateServer(conf.serverPath, *conf.goPackageName, conf.goPackagePath, swaggerSpec); err != nil { + if err := generateServer(*conf.goPackageName, conf.goAbsolutePackagePath, swaggerSpec); err != nil { log.Fatal(err.Error()) } } if conf.generateTracing { - if err := generateTracing(conf.tracingPath, conf.goPackagePath); err != nil { + if err := generateTracing(conf.goAbsolutePackagePath); err != nil { log.Fatal(err.Error()) } } if conf.generateDynamo { - if err := generateDynamo(conf.dynamoPath, *conf.goPackageName, conf.goPackagePath, *conf.relativeDynamoPath, swaggerSpec); err != nil { + if err := generateDynamo(conf.dynamoPath, *conf.goPackageName, conf.goAbsolutePackagePath, *conf.relativeDynamoPath, swaggerSpec); err != nil { log.Fatal(err.Error()) } } if conf.generateGoClient { - if err := generateGoClient(conf.goClientPath, *conf.goPackageName, conf.goPackagePath, swaggerSpec); err != nil { + if err := generateGoClient(*conf.goPackageName, conf.goAbsolutePackagePath, swaggerSpec); err != nil { log.Fatal(err.Error()) } } @@ -131,24 +127,24 @@ func main() { } } -func generateGoModels(modelsPath, goPackagePath string, swaggerSpec spec.Swagger) error { - if err := prepareDir(modelsPath); err != nil { +func generateGoModels(basePath string, swaggerSpec spec.Swagger) error { + if err := prepareDir(filepath.Join(basePath, "models")); err != nil { return err } - if err := models.Generate(goPackagePath, swaggerSpec); err != nil { + if err := models.Generate(basePath, swaggerSpec); err != nil { return fmt.Errorf("Error generating models: %s", err) } return nil } -func generateServer(serverPath, goPackageName, goPackagePath string, swaggerSpec spec.Swagger) error { - if err := prepareDir(serverPath); err != nil { +func generateServer(goPackageName, basePath string, swaggerSpec spec.Swagger) error { + if err := prepareDir(filepath.Join(basePath, "server")); err != nil { return err } - if err := server.Generate(goPackageName, goPackagePath, swaggerSpec); err != nil { + if err := server.Generate(goPackageName, basePath, swaggerSpec); err != nil { return fmt.Errorf("Failed to generate server: %s", err) } - middlewareGenerator := swagger.Generator{PackagePath: goPackagePath} + middlewareGenerator := swagger.Generator{BasePath: basePath} middlewareGenerator.Write(hardcoded.MustAsset("../_hardcoded/middleware.go")) if err := middlewareGenerator.WriteFile("server/middleware.go"); err != nil { return fmt.Errorf("Failed to copy middleware.go: %s", err) @@ -157,12 +153,12 @@ func generateServer(serverPath, goPackageName, goPackagePath string, swaggerSpec return nil } -func generateTracing(tracingPath, goPackagePath string) error { - if err := prepareDir(tracingPath); err != nil { +func generateTracing(basePath string) error { + if err := prepareDir(filepath.Join(basePath, "tracing")); err != nil { return err } - tracingGenerator := swagger.Generator{PackagePath: goPackagePath} + tracingGenerator := swagger.Generator{BasePath: basePath} tracingGenerator.Write(hardcoded.MustAsset("../_hardcoded/tracing.go")) if err := tracingGenerator.WriteFile("tracing/tracing.go"); err != nil { log.Fatalf("Failed to copy tracing.go: %s", err) @@ -171,24 +167,24 @@ func generateTracing(tracingPath, goPackagePath string) error { return nil } -func generateDynamo(dynamoPath, goPackageName, goPackagePath, outputPath string, swaggerSpec spec.Swagger) error { +func generateDynamo(dynamoPath, goPackageName, basePath, outputPath string, swaggerSpec spec.Swagger) error { if err := prepareDir(dynamoPath); err != nil { return err } - if err := gendb.GenerateDB(goPackageName, goPackagePath, &swaggerSpec, outputPath); err != nil { + if err := gendb.GenerateDB(goPackageName, basePath, &swaggerSpec, outputPath); err != nil { return fmt.Errorf("Failed to generate database: %s", err) } return nil } -func generateGoClient(goClientPath, goPackageName, goPackagePath string, swaggerSpec spec.Swagger) error { - if err := prepareDir(goClientPath); err != nil { +func generateGoClient(goPackageName, basePath string, swaggerSpec spec.Swagger) error { + if err := prepareDir(filepath.Join(basePath, "client")); err != nil { return err } - if err := goclient.Generate(goPackageName, goPackagePath, swaggerSpec); err != nil { + if err := goclient.Generate(goPackageName, basePath, swaggerSpec); err != nil { return fmt.Errorf("Failed generating go client %s", err) } - doerGenerator := swagger.Generator{PackagePath: goPackagePath} + doerGenerator := swagger.Generator{BasePath: basePath} doerGenerator.Write(hardcoded.MustAsset("../_hardcoded/doer.go")) if err := doerGenerator.WriteFile("client/doer.go"); err != nil { return fmt.Errorf("Failed to copy doer.go: %s", err) @@ -322,15 +318,8 @@ func (c *config) setClientLanguage(clientLanguage, jsModulePath string) error { // setGeneratedFilePaths determines where to output the generated files. func (c *config) setGeneratedFilePaths() { const serverDir = "server" - const tracingDir = "tracing" - // determine paths for generated files c.goAbsolutePackagePath = filepath.Join(os.Getenv("GOPATH"), "src", c.goPackagePath) - c.modelsPath = filepath.Join(c.goAbsolutePackagePath, "models") - c.serverPath = filepath.Join(c.goAbsolutePackagePath, serverDir) - c.tracingPath = filepath.Join(c.goAbsolutePackagePath, tracingDir) - c.goClientPath = filepath.Join(c.goAbsolutePackagePath, "client") - if c.generateDynamo { // set path of generated dynamo code if none specified if swag.StringValue(c.relativeDynamoPath) == "" { diff --git a/main_test.go b/main_test.go index 99da15e3..6394aebd 100644 --- a/main_test.go +++ b/main_test.go @@ -217,10 +217,6 @@ func Test_config_validate(t *testing.T) { t.Errorf("config.validate() error = %v, wantErr %v", err, tt.wantErr) } else if !tt.wantErr { // clear paths so they are not diffed - tt.input.modelsPath = "" - tt.input.serverPath = "" - tt.input.tracingPath = "" - tt.input.goClientPath = "" tt.input.jsClientPath = "" tt.input.dynamoPath = "" tt.input.goAbsolutePackagePath = "" diff --git a/models/genmodels.go b/models/genmodels.go index b7232f68..0ff401eb 100644 --- a/models/genmodels.go +++ b/models/genmodels.go @@ -18,7 +18,7 @@ import ( ) // Generate writes the files to the client directories -func Generate(packagePath string, s spec.Swagger) error { +func Generate(basePath string, s spec.Swagger) error { tmpFile, err := swagger.WriteToFile(&s) if err != nil { @@ -29,7 +29,7 @@ func Generate(packagePath string, s spec.Swagger) error { genopts := generator.GenOpts{ Spec: tmpFile, ModelPackage: "models", - Target: fmt.Sprintf("%s/src/%s/", os.Getenv("GOPATH"), packagePath), + Target: basePath, IncludeModel: true, IncludeHandler: false, IncludeSupport: false, @@ -45,18 +45,18 @@ func Generate(packagePath string, s spec.Swagger) error { return fmt.Errorf("error generating go-swagger models: %s", err) } - if err := generateOutputs(packagePath, s); err != nil { + if err := generateOutputs(basePath, s); err != nil { return fmt.Errorf("error generating outputs: %s", err) } - if err := generateInputs(packagePath, s); err != nil { + if err := generateInputs(basePath, s); err != nil { return fmt.Errorf("error generating inputs: %s", err) } return nil } -func generateInputs(packagePath string, s spec.Swagger) error { +func generateInputs(basePath string, s spec.Swagger) error { - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(` package models @@ -305,8 +305,8 @@ var queryParamStr = ` {{end}} ` -func generateOutputs(packagePath string, s spec.Swagger) error { - g := swagger.Generator{PackagePath: packagePath} +func generateOutputs(basePath string, s spec.Swagger) error { + g := swagger.Generator{BasePath: basePath} g.Printf("package models\n\n") diff --git a/server/gendb/gendb.go b/server/gendb/gendb.go index dfbed814..fd73ebe1 100644 --- a/server/gendb/gendb.go +++ b/server/gendb/gendb.go @@ -138,7 +138,7 @@ func findCompositeAttribute(config XDBConfig, attributeName string) *CompositeAt } // GenerateDB generates DB code for schemas annotated with the x-db extension. -func GenerateDB(packageName, packagePath string, s *spec.Swagger, outputPath string) error { +func GenerateDB(packageName, basePath string, s *spec.Swagger, outputPath string) error { var schemaNames []string for schemaName := range s.Definitions { schemaNames = append(schemaNames, schemaName) @@ -175,7 +175,7 @@ func GenerateDB(packageName, packagePath string, s *spec.Swagger, outputPath str return err } - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(tmpBuf.String()) return g.WriteFile(outputFilename) } diff --git a/server/genserver.go b/server/genserver.go index c6e7d4c5..b8b1a9e1 100644 --- a/server/genserver.go +++ b/server/genserver.go @@ -14,15 +14,15 @@ import ( ) // Generate server package for a swagger spec. -func Generate(packageName, packagePath string, s spec.Swagger) error { +func Generate(packageName, basePath string, s spec.Swagger) error { - if err := generateRouter(packageName, packagePath, s, s.Paths); err != nil { + if err := generateRouter(packageName, basePath, s, s.Paths); err != nil { return err } - if err := generateInterface(packageName, packagePath, &s, s.Info.InfoProps.Title, s.Paths); err != nil { + if err := generateInterface(packageName, basePath, &s, s.Info.InfoProps.Title, s.Paths); err != nil { return err } - if err := generateHandlers(packageName, packagePath, &s, s.Paths); err != nil { + if err := generateHandlers(packageName, basePath, &s, s.Paths); err != nil { return err } return nil @@ -41,7 +41,7 @@ type routerTemplate struct { Functions []routerFunction } -func generateRouter(packageName, packagePath string, s spec.Swagger, paths *spec.Paths) error { +func generateRouter(packageName, basePath string, s spec.Swagger, paths *spec.Paths) error { var template routerTemplate template.Title = s.Info.Title @@ -82,7 +82,7 @@ func generateRouter(packageName, packagePath string, s spec.Swagger, paths *spec if err != nil { return err } - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(routerCode) return g.WriteFile("server/router.go") } @@ -115,7 +115,7 @@ type Controller interface { } ` -func generateInterface(packageName, packagePath string, s *spec.Swagger, serviceName string, paths *spec.Paths) error { +func generateInterface(packageName, basePath string, s *spec.Swagger, serviceName string, paths *spec.Paths) error { tmpl := interfaceFileTemplate{ ImportStatements: swagger.ImportStatements([]string{"context", packageName + "/models"}), @@ -141,7 +141,7 @@ func generateInterface(packageName, packagePath string, s *spec.Swagger, service if err != nil { return err } - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(interfaceCode) return g.WriteFile("server/interface.go") } @@ -186,7 +186,7 @@ func jsonMarshalNoError(i interface{}) string { {{end}} ` -func generateHandlers(packageName, packagePath string, s *spec.Swagger, paths *spec.Paths) error { +func generateHandlers(packageName, basePath string, s *spec.Swagger, paths *spec.Paths) error { tmpl := handlerFileTemplate{ ImportStatements: swagger.ImportStatements([]string{"context", "github.com/gorilla/mux", "gopkg.in/Clever/kayvee-go.v6/logger", @@ -215,7 +215,7 @@ func generateHandlers(packageName, packagePath string, s *spec.Swagger, paths *s if err != nil { return err } - g := swagger.Generator{PackagePath: packagePath} + g := swagger.Generator{BasePath: basePath} g.Printf(handlerCode) return g.WriteFile("server/handlers.go") } diff --git a/swagger/swagger.go b/swagger/swagger.go index cb6669e4..ed4053c1 100644 --- a/swagger/swagger.go +++ b/swagger/swagger.go @@ -15,8 +15,8 @@ import ( // Generator handles common code generation operations when generating a file in a Go package. type Generator struct { - PackagePath string - buf bytes.Buffer + BasePath string + buf bytes.Buffer } // Printf writes a formatted string to the buffer. @@ -30,7 +30,7 @@ func (g *Generator) Write(p []byte) (n int, err error) { } // WriteFile writes the buffer to a gofmt-ed file. -// The file will be located at $GOPATH/src/{PackagePath}/{path}. +// The file will be located at {(Generator).BasePath}/{path}. func (g *Generator) WriteFile(path string) error { if len(path) == 0 || path[0] == '/' { return fmt.Errorf("path must be relative") @@ -52,7 +52,7 @@ func (g *Generator) WriteFile(path string) error { } fileBytes = formattedFileBytes } - absPath := filepath.Join(os.Getenv("GOPATH"), "src", g.PackagePath, path) + absPath := filepath.Join(g.BasePath, path) dir := filepath.Dir(absPath) if err := os.MkdirAll(dir, 0755); err != nil { return err From 4f9c25e0982e1c57c31eb078dd04f0843ad2e07f Mon Sep 17 00:00:00 2001 From: Taylor Sutton Date: Thu, 21 Apr 2022 14:07:14 -0700 Subject: [PATCH 2/2] BREAKING: Remove GOPATH entirely in go.mod mode. This is a breaking change: now the -output-path, if it's relative, is interpreted relative to current working directory. Previously, we would combine GOPATH with the module path in go.mod to guesstimate where to interpret it relative to. --- main.go | 25 +++++++------------------ main_test.go | 8 -------- 2 files changed, 7 insertions(+), 26 deletions(-) diff --git a/main.go b/main.go index eb94e47d..a46c6916 100644 --- a/main.go +++ b/main.go @@ -38,7 +38,6 @@ type config struct { goPackageName *string dynamoPath string - goPackagePath string goAbsolutePackagePath string jsClientPath string @@ -269,20 +268,20 @@ func (c *config) setGoPaths(outputPath, goPackageName string) error { return fmt.Errorf("go-package is required") } // if the repo does not use modules, the package name is equivalent to the package path - c.goPackagePath = goPackageName + c.goAbsolutePackagePath = filepath.Join(os.Getenv("GOPATH"), "src", goPackageName) } else { defer modFile.Close() - // TODO: do not rely on GOPATH when repo uses modules - goPath := os.Getenv("GOPATH") - if goPath == "" { - return fmt.Errorf("GOPATH must be set") - } if outputPath == "" { return fmt.Errorf("output-path is required") } - c.goPackagePath = getModulePackagePath(goPath, path.Clean(outputPath)) + absolutePath, err := filepath.Abs(outputPath) + if err != nil { + return fmt.Errorf("converting output-path to absolute path: %v", err) + } + c.goAbsolutePackagePath = absolutePath *c.goPackageName = getModulePackageName(modFile, path.Clean(outputPath)) + } return nil } @@ -319,7 +318,6 @@ func (c *config) setClientLanguage(clientLanguage, jsModulePath string) error { func (c *config) setGeneratedFilePaths() { const serverDir = "server" - c.goAbsolutePackagePath = filepath.Join(os.Getenv("GOPATH"), "src", c.goPackagePath) if c.generateDynamo { // set path of generated dynamo code if none specified if swag.StringValue(c.relativeDynamoPath) == "" { @@ -333,15 +331,6 @@ func (c *config) setGeneratedFilePaths() { } } -func getModulePackagePath(goPath, outputPath string) string { - pwd, err := os.Getwd() - if err != nil { - log.Fatalf("Error getting current directory: %s", err.Error()) - } - goSrcPath := fmt.Sprintf("%v%v", goPath, "/src/") - return path.Join(strings.TrimPrefix(pwd, goSrcPath), outputPath) -} - // getModulePackageName gets the package name of the generated code // Example: if packagePath = github.com/Clever/wag/v8/gen-go and the module name is github.com/Clever/wag/v8/v2 // the function will return github.com/Clever/wag/v8/v2/gen-go diff --git a/main_test.go b/main_test.go index 6394aebd..902cc199 100644 --- a/main_test.go +++ b/main_test.go @@ -25,7 +25,6 @@ func Test_config_validate(t *testing.T) { outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), jsModulePath: swag.String("jsModulePath"), - goPackagePath: "github.com/Clever/wag/output-path", relativeDynamoPath: swag.String("server/db"), generateDynamo: true, generateServer: true, @@ -48,7 +47,6 @@ func Test_config_validate(t *testing.T) { outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), jsModulePath: swag.String("jsModulePath"), - goPackagePath: "github.com/Clever/wag/output-path", generateServer: false, generateDynamo: false, generateTracing: true, @@ -70,7 +68,6 @@ func Test_config_validate(t *testing.T) { clientLanguage: swag.String("go"), outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), - goPackagePath: "github.com/Clever/wag/output-path", generateServer: false, generateDynamo: false, generateTracing: true, @@ -94,7 +91,6 @@ func Test_config_validate(t *testing.T) { outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), jsModulePath: swag.String("jsModulePath"), - goPackagePath: "github.com/Clever/wag/output-path", generateServer: false, generateDynamo: false, generateTracing: false, @@ -117,7 +113,6 @@ func Test_config_validate(t *testing.T) { goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), jsModulePath: swag.String("jsModulePath"), relativeDynamoPath: swag.String("server/db"), - goPackagePath: "github.com/Clever/wag/output-path", generateServer: true, generateDynamo: true, generateTracing: true, @@ -138,7 +133,6 @@ func Test_config_validate(t *testing.T) { outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), relativeDynamoPath: swag.String("server/db"), - goPackagePath: "github.com/Clever/wag/output-path", generateServer: true, generateTracing: true, generateGoClient: true, @@ -177,7 +171,6 @@ func Test_config_validate(t *testing.T) { output: config{ outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), - goPackagePath: "github.com/Clever/wag/output-path", relativeDynamoPath: swag.String("gen-db/db"), dynamoOnly: swag.Bool(true), generateServer: false, @@ -198,7 +191,6 @@ func Test_config_validate(t *testing.T) { output: config{ outputPath: swag.String("output-path"), goPackageName: swag.String("github.com/Clever/wag/v8/output-path"), - goPackagePath: "github.com/Clever/wag/output-path", relativeDynamoPath: swag.String("db"), dynamoOnly: swag.Bool(true), generateServer: false,