123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- // Copyright (c) 2017 Ernest Micklei
- //
- // MIT License
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- package proto
- import (
- "text/scanner"
- )
- // Enum definition consists of a name and an enum body.
- type Enum struct {
- Position scanner.Position
- Comment *Comment
- Name string
- Elements []Visitee
- Parent Visitee
- }
- // Accept dispatches the call to the visitor.
- func (e *Enum) Accept(v Visitor) {
- v.VisitEnum(e)
- }
- // Doc is part of Documented
- func (e *Enum) Doc() *Comment {
- return e.Comment
- }
- // addElement is part of elementContainer
- func (e *Enum) addElement(v Visitee) {
- v.parent(e)
- e.Elements = append(e.Elements, v)
- }
- // elements is part of elementContainer
- func (e *Enum) elements() []Visitee {
- return e.Elements
- }
- // takeLastComment is part of elementContainer
- // removes and returns the last element of the list if it is a Comment.
- func (e *Enum) takeLastComment(expectedOnLine int) (last *Comment) {
- last, e.Elements = takeLastCommentIfEndsOnLine(e.Elements, expectedOnLine)
- return
- }
- func (e *Enum) parse(p *Parser) error {
- pos, tok, lit := p.next()
- if tok != tIDENT {
- if !isKeyword(tok) {
- return p.unexpected(lit, "enum identifier", e)
- }
- }
- e.Name = lit
- _, tok, lit = p.next()
- if tok != tLEFTCURLY {
- return p.unexpected(lit, "enum opening {", e)
- }
- for {
- pos, tok, lit = p.next()
- switch tok {
- case tCOMMENT:
- if com := mergeOrReturnComment(e.elements(), lit, pos); com != nil { // not merged?
- e.addElement(com)
- }
- case tOPTION:
- v := new(Option)
- v.Position = pos
- v.Comment = e.takeLastComment(pos.Line)
- err := v.parse(p)
- if err != nil {
- return err
- }
- e.addElement(v)
- case tRIGHTCURLY, tEOF:
- goto done
- case tSEMICOLON:
- maybeScanInlineComment(p, e)
- case tRESERVED:
- r := new(Reserved)
- r.Position = pos
- r.Comment = e.takeLastComment(pos.Line - 1)
- if err := r.parse(p); err != nil {
- return err
- }
- e.addElement(r)
- default:
- p.nextPut(pos, tok, lit)
- f := new(EnumField)
- f.Position = pos
- f.Comment = e.takeLastComment(pos.Line - 1)
- if err := f.parse(p); err != nil {
- return err
- }
- e.addElement(f)
- }
- }
- done:
- if tok != tRIGHTCURLY {
- return p.unexpected(lit, "enum closing }", e)
- }
- return nil
- }
- // parent is part of elementContainer
- func (e *Enum) parent(p Visitee) { e.Parent = p }
- // EnumField is part of the body of an Enum.
- type EnumField struct {
- Position scanner.Position
- Comment *Comment
- Name string
- Integer int
- // ValueOption is deprecated, use Elements instead
- ValueOption *Option
- Elements []Visitee // such as Option and Comment
- InlineComment *Comment
- Parent Visitee
- }
- // Accept dispatches the call to the visitor.
- func (f *EnumField) Accept(v Visitor) {
- v.VisitEnumField(f)
- }
- // inlineComment is part of commentInliner.
- func (f *EnumField) inlineComment(c *Comment) {
- f.InlineComment = c
- }
- // Doc is part of Documented
- func (f *EnumField) Doc() *Comment {
- return f.Comment
- }
- func (f *EnumField) parse(p *Parser) (err error) {
- _, tok, lit := p.nextIdentifier()
- if tok != tIDENT {
- if !isKeyword(tok) {
- return p.unexpected(lit, "enum field identifier", f)
- }
- }
- f.Name = lit
- pos, tok, lit := p.next()
- if tok != tEQUALS {
- return p.unexpected(lit, "enum field =", f)
- }
- var i int
- if i, err = p.nextInteger(); err != nil {
- return p.unexpected(err.Error(), "enum field integer", f)
- }
- f.Integer = i
- pos, tok, lit = p.next()
- if tok == tLEFTSQUARE {
- for {
- o := new(Option)
- o.Position = pos
- o.IsEmbedded = true
- err := o.parse(p)
- if err != nil {
- return err
- }
- // update deprecated field with the last option found
- f.ValueOption = o
- f.addElement(o)
- pos, tok, lit = p.next()
- if tok == tCOMMA {
- continue
- }
- if tok == tRIGHTSQUARE {
- break
- }
- }
- }
- if tSEMICOLON == tok {
- p.nextPut(pos, tok, lit) // put back this token for scanning inline comment
- }
- return
- }
- // addElement is part of elementContainer
- func (f *EnumField) addElement(v Visitee) {
- v.parent(f)
- f.Elements = append(f.Elements, v)
- }
- func (f *EnumField) parent(v Visitee) { f.Parent = v }
|