caddy/config/parsing.go

201 lines
4.9 KiB
Go
Raw Normal View History

2015-01-13 22:43:45 +03:00
package config
// This file contains the recursive-descent parsing
// functions.
// begin is the top of the recursive-descent parsing.
// It parses at most one server configuration (an address
2015-01-13 22:43:45 +03:00
// and its directives).
func (p *parser) begin() error {
2015-01-13 22:43:45 +03: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 {
if p.tkn() == "}" || p.tkn() == "{" {
return p.err("Syntax", "'"+p.tkn()+"' is not EOF or address")
}
2015-01-14 03:25:55 +03:00
p.cfg.Host, p.cfg.Port = parseAddress(p.tkn())
2015-01-13 22:43:45 +03:00
return nil
}
// 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 22:43:45 +03:00
// directives not enclosed by curly braces.
func (p *parser) addressBlock() error {
if !p.next() {
// file consisted of only an address
return nil
}
2015-01-13 22:43:45 +03:00
err := p.openCurlyBrace()
if err != nil {
// meh, single-server configs don't need curly braces
p.unused = true // we read the token but aren't consuming it
2015-01-13 22:43:45 +03:00
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
// opening curly brace. This acts like an assertion
// because it returns an error if the token is not
// a opening curly brace. It does not advance the token.
2015-01-13 22:43:45 +03:00
func (p *parser) openCurlyBrace() error {
if p.tkn() != "{" {
return p.syntaxErr("{")
}
return nil
}
// closeCurlyBrace expects the current token to be
// a closing curly brace. This acts like an assertion
// because it returns an error if the token is not
// a closing curly brace. It does not advance the token.
2015-01-13 22:43:45 +03:00
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 which ends the address block.
2015-01-13 22:43:45 +03:00
func (p *parser) directives() error {
for p.next() {
2015-01-13 22:43:45 +03:00
if p.tkn() == "}" {
// end of address scope
2015-01-13 22:43:45 +03:00
break
}
if p.tkn()[0] == '/' {
// Path scope (a.k.a. location context)
// TODO: The parser can handle the syntax (obviously), but the
// implementation is incomplete. This is intentional,
// until we can better decide what kind of feature set we
// want to support. Until this is ready, we leave this
// syntax undocumented.
// location := p.tkn()
if !p.next() {
return p.eofErr()
2015-01-13 22:43:45 +03:00
}
err := p.openCurlyBrace()
if err != nil {
return err
}
for p.next() {
err := p.closeCurlyBrace()
if err == nil { // end of location context
break
}
// TODO: How should we give the context to the directives?
// Or how do we tell the server that these directives should only
// be executed for requests routed to the current path?
err = p.directive()
if err != nil {
return err
}
}
} else if err := p.directive(); err != nil {
return err
}
}
return nil
}
// directive asserts that the current token is either a built-in
// directive or a registered middleware directive; otherwise an error
// will be returned.
func (p *parser) directive() error {
if fn, ok := validDirectives[p.tkn()]; ok {
// Built-in (standard) directive
err := fn(p)
if err != nil {
return err
}
} else if middlewareRegistered(p.tkn()) {
// Middleware directive
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).
// It creates a controller which is stored in the parser for
// later use by the middleware.
func (p *parser) collectTokens() error {
directive := p.tkn()
line := p.line()
nesting := 0
breakOk := false
2015-01-30 20:00:41 +03:00
cont := newController(p)
2015-01-30 20:00:41 +03:00
// Re-use a duplicate directive's controller 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 {
2015-01-30 20:00:41 +03:00
cont = existing
}
// The directive is appended as a relevant token
2015-01-30 20:00:41 +03:00
cont.tokens = append(cont.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 22:43:45 +03:00
}
2015-01-30 20:00:41 +03:00
cont.tokens = append(cont.tokens, p.lexer.token)
2015-01-13 22:43:45 +03:00
}
if !breakOk || nesting > 0 {
return p.eofErr()
}
2015-01-30 20:00:41 +03:00
p.other[directive] = cont
2015-01-13 22:43:45 +03:00
return nil
}