2018-08-15 18:24:18 +00:00
|
|
|
// Copyright 2018 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package protogen
|
|
|
|
|
|
|
|
import (
|
2018-09-11 20:53:14 +00:00
|
|
|
"flag"
|
2018-09-06 17:23:53 +00:00
|
|
|
"fmt"
|
2018-08-15 18:24:18 +00:00
|
|
|
"testing"
|
|
|
|
|
2019-03-13 04:28:30 +00:00
|
|
|
"github.com/google/go-cmp/cmp"
|
2020-02-25 20:51:10 +00:00
|
|
|
|
2019-07-10 23:17:16 +00:00
|
|
|
"google.golang.org/protobuf/proto"
|
2019-05-14 06:55:40 +00:00
|
|
|
"google.golang.org/protobuf/reflect/protoreflect"
|
2018-11-27 02:55:29 +00:00
|
|
|
|
2019-05-16 19:47:20 +00:00
|
|
|
"google.golang.org/protobuf/types/descriptorpb"
|
|
|
|
"google.golang.org/protobuf/types/pluginpb"
|
2018-08-15 18:24:18 +00:00
|
|
|
)
|
|
|
|
|
2020-03-06 21:58:41 +00:00
|
|
|
func init() {
|
|
|
|
warnings = false // avoid spam in tests
|
|
|
|
}
|
|
|
|
|
2018-09-11 20:53:14 +00:00
|
|
|
func TestPluginParameters(t *testing.T) {
|
|
|
|
var flags flag.FlagSet
|
|
|
|
value := flags.Int("integer", 0, "")
|
|
|
|
const params = "integer=2"
|
2020-02-27 22:47:29 +00:00
|
|
|
_, err := Options{
|
|
|
|
ParamFunc: flags.Set,
|
|
|
|
}.New(&pluginpb.CodeGeneratorRequest{
|
2019-07-10 23:17:16 +00:00
|
|
|
Parameter: proto.String(params),
|
2020-02-27 22:47:29 +00:00
|
|
|
})
|
2018-09-11 20:53:14 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("New(generator parameters %q): %v", params, err)
|
|
|
|
}
|
|
|
|
if *value != 2 {
|
|
|
|
t.Errorf("New(generator parameters %q): integer=%v, want 2", params, *value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPluginParameterErrors(t *testing.T) {
|
|
|
|
for _, parameter := range []string{
|
|
|
|
"unknown=1",
|
|
|
|
"boolean=error",
|
|
|
|
} {
|
|
|
|
var flags flag.FlagSet
|
|
|
|
flags.Bool("boolean", false, "")
|
2020-02-27 22:47:29 +00:00
|
|
|
_, err := Options{
|
2018-09-11 20:53:14 +00:00
|
|
|
ParamFunc: flags.Set,
|
2020-02-27 22:47:29 +00:00
|
|
|
}.New(&pluginpb.CodeGeneratorRequest{
|
2019-07-10 23:17:16 +00:00
|
|
|
Parameter: proto.String(parameter),
|
2020-02-27 22:47:29 +00:00
|
|
|
})
|
2018-09-11 20:53:14 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("New(generator parameters %q): want error, got nil", parameter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-13 04:28:30 +00:00
|
|
|
func TestNoGoPackage(t *testing.T) {
|
2020-02-27 22:47:29 +00:00
|
|
|
gen, err := Options{}.New(&pluginpb.CodeGeneratorRequest{
|
2019-03-13 04:28:30 +00:00
|
|
|
ProtoFile: []*descriptorpb.FileDescriptorProto{
|
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("testdata/go_package/no_go_package.proto"),
|
|
|
|
Syntax: proto.String(protoreflect.Proto3.String()),
|
|
|
|
Package: proto.String("goproto.testdata"),
|
2019-03-13 04:28:30 +00:00
|
|
|
},
|
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("testdata/go_package/no_go_package_import.proto"),
|
|
|
|
Syntax: proto.String(protoreflect.Proto3.String()),
|
|
|
|
Package: proto.String("goproto.testdata"),
|
2020-02-10 22:04:25 +00:00
|
|
|
Dependency: []string{"testdata/go_package/no_go_package.proto"},
|
2019-03-13 04:28:30 +00:00
|
|
|
},
|
|
|
|
},
|
2020-02-27 22:47:29 +00:00
|
|
|
})
|
2018-08-15 18:24:18 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-03-13 04:28:30 +00:00
|
|
|
|
|
|
|
for i, f := range gen.Files {
|
|
|
|
if got, want := string(f.GoPackageName), "goproto_testdata"; got != want {
|
|
|
|
t.Errorf("gen.Files[%d].GoPackageName = %v, want %v", i, got, want)
|
2018-08-15 18:24:18 +00:00
|
|
|
}
|
2019-03-13 04:28:30 +00:00
|
|
|
if got, want := string(f.GoImportPath), "testdata/go_package"; got != want {
|
|
|
|
t.Errorf("gen.Files[%d].GoImportPath = %v, want %v", i, got, want)
|
2018-08-15 18:24:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-06 17:23:53 +00:00
|
|
|
func TestPackageNamesAndPaths(t *testing.T) {
|
|
|
|
const (
|
|
|
|
filename = "dir/filename.proto"
|
|
|
|
protoPackageName = "proto.package"
|
|
|
|
)
|
|
|
|
for _, test := range []struct {
|
2020-02-13 07:38:30 +00:00
|
|
|
desc string
|
|
|
|
parameter string
|
|
|
|
goPackageOption string
|
|
|
|
generate bool
|
|
|
|
wantPackageName GoPackageName
|
|
|
|
wantImportPath GoImportPath
|
|
|
|
wantFilename string
|
2018-09-06 17:23:53 +00:00
|
|
|
}{
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "no parameters, no go_package option",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "proto_package",
|
|
|
|
wantImportPath: "dir",
|
|
|
|
wantFilename: "dir/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "go_package option sets import path",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/foo",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "go_package option sets import path and package",
|
|
|
|
goPackageOption: "golang.org/x/foo;bar",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "bar",
|
|
|
|
wantImportPath: "golang.org/x/foo",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "go_package option sets package",
|
|
|
|
goPackageOption: "foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "dir",
|
|
|
|
wantFilename: "dir/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "command line sets import path for a file",
|
|
|
|
parameter: "Mdir/filename.proto=golang.org/x/bar",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/bar",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
2020-03-18 07:59:09 +00:00
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "command line sets import path for a file with package name specified",
|
|
|
|
parameter: "Mdir/filename.proto=golang.org/x/bar;bar",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "bar",
|
|
|
|
wantImportPath: "golang.org/x/bar",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2020-03-18 07:59:09 +00:00
|
|
|
},
|
2018-09-06 17:23:53 +00:00
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "import_path parameter sets import path of generated files",
|
|
|
|
parameter: "import_path=golang.org/x/bar",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/bar",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "import_path parameter does not set import path of dependencies",
|
|
|
|
parameter: "import_path=golang.org/x/bar",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: false,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/foo",
|
|
|
|
wantFilename: "golang.org/x/foo/filename",
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
2020-02-15 22:28:51 +00:00
|
|
|
{
|
2020-02-13 07:38:30 +00:00
|
|
|
desc: "module option set",
|
|
|
|
parameter: "module=golang.org/x",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: false,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/foo",
|
|
|
|
wantFilename: "foo/filename",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "paths=import uses import path from command line",
|
|
|
|
parameter: "paths=import,Mdir/filename.proto=golang.org/x/bar",
|
|
|
|
goPackageOption: "golang.org/x/foo",
|
|
|
|
generate: true,
|
|
|
|
wantPackageName: "foo",
|
|
|
|
wantImportPath: "golang.org/x/bar",
|
|
|
|
wantFilename: "golang.org/x/bar/filename",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "module option implies paths=import",
|
|
|
|
parameter: "module=golang.org/x,Mdir/filename.proto=golang.org/x/foo",
|
|
|
|
generate: false,
|
|
|
|
wantPackageName: "proto_package",
|
|
|
|
wantImportPath: "golang.org/x/foo",
|
|
|
|
wantFilename: "foo/filename",
|
2020-02-15 22:28:51 +00:00
|
|
|
},
|
2018-09-06 17:23:53 +00:00
|
|
|
} {
|
|
|
|
context := fmt.Sprintf(`
|
|
|
|
TEST: %v
|
|
|
|
--go_out=%v:.
|
|
|
|
file %q: generate=%v
|
|
|
|
option go_package = %q;
|
|
|
|
|
|
|
|
`,
|
|
|
|
test.desc, test.parameter, filename, test.generate, test.goPackageOption)
|
|
|
|
|
|
|
|
req := &pluginpb.CodeGeneratorRequest{
|
2019-07-10 23:17:16 +00:00
|
|
|
Parameter: proto.String(test.parameter),
|
2018-11-27 02:55:29 +00:00
|
|
|
ProtoFile: []*descriptorpb.FileDescriptorProto{
|
2018-09-06 17:23:53 +00:00
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String(filename),
|
|
|
|
Package: proto.String(protoPackageName),
|
2018-11-27 02:55:29 +00:00
|
|
|
Options: &descriptorpb.FileOptions{
|
2019-07-10 23:17:16 +00:00
|
|
|
GoPackage: proto.String(test.goPackageOption),
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if test.generate {
|
|
|
|
req.FileToGenerate = []string{filename}
|
|
|
|
}
|
2020-02-27 22:47:29 +00:00
|
|
|
gen, err := Options{}.New(req)
|
2018-09-06 17:23:53 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%vNew(req) = %v", context, err)
|
|
|
|
continue
|
|
|
|
}
|
2019-08-21 03:14:19 +00:00
|
|
|
gotFile, ok := gen.FilesByPath[filename]
|
2018-09-06 17:23:53 +00:00
|
|
|
if !ok {
|
|
|
|
t.Errorf("%v%v: missing file info", context, filename)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if got, want := gotFile.GoPackageName, test.wantPackageName; got != want {
|
|
|
|
t.Errorf("%vGoPackageName=%v, want %v", context, got, want)
|
|
|
|
}
|
|
|
|
if got, want := gotFile.GoImportPath, test.wantImportPath; got != want {
|
|
|
|
t.Errorf("%vGoImportPath=%v, want %v", context, got, want)
|
|
|
|
}
|
2020-02-13 07:38:30 +00:00
|
|
|
gen.NewGeneratedFile(gotFile.GeneratedFilenamePrefix, "")
|
|
|
|
resp := gen.Response()
|
|
|
|
if got, want := resp.File[0].GetName(), test.wantFilename; got != want {
|
|
|
|
t.Errorf("%vgenerated filename=%v, want %v", context, got, want)
|
2018-09-06 17:23:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPackageNameInference(t *testing.T) {
|
2020-02-27 22:47:29 +00:00
|
|
|
gen, err := Options{}.New(&pluginpb.CodeGeneratorRequest{
|
2018-11-27 02:55:29 +00:00
|
|
|
ProtoFile: []*descriptorpb.FileDescriptorProto{
|
2018-09-06 17:23:53 +00:00
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("dir/file1.proto"),
|
|
|
|
Package: proto.String("proto.package"),
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("dir/file2.proto"),
|
|
|
|
Package: proto.String("proto.package"),
|
2018-11-27 02:55:29 +00:00
|
|
|
Options: &descriptorpb.FileOptions{
|
2019-07-10 23:17:16 +00:00
|
|
|
GoPackage: proto.String("foo"),
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
FileToGenerate: []string{"dir/file1.proto", "dir/file2.proto"},
|
2020-02-27 22:47:29 +00:00
|
|
|
})
|
2018-09-06 17:23:53 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("New(req) = %v", err)
|
|
|
|
}
|
2019-08-21 03:14:19 +00:00
|
|
|
if f1, ok := gen.FilesByPath["dir/file1.proto"]; !ok {
|
2018-09-06 17:23:53 +00:00
|
|
|
t.Errorf("missing file info for dir/file1.proto")
|
|
|
|
} else if f1.GoPackageName != "foo" {
|
|
|
|
t.Errorf("dir/file1.proto: GoPackageName=%v, want foo; package name should be derived from dir/file2.proto", f1.GoPackageName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInconsistentPackageNames(t *testing.T) {
|
2020-02-27 22:47:29 +00:00
|
|
|
_, err := Options{}.New(&pluginpb.CodeGeneratorRequest{
|
2018-11-27 02:55:29 +00:00
|
|
|
ProtoFile: []*descriptorpb.FileDescriptorProto{
|
2018-09-06 17:23:53 +00:00
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("dir/file1.proto"),
|
|
|
|
Package: proto.String("proto.package"),
|
2018-11-27 02:55:29 +00:00
|
|
|
Options: &descriptorpb.FileOptions{
|
2019-07-10 23:17:16 +00:00
|
|
|
GoPackage: proto.String("golang.org/x/foo"),
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2019-07-10 23:17:16 +00:00
|
|
|
Name: proto.String("dir/file2.proto"),
|
|
|
|
Package: proto.String("proto.package"),
|
2018-11-27 02:55:29 +00:00
|
|
|
Options: &descriptorpb.FileOptions{
|
2019-07-10 23:17:16 +00:00
|
|
|
GoPackage: proto.String("golang.org/x/foo;bar"),
|
2018-09-06 17:23:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
FileToGenerate: []string{"dir/file1.proto", "dir/file2.proto"},
|
2020-02-27 22:47:29 +00:00
|
|
|
})
|
2018-09-06 17:23:53 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("inconsistent package names for the same import path: New(req) = nil, want error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-23 21:39:30 +00:00
|
|
|
func TestImports(t *testing.T) {
|
2020-02-27 22:47:29 +00:00
|
|
|
gen, err := Options{}.New(&pluginpb.CodeGeneratorRequest{})
|
2018-08-23 21:39:30 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
g := gen.NewGeneratedFile("foo.go", "golang.org/x/foo")
|
|
|
|
g.P("package foo")
|
|
|
|
g.P()
|
|
|
|
for _, importPath := range []GoImportPath{
|
|
|
|
"golang.org/x/foo",
|
|
|
|
// Multiple references to the same package.
|
|
|
|
"golang.org/x/bar",
|
|
|
|
"golang.org/x/bar",
|
|
|
|
// Reference to a different package with the same basename.
|
|
|
|
"golang.org/y/bar",
|
|
|
|
"golang.org/x/baz",
|
2018-10-05 18:23:35 +00:00
|
|
|
// Reference to a package conflicting with a predeclared identifier.
|
|
|
|
"golang.org/z/string",
|
2018-08-23 21:39:30 +00:00
|
|
|
} {
|
|
|
|
g.P("var _ = ", GoIdent{GoName: "X", GoImportPath: importPath}, " // ", importPath)
|
|
|
|
}
|
|
|
|
want := `package foo
|
|
|
|
|
|
|
|
import (
|
|
|
|
bar "golang.org/x/bar"
|
|
|
|
baz "golang.org/x/baz"
|
2018-09-13 20:12:36 +00:00
|
|
|
bar1 "golang.org/y/bar"
|
2018-10-05 18:23:35 +00:00
|
|
|
string1 "golang.org/z/string"
|
2018-08-23 21:39:30 +00:00
|
|
|
)
|
|
|
|
|
2018-10-05 18:23:35 +00:00
|
|
|
var _ = X // "golang.org/x/foo"
|
|
|
|
var _ = bar.X // "golang.org/x/bar"
|
|
|
|
var _ = bar.X // "golang.org/x/bar"
|
|
|
|
var _ = bar1.X // "golang.org/y/bar"
|
|
|
|
var _ = baz.X // "golang.org/x/baz"
|
|
|
|
var _ = string1.X // "golang.org/z/string"
|
2018-08-23 21:39:30 +00:00
|
|
|
`
|
2018-12-21 23:54:06 +00:00
|
|
|
got, err := g.Content()
|
2018-08-23 21:39:30 +00:00
|
|
|
if err != nil {
|
2018-12-21 23:54:06 +00:00
|
|
|
t.Fatalf("g.Content() = %v", err)
|
2018-08-23 21:39:30 +00:00
|
|
|
}
|
2019-03-13 04:28:30 +00:00
|
|
|
if diff := cmp.Diff(string(want), string(got)); diff != "" {
|
|
|
|
t.Fatalf("content mismatch (-want +got):\n%s", diff)
|
2018-08-23 21:39:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-27 22:51:05 +00:00
|
|
|
func TestImportRewrites(t *testing.T) {
|
2020-02-27 22:47:29 +00:00
|
|
|
gen, err := Options{
|
2018-09-27 22:51:05 +00:00
|
|
|
ImportRewriteFunc: func(i GoImportPath) GoImportPath {
|
|
|
|
return "prefix/" + i
|
|
|
|
},
|
2020-02-27 22:47:29 +00:00
|
|
|
}.New(&pluginpb.CodeGeneratorRequest{})
|
2018-09-27 22:51:05 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
g := gen.NewGeneratedFile("foo.go", "golang.org/x/foo")
|
|
|
|
g.P("package foo")
|
|
|
|
g.P("var _ = ", GoIdent{GoName: "X", GoImportPath: "golang.org/x/bar"})
|
|
|
|
want := `package foo
|
|
|
|
|
2019-03-13 04:28:30 +00:00
|
|
|
import (
|
|
|
|
bar "prefix/golang.org/x/bar"
|
|
|
|
)
|
2018-09-27 22:51:05 +00:00
|
|
|
|
|
|
|
var _ = bar.X
|
|
|
|
`
|
2018-12-21 23:54:06 +00:00
|
|
|
got, err := g.Content()
|
2018-09-27 22:51:05 +00:00
|
|
|
if err != nil {
|
2018-12-21 23:54:06 +00:00
|
|
|
t.Fatalf("g.Content() = %v", err)
|
2018-09-27 22:51:05 +00:00
|
|
|
}
|
2019-03-13 04:28:30 +00:00
|
|
|
if diff := cmp.Diff(string(want), string(got)); diff != "" {
|
|
|
|
t.Fatalf("content mismatch (-want +got):\n%s", diff)
|
2018-08-15 18:24:18 +00:00
|
|
|
}
|
|
|
|
}
|