| // Copyright 2017 Frank Schroeder. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package properties |
| |
| import ( |
| "fmt" |
| "runtime" |
| ) |
| |
| type parser struct { |
| lex *lexer |
| } |
| |
| func parse(input string) (properties *Properties, err error) { |
| p := &parser{lex: lex(input)} |
| defer p.recover(&err) |
| |
| properties = NewProperties() |
| key := "" |
| comments := []string{} |
| |
| for { |
| token := p.expectOneOf(itemComment, itemKey, itemEOF) |
| switch token.typ { |
| case itemEOF: |
| goto done |
| case itemComment: |
| comments = append(comments, token.val) |
| continue |
| case itemKey: |
| key = token.val |
| if _, ok := properties.m[key]; !ok { |
| properties.k = append(properties.k, key) |
| } |
| } |
| |
| token = p.expectOneOf(itemValue, itemEOF) |
| if len(comments) > 0 { |
| properties.c[key] = comments |
| comments = []string{} |
| } |
| switch token.typ { |
| case itemEOF: |
| properties.m[key] = "" |
| goto done |
| case itemValue: |
| properties.m[key] = token.val |
| } |
| } |
| |
| done: |
| return properties, nil |
| } |
| |
| func (p *parser) errorf(format string, args ...interface{}) { |
| format = fmt.Sprintf("properties: Line %d: %s", p.lex.lineNumber(), format) |
| panic(fmt.Errorf(format, args...)) |
| } |
| |
| func (p *parser) expect(expected itemType) (token item) { |
| token = p.lex.nextItem() |
| if token.typ != expected { |
| p.unexpected(token) |
| } |
| return token |
| } |
| |
| func (p *parser) expectOneOf(expected ...itemType) (token item) { |
| token = p.lex.nextItem() |
| for _, v := range expected { |
| if token.typ == v { |
| return token |
| } |
| } |
| p.unexpected(token) |
| panic("unexpected token") |
| } |
| |
| func (p *parser) unexpected(token item) { |
| p.errorf(token.String()) |
| } |
| |
| // recover is the handler that turns panics into returns from the top level of Parse. |
| func (p *parser) recover(errp *error) { |
| e := recover() |
| if e != nil { |
| if _, ok := e.(runtime.Error); ok { |
| panic(e) |
| } |
| *errp = e.(error) |
| } |
| return |
| } |