2016-11-04 21:45:25 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
2016-11-07 08:45:02 +01:00
|
|
|
"log"
|
2016-11-04 21:45:25 +01:00
|
|
|
"os"
|
2016-11-07 08:45:02 +01:00
|
|
|
"strings"
|
2016-11-04 21:45:25 +01:00
|
|
|
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
"github.com/golang/protobuf/protoc-gen-go/generator"
|
2017-01-17 11:48:50 +01:00
|
|
|
"github.com/golang/protobuf/protoc-gen-go/plugin"
|
2017-05-18 19:44:05 +02:00
|
|
|
ggdescriptor "github.com/grpc-ecosystem/grpc-gateway/protoc-gen-grpc-gateway/descriptor"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
registry *ggdescriptor.Registry // some helpers need access to registry
|
2016-11-04 21:45:25 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
g := generator.New()
|
|
|
|
|
|
|
|
data, err := ioutil.ReadAll(os.Stdin)
|
|
|
|
if err != nil {
|
|
|
|
g.Error(err, "reading input")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := proto.Unmarshal(data, g.Request); err != nil {
|
|
|
|
g.Error(err, "parsing input proto")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(g.Request.FileToGenerate) == 0 {
|
|
|
|
g.Fail("no files to generate")
|
|
|
|
}
|
|
|
|
|
|
|
|
g.CommandLineParameters(g.Request.GetParameter())
|
|
|
|
|
2016-11-07 08:45:02 +01:00
|
|
|
// Parse parameters
|
2017-05-18 19:44:05 +02:00
|
|
|
var (
|
|
|
|
templateDir = "./templates"
|
|
|
|
destinationDir = "."
|
|
|
|
debug = false
|
|
|
|
all = false
|
|
|
|
singlePackageMode = false
|
|
|
|
)
|
2016-11-07 08:45:02 +01:00
|
|
|
if parameter := g.Request.GetParameter(); parameter != "" {
|
|
|
|
for _, param := range strings.Split(parameter, ",") {
|
|
|
|
parts := strings.Split(param, "=")
|
|
|
|
if len(parts) != 2 {
|
|
|
|
log.Printf("Err: invalid parameter: %q", param)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
switch parts[0] {
|
|
|
|
case "template_dir":
|
|
|
|
templateDir = parts[1]
|
|
|
|
break
|
2016-12-15 16:25:44 +01:00
|
|
|
case "destination_dir":
|
|
|
|
destinationDir = parts[1]
|
|
|
|
break
|
2017-05-18 19:44:05 +02:00
|
|
|
case "single-package-mode":
|
|
|
|
switch strings.ToLower(parts[1]) {
|
|
|
|
case "true", "t":
|
|
|
|
singlePackageMode = true
|
|
|
|
case "false", "f":
|
|
|
|
default:
|
|
|
|
log.Printf("Err: invalid value for single-package-mode: %q", parts[1])
|
|
|
|
}
|
|
|
|
break
|
2016-11-07 08:45:02 +01:00
|
|
|
case "debug":
|
2016-12-14 11:08:51 +01:00
|
|
|
switch strings.ToLower(parts[1]) {
|
|
|
|
case "true", "t":
|
2016-11-07 08:45:02 +01:00
|
|
|
debug = true
|
2016-12-14 11:08:51 +01:00
|
|
|
case "false", "f":
|
|
|
|
default:
|
2016-11-07 08:45:02 +01:00
|
|
|
log.Printf("Err: invalid value for debug: %q", parts[1])
|
|
|
|
}
|
|
|
|
break
|
2017-01-12 14:46:43 +01:00
|
|
|
case "all":
|
|
|
|
switch strings.ToLower(parts[1]) {
|
|
|
|
case "true", "t":
|
|
|
|
all = true
|
|
|
|
case "false", "f":
|
|
|
|
default:
|
|
|
|
log.Printf("Err: invalid value for debug: %q", parts[1])
|
|
|
|
}
|
|
|
|
break
|
2016-11-07 08:45:02 +01:00
|
|
|
default:
|
|
|
|
log.Printf("Err: unknown parameter: %q", param)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-17 11:48:50 +01:00
|
|
|
tmplMap := make(map[string]*plugin_go.CodeGeneratorResponse_File)
|
|
|
|
concatOrAppend := func(file *plugin_go.CodeGeneratorResponse_File) {
|
2017-05-18 19:44:05 +02:00
|
|
|
if val, ok := tmplMap[file.GetName()]; ok {
|
|
|
|
*val.Content += file.GetContent()
|
2017-01-17 11:48:50 +01:00
|
|
|
} else {
|
2017-05-18 19:44:05 +02:00
|
|
|
tmplMap[file.GetName()] = file
|
2017-01-17 11:48:50 +01:00
|
|
|
g.Response.File = append(g.Response.File, file)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-18 19:44:05 +02:00
|
|
|
if singlePackageMode {
|
|
|
|
registry = ggdescriptor.NewRegistry()
|
|
|
|
if err := registry.Load(g.Request); err != nil {
|
|
|
|
g.Error(err, "registry: failed to load the request")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-07 08:45:02 +01:00
|
|
|
// Generate the encoders
|
2016-11-04 21:45:25 +01:00
|
|
|
for _, file := range g.Request.GetProtoFile() {
|
2017-01-12 14:46:43 +01:00
|
|
|
if all {
|
2017-05-18 19:44:05 +02:00
|
|
|
if singlePackageMode {
|
|
|
|
if _, err := registry.LookupFile(file.GetName()); err != nil {
|
|
|
|
g.Error(err, "registry: failed to lookup file %q", file.GetName())
|
|
|
|
}
|
|
|
|
}
|
2017-01-12 14:46:43 +01:00
|
|
|
encoder := NewGenericTemplateBasedEncoder(templateDir, file, debug, destinationDir)
|
2017-01-17 11:48:50 +01:00
|
|
|
for _, tmpl := range encoder.Files() {
|
|
|
|
concatOrAppend(tmpl)
|
|
|
|
}
|
|
|
|
|
2017-01-12 14:46:43 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2016-11-04 21:45:25 +01:00
|
|
|
for _, service := range file.GetService() {
|
2017-01-12 14:46:43 +01:00
|
|
|
encoder := NewGenericServiceTemplateBasedEncoder(templateDir, service, file, debug, destinationDir)
|
2017-01-17 11:48:50 +01:00
|
|
|
for _, tmpl := range encoder.Files() {
|
|
|
|
concatOrAppend(tmpl)
|
|
|
|
}
|
2016-11-04 21:45:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the protobufs
|
|
|
|
g.GenerateAllFiles()
|
|
|
|
|
|
|
|
data, err = proto.Marshal(g.Response)
|
|
|
|
if err != nil {
|
|
|
|
g.Error(err, "failed to marshal output proto")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = os.Stdout.Write(data)
|
|
|
|
if err != nil {
|
|
|
|
g.Error(err, "failed to write output proto")
|
|
|
|
}
|
|
|
|
}
|