2015-01-13 14:43:45 -05:00
|
|
|
package config
|
|
|
|
|
2015-01-19 01:11:21 -05:00
|
|
|
import "github.com/mholt/caddy/middleware"
|
|
|
|
|
2015-01-13 14:43:45 -05:00
|
|
|
// This file contains the recursive-descent parsing
|
|
|
|
// functions.
|
|
|
|
|
2015-01-19 01:11:21 -05:00
|
|
|
// begin is the top of the recursive-descent parsing.
|
|
|
|
// It parses at most one server configuration (an address
|
2015-01-13 14:43:45 -05:00
|
|
|
// and its directives).
|
2015-01-19 01:11:21 -05:00
|
|
|
func (p *parser) begin() error {
|
2015-01-13 14:43:45 -05:00
|
|
|
err := p.address()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = p.addressBlock()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// address expects that the current token is a host:port
|
|
|
|
// combination.
|
|
|
|
func (p *parser) address() error {
|
2015-01-19 01:11:21 -05:00
|
|
|
if p.tkn() == "}" || p.tkn() == "{" {
|
|
|
|
return p.err("Syntax", "'"+p.tkn()+"' is not a listening address or EOF")
|
|
|
|
}
|
2015-01-13 19:25:55 -05:00
|
|
|
p.cfg.Host, p.cfg.Port = parseAddress(p.tkn())
|
2015-01-13 14:43:45 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-01-19 01:11:21 -05:00
|
|
|
// addressBlock leads into parsing directives, including
|
|
|
|
// possible opening/closing curly braces around the block.
|
|
|
|
// It handles directives enclosed by curly braces and
|
2015-01-13 14:43:45 -05:00
|
|
|
// directives not enclosed by curly braces.
|
|
|
|
func (p *parser) addressBlock() error {
|
2015-01-19 01:11:21 -05:00
|
|
|
if !p.next() {
|
|
|
|
// file consisted of only an address
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-01-13 14:43:45 -05:00
|
|
|
err := p.openCurlyBrace()
|
|
|
|
if err != nil {
|
|
|
|
// meh, single-server configs don't need curly braces
|
|
|
|
return p.directives()
|
|
|
|
}
|
|
|
|
|
|
|
|
err = p.directives()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = p.closeCurlyBrace()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// openCurlyBrace expects the current token to be an
|
2015-01-19 01:11:21 -05:00
|
|
|
// opening curly brace. This acts like an assertion
|
|
|
|
// because it returns an error if the token is not
|
|
|
|
// a opening curly brace.
|
2015-01-13 14:43:45 -05:00
|
|
|
func (p *parser) openCurlyBrace() error {
|
|
|
|
if p.tkn() != "{" {
|
|
|
|
return p.syntaxErr("{")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// closeCurlyBrace expects the current token to be
|
2015-01-19 01:11:21 -05:00
|
|
|
// a closing curly brace. This acts like an assertion
|
|
|
|
// because it returns an error if the token is not
|
2015-01-13 14:43:45 -05:00
|
|
|
// a closing curly brace.
|
|
|
|
func (p *parser) closeCurlyBrace() error {
|
|
|
|
if p.tkn() != "}" {
|
|
|
|
return p.syntaxErr("}")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// directives parses through all the directives
|
|
|
|
// and it expects the current token to be the first
|
|
|
|
// directive. It goes until EOF or closing curly
|
|
|
|
// brace.
|
|
|
|
func (p *parser) directives() error {
|
2015-01-19 01:11:21 -05:00
|
|
|
for p.next() {
|
2015-01-13 14:43:45 -05:00
|
|
|
if p.tkn() == "}" {
|
2015-01-19 01:11:21 -05:00
|
|
|
// end of address scope
|
2015-01-13 14:43:45 -05:00
|
|
|
break
|
|
|
|
}
|
2015-01-19 01:11:21 -05:00
|
|
|
if fn, ok := validDirectives[p.tkn()]; ok {
|
2015-01-13 14:43:45 -05:00
|
|
|
err := fn(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-01-19 01:11:21 -05:00
|
|
|
} else if middleware.Registered(p.tkn()) {
|
|
|
|
err := p.collectTokens()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return p.err("Syntax", "Unexpected token '"+p.tkn()+"', expecting a valid directive")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// collectTokens consumes tokens until the directive's scope
|
|
|
|
// closes (either end of line or end of curly brace block).
|
|
|
|
func (p *parser) collectTokens() error {
|
|
|
|
directive := p.tkn()
|
|
|
|
line := p.line()
|
|
|
|
nesting := 0
|
|
|
|
breakOk := false
|
|
|
|
disp := newDispenser(p)
|
|
|
|
|
|
|
|
// Re-use a duplicate directive's dispenser from before
|
|
|
|
// (the parsing logic in the middleware generator must
|
|
|
|
// account for multiple occurrences of its directive, even
|
|
|
|
// if that means returning an error or overwriting settings)
|
|
|
|
if existing, ok := p.other[directive]; ok {
|
|
|
|
disp = existing
|
|
|
|
}
|
|
|
|
|
|
|
|
// The directive is appended as a relevant token
|
|
|
|
disp.tokens = append(disp.tokens, p.lexer.token)
|
|
|
|
|
|
|
|
for p.next() {
|
|
|
|
if p.tkn() == "{" {
|
|
|
|
nesting++
|
|
|
|
} else if p.line() > line && nesting == 0 {
|
|
|
|
p.unused = true
|
|
|
|
breakOk = true
|
|
|
|
break
|
|
|
|
} else if p.tkn() == "}" && nesting > 0 {
|
|
|
|
nesting--
|
|
|
|
} else if p.tkn() == "}" && nesting == 0 {
|
|
|
|
return p.err("Syntax", "Unexpected '}' because no matching open curly brace '{'")
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
2015-01-19 01:11:21 -05:00
|
|
|
disp.tokens = append(disp.tokens, p.lexer.token)
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
2015-01-19 01:11:21 -05:00
|
|
|
|
|
|
|
if !breakOk || nesting > 0 {
|
|
|
|
return p.eofErr()
|
|
|
|
}
|
|
|
|
|
|
|
|
p.other[directive] = disp
|
2015-01-13 14:43:45 -05:00
|
|
|
return nil
|
|
|
|
}
|