blob: c5c8c142408e78e153263dfa66a9b3da30dcaf9a [file] [log] [blame]
Marcel van Lohuizen5274e982019-04-28 17:51:43 +02001// Copyright 2019 CUE Authors
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package protobuf
16
17import (
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020018 "bytes"
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020019 "fmt"
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020020 "os"
21 "path"
22 "path/filepath"
23 "sort"
24 "strconv"
25 "strings"
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020026 "text/scanner"
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020027 "unicode"
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020028
Koichi Shiraishi30a4bee2019-12-26 03:06:48 +090029 "github.com/emicklei/proto"
30
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020031 "cuelang.org/go/cue/ast"
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020032 "cuelang.org/go/cue/errors"
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020033 "cuelang.org/go/cue/parser"
34 "cuelang.org/go/cue/token"
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020035 "cuelang.org/go/internal/source"
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020036)
37
Marcel van Lohuizenf94a5482019-07-02 20:15:03 +020038func (s *Extractor) parse(filename string, src interface{}) (p *protoConverter, err error) {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020039 if filename == "" {
40 return nil, errors.Newf(token.NoPos, "empty filename")
41 }
42 if r, ok := s.fileCache[filename]; ok {
43 return r.p, r.err
44 }
45 defer func() {
46 s.fileCache[filename] = result{p, err}
47 }()
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020048
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020049 b, err := source.Read(filename, src)
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020050 if err != nil {
51 return nil, err
52 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020053
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020054 parser := proto.NewParser(bytes.NewReader(b))
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020055 if filename != "" {
56 parser.Filename(filename)
57 }
58 d, err := parser.Parse()
59 if err != nil {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020060 return nil, errors.Newf(token.NoPos, "protobuf: %v", err)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020061 }
62
Marcel van Lohuizen93e95972019-06-27 16:47:52 +020063 tfile := token.NewFile(filename, 0, len(b))
64 tfile.SetLinesForContent(b)
65
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020066 p = &protoConverter{
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +020067 id: filename,
68 state: s,
69 tfile: tfile,
70 imported: map[string]bool{},
71 symbols: map[string]bool{},
72 aliases: map[string]string{},
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020073 }
74
75 defer func() {
76 switch x := recover().(type) {
77 case nil:
78 case protoError:
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +020079 err = &protobufError{
80 path: p.path,
81 pos: p.toCUEPos(x.pos),
82 err: x.error,
Marcel van Lohuizen5274e982019-04-28 17:51:43 +020083 }
84 default:
85 panic(x)
86 }
87 }()
88
89 p.file = &ast.File{Filename: filename}
90
91 p.addNames(d.Elements)
92
93 // Parse package definitions.
94 for _, e := range d.Elements {
95 switch x := e.(type) {
96 case *proto.Package:
97 p.protoPkg = x.Name
98 case *proto.Option:
99 if x.Name == "go_package" {
100 str, err := strconv.Unquote(x.Constant.SourceRepresentation())
101 if err != nil {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200102 failf(x.Position, "unquoting package filed: %v", err)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200103 }
104 split := strings.Split(str, ";")
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200105 switch {
106 case strings.Contains(split[0], "."):
107 p.cuePkgPath = split[0]
108 switch len(split) {
109 case 1:
110 p.shortPkgName = path.Base(str)
111 case 2:
112 p.shortPkgName = split[1]
113 default:
114 failf(x.Position, "unexpected ';' in %q", str)
115 }
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200116
117 case len(split) == 1:
118 p.shortPkgName = split[0]
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200119
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200120 default:
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200121 failf(x.Position, "malformed go_package clause %s", str)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200122 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200123 // name.AddComment(comment(x.Comment, true))
124 // name.AddComment(comment(x.InlineComment, false))
125 }
126 }
127 }
128
Marcel van Lohuizen3908dac2019-08-14 01:39:05 +0200129 if name := p.shortName(); name != "" {
130 p.file.Decls = append(p.file.Decls, &ast.Package{Name: ast.NewIdent(name)})
131 }
132
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200133 for _, e := range d.Elements {
134 switch x := e.(type) {
135 case *proto.Import:
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200136 if err := p.doImport(x); err != nil {
137 return nil, err
138 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200139 }
140 }
141
142 imports := &ast.ImportDecl{}
Marcel van Lohuizen3908dac2019-08-14 01:39:05 +0200143 importIdx := len(p.file.Decls)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200144 p.file.Decls = append(p.file.Decls, imports)
145
146 for _, e := range d.Elements {
147 p.topElement(e)
148 }
149
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200150 imported := []string{}
151 for k := range p.imported {
152 imported = append(imported, k)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200153 }
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200154 sort.Strings(imported)
155 p.sorted = imported
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200156
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200157 for _, v := range imported {
Marcel van Lohuizen749a6672019-09-11 08:47:41 +0200158 spec := ast.NewImport(nil, v)
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200159 imports.Specs = append(imports.Specs, spec)
160 p.file.Imports = append(p.file.Imports, spec)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200161 }
162
163 if len(imports.Specs) == 0 {
Marcel van Lohuizen3908dac2019-08-14 01:39:05 +0200164 a := p.file.Decls
165 copy(a[importIdx:], a[importIdx+1:])
166 p.file.Decls = a[:len(a)-1]
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200167 }
168
169 return p, nil
170}
171
172// A protoConverter converts a proto definition to CUE. Proto files map to
173// CUE files one to one.
174type protoConverter struct {
Marcel van Lohuizenf94a5482019-07-02 20:15:03 +0200175 state *Extractor
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200176 tfile *token.File
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200177
178 proto3 bool
179
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200180 id string
181 protoPkg string
182 shortPkgName string
183 cuePkgPath string
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200184
185 // w bytes.Buffer
186 file *ast.File
187 inBody bool
188
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200189 sorted []string
190 imported map[string]bool
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200191
192 path []string
193 scope []map[string]mapping // for symbols resolution within package.
194 symbols map[string]bool // symbols provided by package
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200195 aliases map[string]string // for shadowed packages
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200196}
197
198type mapping struct {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200199 ref string
200 alias string // alias for the type, if exists.
201 pkg *protoConverter
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200202}
203
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200204func (p *protoConverter) importPath() string {
205 if p.cuePkgPath == "" && p.protoPkg != "" {
206 dir := strings.Replace(p.protoPkg, ".", "/", -1)
207 p.cuePkgPath = path.Join("googleapis.com", dir)
208 }
209 return p.cuePkgPath
210}
211
212func (p *protoConverter) shortName() string {
213 if p.shortPkgName == "" && p.protoPkg != "" {
214 split := strings.Split(p.protoPkg, ".")
215 p.shortPkgName = split[len(split)-1]
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200216 }
217 return p.shortPkgName
218}
219
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200220func (p *protoConverter) toCUEPos(pos scanner.Position) token.Pos {
221 return p.tfile.Pos(pos.Offset, 0)
222}
223
224func (p *protoConverter) addRef(pos scanner.Position, from, to string) {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200225 top := p.scope[len(p.scope)-1]
226 if _, ok := top[from]; ok {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200227 failf(pos, "entity %q already defined", from)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200228 }
229 top[from] = mapping{ref: to}
230}
231
232func (p *protoConverter) addNames(elems []proto.Visitee) {
233 p.scope = append(p.scope, map[string]mapping{})
234 for _, e := range elems {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200235 var pos scanner.Position
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200236 var name string
237 switch x := e.(type) {
238 case *proto.Message:
239 if x.IsExtend {
240 continue
241 }
242 name = x.Name
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200243 pos = x.Position
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200244 case *proto.Enum:
245 name = x.Name
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200246 pos = x.Position
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200247 case *proto.NormalField:
248 name = x.Name
249 pos = x.Position
250 case *proto.MapField:
251 name = x.Name
252 pos = x.Position
253 case *proto.Oneof:
254 name = x.Name
255 pos = x.Position
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200256 default:
257 continue
258 }
259 sym := strings.Join(append(p.path, name), ".")
260 p.symbols[sym] = true
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200261 p.addRef(pos, name, strings.Join(append(p.path, name), "_"))
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200262 }
263}
264
265func (p *protoConverter) popNames() {
266 p.scope = p.scope[:len(p.scope)-1]
267}
268
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200269func (p *protoConverter) uniqueTop(name string) string {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200270 a := strings.SplitN(name, ".", 2)
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200271 for i := len(p.scope) - 1; i > 0; i-- {
272 if _, ok := p.scope[i][a[0]]; ok {
273 first := a[0]
274 alias, ok := p.aliases[first]
275 if !ok {
276 // TODO: this is likely to be okay, but find something better.
Marcel van Lohuizene7abb202019-10-08 11:17:17 +0200277 alias = "_" + first + "_"
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200278 p.file.Decls = append(p.file.Decls, &ast.Alias{
279 Ident: ast.NewIdent(alias),
280 Expr: ast.NewIdent(first),
281 })
282 p.aliases[first] = alias
283 }
284 if len(a) > 1 {
285 alias += "." + a[1]
286 }
287 return alias
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200288 }
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200289 }
290 return name
291}
292
293func (p *protoConverter) toExpr(pos scanner.Position, name string) (expr ast.Expr) {
Marcel van Lohuizen0797e462020-01-16 14:32:29 +0100294 expr, err := parser.ParseExpr("", name, parser.ParseComments)
295 if err != nil {
296 panic(err)
297 }
298 ast.SetPos(expr, p.toCUEPos(pos))
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200299 return expr
300}
301
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200302func (p *protoConverter) resolve(pos scanner.Position, name string, options []*proto.Option) string {
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200303 if s, ok := protoToCUE(name, options); ok {
Marcel van Lohuizenc1d1f1a2019-08-13 00:21:24 +0200304 return p.uniqueTop(s)
Marcel van Lohuizen64cb20a2019-08-06 21:24:14 +0200305 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200306 if strings.HasPrefix(name, ".") {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200307 return p.resolveTopScope(pos, name[1:], options)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200308 }
309 for i := len(p.scope) - 1; i > 0; i-- {
310 if m, ok := p.scope[i][name]; ok {
Marcel van Lohuizenafd1cc52019-08-07 00:28:13 +0200311 cueName := strings.Replace(m.ref, ".", "_", -1)
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200312 return cueName
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200313 }
314 }
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200315 return p.resolveTopScope(pos, name, options)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200316}
317
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200318func (p *protoConverter) resolveTopScope(pos scanner.Position, name string, options []*proto.Option) string {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200319 for i := 0; i < len(name); i++ {
320 k := strings.IndexByte(name[i:], '.')
321 i += k
322 if k == -1 {
323 i = len(name)
324 }
325 if m, ok := p.scope[0][name[:i]]; ok {
326 if m.pkg != nil {
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200327 p.imported[m.pkg.importPath()] = true
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200328 // TODO: do something more principled.
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200329 }
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200330 cueName := strings.Replace(name[i:], ".", "_", -1)
331 return p.uniqueTop(m.ref + cueName)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200332 }
333 }
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200334 failf(pos, "name %q not found", name)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200335 return ""
336}
337
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200338func (p *protoConverter) doImport(v *proto.Import) error {
Marcel van Lohuizen9fe5fed2019-06-25 13:04:40 +0200339 if v.Filename == "cue/cue.proto" {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200340 return nil
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200341 }
342
343 filename := ""
344 for _, p := range p.state.paths {
345 name := filepath.Join(p, v.Filename)
346 _, err := os.Stat(name)
347 if err != nil {
348 continue
349 }
350 filename = name
351 break
352 }
353
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200354 if filename == "" {
355 err := errors.Newf(p.toCUEPos(v.Position), "could not find import %q", v.Filename)
356 p.state.addErr(err)
357 return err
358 }
359
Marcel van Lohuizend8f44502019-08-01 15:05:02 +0200360 if !p.mapBuiltinPackage(v.Position, v.Filename, filename == "") {
361 return nil
362 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200363
364 imp, err := p.state.parse(filename, nil)
365 if err != nil {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200366 fail(v.Position, err)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200367 }
368
369 prefix := ""
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200370 if imp.importPath() != p.importPath() {
371 prefix = imp.shortName() + "."
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200372 }
373
374 pkgNamespace := strings.Split(imp.protoPkg, ".")
375 curNamespace := strings.Split(p.protoPkg, ".")
376 for {
377 for k := range imp.symbols {
378 ref := k
379 if len(pkgNamespace) > 0 {
380 ref = strings.Join(append(pkgNamespace, k), ".")
381 }
382 if _, ok := p.scope[0][ref]; !ok {
383 pkg := imp
Marcel van Lohuizen13c97182019-08-13 13:39:50 +0200384 if imp.importPath() == p.importPath() {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200385 pkg = nil
386 }
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200387 p.scope[0][ref] = mapping{prefix + k, "", pkg}
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200388 }
389 }
390 if len(pkgNamespace) == 0 {
391 break
392 }
393 if len(curNamespace) == 0 || pkgNamespace[0] != curNamespace[0] {
394 break
395 }
396 pkgNamespace = pkgNamespace[1:]
397 curNamespace = curNamespace[1:]
398 }
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200399 return nil
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200400}
401
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200402func (p *protoConverter) stringLit(pos scanner.Position, s string) *ast.BasicLit {
403 return &ast.BasicLit{
404 ValuePos: p.toCUEPos(pos),
405 Kind: token.STRING,
406 Value: strconv.Quote(s)}
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200407}
408
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200409func (p *protoConverter) ident(pos scanner.Position, name string) *ast.Ident {
410 return &ast.Ident{NamePos: p.toCUEPos(pos), Name: labelName(name)}
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200411}
412
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200413func (p *protoConverter) ref(pos scanner.Position) *ast.Ident {
414 return &ast.Ident{NamePos: p.toCUEPos(pos), Name: strings.Join(p.path, "_")}
415}
416
417func (p *protoConverter) subref(pos scanner.Position, name string) *ast.Ident {
418 return &ast.Ident{
419 NamePos: p.toCUEPos(pos),
420 Name: strings.Join(append(p.path, name), "_"),
421 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200422}
423
424func (p *protoConverter) addTag(f *ast.Field, body string) {
425 tag := "@protobuf(" + body + ")"
426 f.Attrs = append(f.Attrs, &ast.Attribute{Text: tag})
427}
428
429func (p *protoConverter) topElement(v proto.Visitee) {
430 switch x := v.(type) {
431 case *proto.Syntax:
432 p.proto3 = x.Value == "proto3"
433
434 case *proto.Comment:
435 if p.inBody {
436 p.file.Decls = append(p.file.Decls, comment(x, true))
437 } else {
438 addComments(p.file, 0, x, nil)
439 }
440
441 case *proto.Enum:
442 p.enum(x)
443
444 case *proto.Package:
445 if doc := x.Doc(); doc != nil {
446 addComments(p.file, 0, doc, nil)
447 }
448 // p.inBody bool
449
450 case *proto.Message:
451 p.message(x)
452
453 case *proto.Option:
454 case *proto.Import:
455 // already handled.
456
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200457 case *proto.Service:
458 // TODO: handle services.
459
460 case *proto.Extensions, *proto.Reserved:
Marcel van Lohuizen9fe5fed2019-06-25 13:04:40 +0200461 // no need to handle
462
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200463 default:
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200464 failf(scanner.Position{}, "unsupported type %T", x)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200465 }
466}
467
468func (p *protoConverter) message(v *proto.Message) {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200469 if v.IsExtend {
470 // TODO: we are not handling extensions as for now.
471 return
472 }
473
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200474 defer func(saved []string) { p.path = saved }(p.path)
475 p.path = append(p.path, v.Name)
476
477 p.addNames(v.Elements)
478 defer p.popNames()
479
480 // TODO: handle IsExtend/ proto2
481
482 s := &ast.StructLit{
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200483 Lbrace: p.toCUEPos(v.Position),
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200484 // TOOD: set proto file position.
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200485 Rbrace: token.Newline.Pos(),
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200486 }
487
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200488 ref := p.ref(v.Position)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200489 if v.Comment == nil {
490 ref.NamePos = newSection
491 }
492 f := &ast.Field{Label: ref, Value: s}
493 addComments(f, 1, v.Comment, nil)
494
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200495 p.file.Decls = append(p.file.Decls, f)
496
497 for i, e := range v.Elements {
498 p.messageField(s, i, e)
499 }
500}
501
502func (p *protoConverter) messageField(s *ast.StructLit, i int, v proto.Visitee) {
503 switch x := v.(type) {
504 case *proto.Comment:
505 s.Elts = append(s.Elts, comment(x, true))
506
507 case *proto.NormalField:
508 f := p.parseField(s, i, x.Field)
509
510 if x.Repeated {
511 f.Value = &ast.ListLit{
Marcel van Lohuizened080852019-08-15 16:33:14 +0200512 Lbrack: p.toCUEPos(x.Position),
513 Elts: []ast.Expr{&ast.Ellipsis{Type: f.Value}},
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200514 }
515 }
516
517 case *proto.MapField:
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200518 defer func(saved []string) { p.path = saved }(p.path)
519 p.path = append(p.path, x.Name)
520
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200521 f := &ast.Field{}
522
523 // All keys are converted to strings.
524 // TODO: support integer keys.
Marcel van Lohuizen6d8c95d2019-10-23 23:07:03 +0200525 f.Label = ast.NewList(ast.NewIdent("string"))
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200526 f.Value = p.toExpr(x.Position, p.resolve(x.Position, x.Type, x.Options))
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200527
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200528 name := p.ident(x.Position, x.Name)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200529 f = &ast.Field{
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200530 Label: name,
Marcel van Lohuizenb236d4a2020-02-07 13:14:49 +0100531 Value: ast.NewStruct(f),
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200532 }
533 addComments(f, i, x.Comment, x.InlineComment)
534
535 o := optionParser{message: s, field: f}
536 o.tags = fmt.Sprintf("%d,type=map<%s,%s>", x.Sequence, x.KeyType, x.Type)
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200537 if x.Name != name.Name {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200538 o.tags += "," + x.Name
539 }
540 s.Elts = append(s.Elts, f)
541 o.parse(x.Options)
542 p.addTag(f, o.tags)
543
Jason Wang5f471672019-08-23 09:59:46 -0700544 if !o.required {
545 f.Optional = token.NoSpace.Pos()
546 }
547
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200548 case *proto.Enum:
549 p.enum(x)
550
551 case *proto.Message:
552 p.message(x)
553
554 case *proto.Oneof:
555 p.oneOf(x)
556
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200557 case *proto.Extensions, *proto.Reserved:
Marcel van Lohuizen9fe5fed2019-06-25 13:04:40 +0200558 // no need to handle
559
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200560 default:
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200561 failf(scanner.Position{}, "unsupported field type %T", v)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200562 }
563}
564
565// enum converts a proto enum definition to CUE.
566//
567// An enum will generate two top-level definitions:
568//
569// Enum:
570// "Value1" |
571// "Value2" |
572// "Value3"
573//
574// and
575//
576// Enum_value: {
577// "Value1": 0
578// "Value2": 1
579// }
580//
581// Enums are always defined at the top level. The name of a nested enum
582// will be prefixed with the name of its parent and an underscore.
583func (p *protoConverter) enum(x *proto.Enum) {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200584
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200585 if len(x.Elements) == 0 {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200586 failf(x.Position, "empty enum")
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200587 }
588
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200589 name := p.subref(x.Position, x.Name)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200590
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200591 defer func(saved []string) { p.path = saved }(p.path)
592 p.path = append(p.path, x.Name)
593
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200594 p.addNames(x.Elements)
595
596 if len(p.path) == 0 {
597 defer func() { p.path = p.path[:0] }()
598 p.path = append(p.path, x.Name)
599 }
600
601 // Top-level enum entry.
602 enum := &ast.Field{Label: name}
603 addComments(enum, 1, x.Comment, nil)
604
605 // Top-level enum values entry.
606 valueName := ast.NewIdent(name.Name + "_value")
607 valueName.NamePos = newSection
608 valueMap := &ast.StructLit{}
609 d := &ast.Field{Label: valueName, Value: valueMap}
610 // addComments(valueMap, 1, x.Comment, nil)
611
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200612 if strings.Contains(name.Name, "google") {
613 panic(name.Name)
614 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200615 p.file.Decls = append(p.file.Decls, enum, d)
616
Marcel van Lohuizeneae20e02019-06-29 13:09:43 +0200617 numEnums := 0
618 for _, v := range x.Elements {
619 if _, ok := v.(*proto.EnumField); ok {
620 numEnums++
621 }
622 }
623
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200624 // The line comments for an enum field need to attach after the '|', which
625 // is only known at the next iteration.
626 var lastComment *proto.Comment
627 for i, v := range x.Elements {
628 switch y := v.(type) {
629 case *proto.EnumField:
630 // Add enum value to map
631 f := &ast.Field{
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200632 Label: p.stringLit(y.Position, y.Name),
Marcel van Lohuizene803ab82020-02-07 14:56:48 +0100633 Value: ast.NewLit(token.INT, strconv.Itoa(y.Integer)),
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200634 }
635 valueMap.Elts = append(valueMap.Elts, f)
636
637 // add to enum disjunction
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200638 value := p.stringLit(y.Position, y.Name)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200639
640 var e ast.Expr = value
641 // Make the first value the default value.
Marcel van Lohuizen7213fa92019-10-10 18:20:19 +0200642 if i > 0 {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200643 value.ValuePos = newline
644 }
645 addComments(e, i, y.Comment, nil)
646 if enum.Value != nil {
647 e = &ast.BinaryExpr{X: enum.Value, Op: token.OR, Y: e}
648 if cg := comment(lastComment, false); cg != nil {
649 cg.Position = 2
650 e.AddComment(cg)
651 }
652 }
653 enum.Value = e
654
655 if y.Comment != nil {
656 lastComment = nil
657 addComments(f, 0, nil, y.InlineComment)
658 } else {
659 lastComment = y.InlineComment
660 }
661
662 // a := fmt.Sprintf("@protobuf(enum,name=%s)", y.Name)
663 // f.Attrs = append(f.Attrs, &ast.Attribute{Text: a})
664 }
665 }
666 addComments(enum.Value, 1, nil, lastComment)
667}
668
Marcel van Lohuizen84ffa3c2019-08-12 21:27:15 +0200669// oneOf converts a Proto OneOf field to CUE. Note that Protobuf defines
670// a oneOf to be at most one of the fields. Rather than making each field
671// optional, we define oneOfs as all required fields, but add one more
672// disjunction allowing no fields. This makes it easier to constrain the
673// result to include at least one of the values.
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200674func (p *protoConverter) oneOf(x *proto.Oneof) {
675 f := &ast.Field{
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200676 Label: p.ref(x.Position),
Marcel van Lohuizen84ffa3c2019-08-12 21:27:15 +0200677 // TODO: Once we have closed structs, a oneOf is represented as a
678 // disjunction of empty structs and closed structs with required fields.
679 // For now we just specify the required fields. This is not correct
680 // but more practical.
681 // Value: &ast.StructLit{}, // Remove to make at least one required.
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200682 }
683 f.AddComment(comment(x.Comment, true))
684
685 p.file.Decls = append(p.file.Decls, f)
686
687 for _, v := range x.Elements {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200688 s := &ast.StructLit{
689 // TODO: make this the default in the formatter.
690 Rbrace: token.Newline.Pos(),
691 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200692 switch x := v.(type) {
693 case *proto.OneOfField:
Marcel van Lohuizen84ffa3c2019-08-12 21:27:15 +0200694 oneOf := p.parseField(s, 0, x.Field)
695 oneOf.Optional = token.NoPos
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200696
697 default:
698 p.messageField(s, 1, v)
699 }
700 var e ast.Expr = s
701 if f.Value != nil {
702 e = &ast.BinaryExpr{X: f.Value, Op: token.OR, Y: s}
703 }
704 f.Value = e
705 }
706}
707
708func (p *protoConverter) parseField(s *ast.StructLit, i int, x *proto.Field) *ast.Field {
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200709 defer func(saved []string) { p.path = saved }(p.path)
710 p.path = append(p.path, x.Name)
711
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200712 f := &ast.Field{}
713 addComments(f, i, x.Comment, x.InlineComment)
714
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200715 name := p.ident(x.Position, x.Name)
716 f.Label = name
717 typ := p.resolve(x.Position, x.Type, x.Options)
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200718 f.Value = p.toExpr(x.Position, typ)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200719 s.Elts = append(s.Elts, f)
720
721 o := optionParser{message: s, field: f}
722
723 // body of @protobuf tag: sequence[,type][,name=<name>][,...]
724 o.tags += fmt.Sprint(x.Sequence)
725 if x.Type != typ {
726 o.tags += ",type=" + x.Type
727 }
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200728 if x.Name != name.Name {
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200729 o.tags += ",name=" + x.Name
730 }
731 o.parse(x.Options)
732 p.addTag(f, o.tags)
733
734 if !o.required {
Marcel van Lohuizend094cbe2019-05-20 17:46:48 -0400735 f.Optional = token.NoSpace.Pos()
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200736 }
737 return f
738}
739
740type optionParser struct {
741 message *ast.StructLit
742 field *ast.Field
743 required bool
744 tags string
745}
746
747func (p *optionParser) parse(options []*proto.Option) {
748
749 // TODO: handle options
750 // - translate options to tags
751 // - interpret CUE options.
752 for _, o := range options {
753 switch o.Name {
754 case "(cue_opt).required":
755 p.required = true
756 // TODO: Dropping comments. Maybe add a dummy tag?
757
758 case "(cue.val)":
759 // TODO: set filename and base offset.
Marcel van Lohuizen3c7c40b2019-05-22 11:31:27 -0400760 expr, err := parser.ParseExpr("", o.Constant.Source)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200761 if err != nil {
Marcel van Lohuizenfbcb3392019-06-25 11:02:21 +0200762 failf(o.Position, "invalid cue.val value: %v", err)
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200763 }
764 // Any further checks will be done at the end.
765 constraint := &ast.Field{Label: p.field.Label, Value: expr}
766 addComments(constraint, 1, o.Comment, o.InlineComment)
767 p.message.Elts = append(p.message.Elts, constraint)
768 if !p.required {
Marcel van Lohuizend094cbe2019-05-20 17:46:48 -0400769 constraint.Optional = token.NoSpace.Pos()
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200770 }
771
772 default:
773 // TODO: dropping comments. Maybe add dummy tag?
774
775 // TODO: should CUE support nested attributes?
776 source := o.Constant.SourceRepresentation()
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200777 p.tags += ","
778 switch source {
779 case "true":
780 p.tags += quoteOption(o.Name)
781 default:
782 p.tags += quoteOption(o.Name + "=" + source)
783 }
Marcel van Lohuizen5274e982019-04-28 17:51:43 +0200784 }
785 }
786}
Marcel van Lohuizen93e95972019-06-27 16:47:52 +0200787
788func quoteOption(s string) string {
789 needQuote := false
790 for _, r := range s {
791 if !unicode.In(r, unicode.L, unicode.N) {
792 needQuote = true
793 break
794 }
795 }
796 if !needQuote {
797 return s
798 }
799 if !strings.ContainsAny(s, `"\`) {
800 return strconv.Quote(s)
801 }
802 esc := `\#`
803 for strings.Contains(s, esc) {
804 esc += "#"
805 }
806 return esc[1:] + `"` + s + `"` + esc[1:]
807}