2017-05-19 11:02:40 +03:00
|
|
|
package pgghelpers
|
2016-12-15 15:28:57 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2016-12-23 01:41:17 +03:00
|
|
|
"fmt"
|
2017-04-13 14:15:40 +03:00
|
|
|
"regexp"
|
2016-12-16 18:08:37 +03:00
|
|
|
"strings"
|
2016-12-15 15:28:57 +03:00
|
|
|
"text/template"
|
|
|
|
|
|
|
|
"github.com/Masterminds/sprig"
|
2016-12-23 01:41:17 +03:00
|
|
|
"github.com/golang/protobuf/proto"
|
2016-12-21 12:26:26 +03:00
|
|
|
"github.com/golang/protobuf/protoc-gen-go/descriptor"
|
2017-05-18 20:44:05 +03:00
|
|
|
ggdescriptor "github.com/grpc-ecosystem/grpc-gateway/protoc-gen-grpc-gateway/descriptor"
|
|
|
|
"github.com/huandu/xstrings"
|
2017-03-31 19:01:58 +03:00
|
|
|
options "google.golang.org/genproto/googleapis/api/annotations"
|
2016-12-15 15:28:57 +03:00
|
|
|
)
|
|
|
|
|
2017-04-13 14:15:40 +03:00
|
|
|
var jsReservedRe *regexp.Regexp = regexp.MustCompile(`(^|[^A-Za-z])(do|if|in|for|let|new|try|var|case|else|enum|eval|false|null|this|true|void|with|break|catch|class|const|super|throw|while|yield|delete|export|import|public|return|static|switch|typeof|default|extends|finally|package|private|continue|debugger|function|arguments|interface|protected|implements|instanceof)($|[^A-Za-z])`)
|
|
|
|
|
2017-05-19 11:02:40 +03:00
|
|
|
var (
|
|
|
|
registry *ggdescriptor.Registry // some helpers need access to registry
|
|
|
|
)
|
|
|
|
|
|
|
|
func SetRegistry(reg *ggdescriptor.Registry) {
|
|
|
|
registry = reg
|
|
|
|
}
|
|
|
|
|
2016-12-15 15:28:57 +03:00
|
|
|
var ProtoHelpersFuncMap = template.FuncMap{
|
|
|
|
"string": func(i interface {
|
|
|
|
String() string
|
|
|
|
}) string {
|
|
|
|
return i.String()
|
|
|
|
},
|
|
|
|
"json": func(v interface{}) string {
|
|
|
|
a, _ := json.Marshal(v)
|
|
|
|
return string(a)
|
|
|
|
},
|
|
|
|
"prettyjson": func(v interface{}) string {
|
|
|
|
a, _ := json.MarshalIndent(v, "", " ")
|
|
|
|
return string(a)
|
|
|
|
},
|
2016-12-16 18:08:37 +03:00
|
|
|
"splitArray": func(sep string, s string) []string {
|
|
|
|
return strings.Split(s, sep)
|
|
|
|
},
|
|
|
|
"first": func(a []string) string {
|
2016-12-15 15:28:57 +03:00
|
|
|
return a[0]
|
|
|
|
},
|
2016-12-16 18:08:37 +03:00
|
|
|
"last": func(a []string) string {
|
2016-12-15 15:28:57 +03:00
|
|
|
return a[len(a)-1]
|
|
|
|
},
|
2016-12-19 17:43:38 +03:00
|
|
|
"upperFirst": func(s string) string {
|
|
|
|
return strings.ToUpper(s[:1]) + s[1:]
|
|
|
|
},
|
2016-12-16 18:08:37 +03:00
|
|
|
"lowerFirst": func(s string) string {
|
|
|
|
return strings.ToLower(s[:1]) + s[1:]
|
|
|
|
},
|
2016-12-20 13:31:46 +03:00
|
|
|
"camelCase": func(s string) string {
|
2017-01-04 01:41:58 +03:00
|
|
|
if len(s) > 1 {
|
|
|
|
return xstrings.ToCamelCase(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.ToUpper(s[:1])
|
2016-12-20 13:31:46 +03:00
|
|
|
},
|
2017-01-11 18:33:03 +03:00
|
|
|
"lowerCamelCase": func(s string) string {
|
|
|
|
if len(s) > 1 {
|
|
|
|
s = xstrings.ToCamelCase(s)
|
|
|
|
}
|
2017-01-04 01:41:58 +03:00
|
|
|
|
2017-01-11 18:33:03 +03:00
|
|
|
return strings.ToLower(s[:1]) + s[1:]
|
|
|
|
},
|
2016-12-20 13:31:46 +03:00
|
|
|
"kebabCase": func(s string) string {
|
|
|
|
return strings.Replace(xstrings.ToSnakeCase(s), "_", "-", -1)
|
|
|
|
},
|
2017-02-01 16:38:14 +03:00
|
|
|
"snakeCase": xstrings.ToSnakeCase,
|
2017-05-18 20:44:05 +03:00
|
|
|
"getProtoFile": getProtoFile,
|
2017-02-01 16:38:14 +03:00
|
|
|
"getMessageType": getMessageType,
|
2017-05-12 21:46:18 +03:00
|
|
|
"getEnumValue": getEnumValue,
|
2017-02-01 16:38:14 +03:00
|
|
|
"isFieldMessage": isFieldMessage,
|
|
|
|
"isFieldRepeated": isFieldRepeated,
|
|
|
|
"goType": goType,
|
2017-05-02 00:55:55 +03:00
|
|
|
"goTypeWithPackage": goTypeWithPackage,
|
2017-02-01 16:38:14 +03:00
|
|
|
"jsType": jsType,
|
2017-04-13 14:15:40 +03:00
|
|
|
"jsSuffixReserved": jsSuffixReservedKeyword,
|
2017-02-01 16:38:14 +03:00
|
|
|
"namespacedFlowType": namespacedFlowType,
|
|
|
|
"httpVerb": httpVerb,
|
|
|
|
"httpPath": httpPath,
|
|
|
|
"shortType": shortType,
|
|
|
|
"urlHasVarsFromMessage": urlHasVarsFromMessage,
|
2016-12-15 15:28:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
for k, v := range sprig.TxtFuncMap() {
|
|
|
|
ProtoHelpersFuncMap[k] = v
|
|
|
|
}
|
|
|
|
}
|
2016-12-21 12:26:26 +03:00
|
|
|
|
2017-05-18 20:44:05 +03:00
|
|
|
func getProtoFile(name string) *ggdescriptor.File {
|
|
|
|
if registry == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
file, err := registry.LookupFile(name)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
|
|
|
|
func getMessageType(f *descriptor.FileDescriptorProto, name string) *ggdescriptor.Message {
|
|
|
|
if registry != nil {
|
|
|
|
msg, err := registry.LookupMsg(".", name)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return msg
|
|
|
|
}
|
|
|
|
|
2016-12-27 19:56:44 +03:00
|
|
|
// name is in the form .packageName.MessageTypeName.InnerMessageTypeName...
|
|
|
|
// e.g. .article.ProductTag
|
|
|
|
splits := strings.Split(name, ".")
|
|
|
|
target := splits[len(splits)-1]
|
2016-12-21 12:26:26 +03:00
|
|
|
for _, m := range f.MessageType {
|
2016-12-27 19:56:44 +03:00
|
|
|
if target == *m.Name {
|
2017-05-18 20:44:05 +03:00
|
|
|
return &ggdescriptor.Message{
|
|
|
|
DescriptorProto: m,
|
|
|
|
}
|
2016-12-21 12:26:26 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-05-12 21:46:18 +03:00
|
|
|
func getEnumValue(f []*descriptor.EnumDescriptorProto, name string) []*descriptor.EnumValueDescriptorProto {
|
|
|
|
for _, item := range f {
|
|
|
|
if strings.EqualFold(*item.Name, name) {
|
|
|
|
return item.GetValue()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-12-21 12:26:26 +03:00
|
|
|
func isFieldMessage(f *descriptor.FieldDescriptorProto) bool {
|
2016-12-29 19:21:53 +03:00
|
|
|
if f.Type != nil && *f.Type == descriptor.FieldDescriptorProto_TYPE_MESSAGE {
|
2016-12-21 12:26:26 +03:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
|
2016-12-26 14:13:04 +03:00
|
|
|
func isFieldRepeated(f *descriptor.FieldDescriptorProto) bool {
|
|
|
|
if f.Type != nil && f.Label != nil && *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-05-02 00:55:55 +03:00
|
|
|
func goTypeWithPackage(f *descriptor.FieldDescriptorProto) string {
|
2017-05-02 13:23:34 +03:00
|
|
|
pkg := ""
|
|
|
|
if *f.Type == descriptor.FieldDescriptorProto_TYPE_MESSAGE {
|
|
|
|
pkg = getPackageTypeName(*f.TypeName)
|
|
|
|
}
|
2017-05-02 00:55:55 +03:00
|
|
|
return goType(pkg, f)
|
|
|
|
}
|
|
|
|
|
2016-12-23 01:41:17 +03:00
|
|
|
func goType(pkg string, f *descriptor.FieldDescriptorProto) string {
|
|
|
|
switch *f.Type {
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_DOUBLE:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]float64"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "float64"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_FLOAT:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]float32"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "float32"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_INT64:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]int64"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "int64"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_UINT64:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]uint64"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "uint64"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_INT32:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]uint32"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "uint32"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BOOL:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]bool"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "bool"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_STRING:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]string"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "string"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
|
2017-04-06 15:40:41 +03:00
|
|
|
if pkg != "" {
|
|
|
|
pkg = pkg + "."
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
2017-04-06 15:40:41 +03:00
|
|
|
return fmt.Sprintf("[]*%s%s", pkg, shortType(*f.TypeName))
|
2016-12-23 01:41:17 +03:00
|
|
|
}
|
2017-04-06 15:40:41 +03:00
|
|
|
return fmt.Sprintf("*%s%s", pkg, shortType(*f.TypeName))
|
2016-12-23 01:41:17 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BYTES:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]byte"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "byte"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_UINT32:
|
2017-02-07 17:09:35 +03:00
|
|
|
if *f.Label == descriptor.FieldDescriptorProto_LABEL_REPEATED {
|
|
|
|
return "[]uint32"
|
|
|
|
}
|
2016-12-23 01:41:17 +03:00
|
|
|
return "uint32"
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_ENUM:
|
|
|
|
return fmt.Sprintf("*%s.%s", pkg, shortType(*f.TypeName))
|
|
|
|
default:
|
|
|
|
return "interface{}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-04 02:35:21 +03:00
|
|
|
func jsType(f *descriptor.FieldDescriptorProto) string {
|
2017-01-05 20:04:20 +03:00
|
|
|
template := "%s"
|
|
|
|
if isFieldRepeated(f) == true {
|
|
|
|
template = "Array<%s>"
|
|
|
|
}
|
|
|
|
|
2017-01-04 02:35:21 +03:00
|
|
|
switch *f.Type {
|
2017-01-10 13:40:48 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_MESSAGE,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_ENUM:
|
2017-01-10 14:24:06 +03:00
|
|
|
return fmt.Sprintf(template, namespacedFlowType(*f.TypeName))
|
2017-01-05 20:04:20 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_FLOAT,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_INT64,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_UINT64,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_INT32,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_FIXED64,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_FIXED32,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_UINT32,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_SFIXED32,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_SFIXED64,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_SINT32,
|
|
|
|
descriptor.FieldDescriptorProto_TYPE_SINT64:
|
|
|
|
return fmt.Sprintf(template, "number")
|
2017-01-04 02:35:21 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BOOL:
|
2017-01-05 20:04:20 +03:00
|
|
|
return fmt.Sprintf(template, "boolean")
|
2017-01-05 20:32:54 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BYTES:
|
2017-01-11 16:52:00 +03:00
|
|
|
return fmt.Sprintf(template, "Uint8Array")
|
2017-01-04 02:35:21 +03:00
|
|
|
case descriptor.FieldDescriptorProto_TYPE_STRING:
|
2017-01-05 20:04:20 +03:00
|
|
|
return fmt.Sprintf(template, "string")
|
2017-01-04 02:35:21 +03:00
|
|
|
default:
|
2017-01-05 20:04:20 +03:00
|
|
|
return fmt.Sprintf(template, "any")
|
2017-01-04 02:35:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-13 14:15:40 +03:00
|
|
|
func jsSuffixReservedKeyword(s string) string {
|
|
|
|
return jsReservedRe.ReplaceAllString(s, "${1}${2}_${3}")
|
|
|
|
}
|
|
|
|
|
2017-05-02 00:55:55 +03:00
|
|
|
func getPackageTypeName(s string) string {
|
|
|
|
if strings.Contains(s, ".") {
|
|
|
|
return strings.Split(s, ".")[1]
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2016-12-23 01:41:17 +03:00
|
|
|
func shortType(s string) string {
|
|
|
|
t := strings.Split(s, ".")
|
|
|
|
return t[len(t)-1]
|
|
|
|
}
|
|
|
|
|
2017-01-10 14:24:06 +03:00
|
|
|
func namespacedFlowType(s string) string {
|
|
|
|
trimmed := strings.TrimLeft(s, ".")
|
|
|
|
splitted := strings.Split(trimmed, ".")
|
2017-02-11 19:18:26 +03:00
|
|
|
return strings.Join(splitted, "$")
|
2017-01-10 14:24:06 +03:00
|
|
|
}
|
|
|
|
|
2016-12-23 01:41:17 +03:00
|
|
|
func httpPath(m *descriptor.MethodDescriptorProto) string {
|
|
|
|
|
|
|
|
ext, err := proto.GetExtension(m.Options, options.E_Http)
|
|
|
|
if err != nil {
|
|
|
|
return err.Error()
|
|
|
|
}
|
|
|
|
opts, ok := ext.(*options.HttpRule)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Sprintf("extension is %T; want an HttpRule", ext)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch t := opts.Pattern.(type) {
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
case *options.HttpRule_Get:
|
|
|
|
return t.Get
|
|
|
|
case *options.HttpRule_Post:
|
|
|
|
return t.Post
|
|
|
|
case *options.HttpRule_Put:
|
|
|
|
return t.Put
|
|
|
|
case *options.HttpRule_Delete:
|
|
|
|
return t.Delete
|
|
|
|
case *options.HttpRule_Patch:
|
|
|
|
return t.Patch
|
|
|
|
case *options.HttpRule_Custom:
|
|
|
|
return t.Custom.Path
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func httpVerb(m *descriptor.MethodDescriptorProto) string {
|
|
|
|
|
|
|
|
ext, err := proto.GetExtension(m.Options, options.E_Http)
|
|
|
|
if err != nil {
|
|
|
|
return err.Error()
|
|
|
|
}
|
|
|
|
opts, ok := ext.(*options.HttpRule)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Sprintf("extension is %T; want an HttpRule", ext)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch t := opts.Pattern.(type) {
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
case *options.HttpRule_Get:
|
|
|
|
return "GET"
|
|
|
|
case *options.HttpRule_Post:
|
|
|
|
return "POST"
|
|
|
|
case *options.HttpRule_Put:
|
|
|
|
return "PUT"
|
|
|
|
case *options.HttpRule_Delete:
|
|
|
|
return "DELETE"
|
|
|
|
case *options.HttpRule_Patch:
|
|
|
|
return "PATCH"
|
|
|
|
case *options.HttpRule_Custom:
|
|
|
|
return t.Custom.Kind
|
|
|
|
}
|
|
|
|
}
|
2017-02-01 16:38:14 +03:00
|
|
|
|
|
|
|
func urlHasVarsFromMessage(path string, d *descriptor.DescriptorProto) bool {
|
|
|
|
for _, field := range d.Field {
|
|
|
|
if !isFieldMessage(field) {
|
|
|
|
if strings.Contains(path, fmt.Sprintf("{%s}", *field.Name)) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|