2015-05-04 11:04:17 -06:00
|
|
|
package parse
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
|
|
|
"os"
|
2015-07-07 22:38:48 -06:00
|
|
|
"path/filepath"
|
2015-05-04 11:04:17 -06:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type parser struct {
|
|
|
|
Dispenser
|
2015-10-14 23:45:28 -06:00
|
|
|
block serverBlock // current server block being parsed
|
2015-08-01 13:08:31 -06:00
|
|
|
eof bool // if we encounter a valid EOF in a hard place
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
2015-10-14 23:45:28 -06:00
|
|
|
func (p *parser) parseAll() ([]serverBlock, error) {
|
|
|
|
var blocks []serverBlock
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
for p.Next() {
|
|
|
|
err := p.parseOne()
|
|
|
|
if err != nil {
|
|
|
|
return blocks, err
|
|
|
|
}
|
2015-08-03 17:31:10 -06:00
|
|
|
if len(p.block.Addresses) > 0 {
|
|
|
|
blocks = append(blocks, p.block)
|
|
|
|
}
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return blocks, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *parser) parseOne() error {
|
2015-10-14 23:45:28 -06:00
|
|
|
p.block = serverBlock{Tokens: make(map[string][]token)}
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
err := p.begin()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *parser) begin() error {
|
2015-05-21 18:46:42 -06:00
|
|
|
if len(p.tokens) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-05-04 11:04:17 -06:00
|
|
|
err := p.addresses()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-05-06 08:58:15 -06:00
|
|
|
if p.eof {
|
|
|
|
// this happens if the Caddyfile consists of only
|
|
|
|
// a line of addresses and nothing else
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-05-04 11:04:17 -06:00
|
|
|
err = p.blockContents()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *parser) addresses() error {
|
|
|
|
var expectingAnother bool
|
|
|
|
|
|
|
|
for {
|
2015-07-07 22:38:48 -06:00
|
|
|
tkn := p.Val()
|
|
|
|
|
|
|
|
// special case: import directive replaces tokens during parse-time
|
|
|
|
if tkn == "import" && p.isNewLine() {
|
|
|
|
err := p.doImport()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
// Open brace definitely indicates end of addresses
|
|
|
|
if tkn == "{" {
|
|
|
|
if expectingAnother {
|
|
|
|
return p.Errf("Expected another address but had '%s' - check for extra comma", tkn)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2015-10-14 23:45:28 -06:00
|
|
|
if tkn != "" { // empty token possible if user typed "" in Caddyfile
|
2015-08-03 17:31:10 -06:00
|
|
|
// Trailing comma indicates another address will follow, which
|
|
|
|
// may possibly be on the next line
|
|
|
|
if tkn[len(tkn)-1] == ',' {
|
|
|
|
tkn = tkn[:len(tkn)-1]
|
|
|
|
expectingAnother = true
|
|
|
|
} else {
|
|
|
|
expectingAnother = false // but we may still see another one on this line
|
|
|
|
}
|
2015-05-04 11:04:17 -06:00
|
|
|
|
2015-08-03 17:31:10 -06:00
|
|
|
// Parse and save this address
|
|
|
|
host, port, err := standardAddress(tkn)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-10-14 23:45:28 -06:00
|
|
|
p.block.Addresses = append(p.block.Addresses, address{host, port})
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Advance token and possibly break out of loop or return error
|
|
|
|
hasNext := p.Next()
|
|
|
|
if expectingAnother && !hasNext {
|
2015-10-09 18:35:34 -04:00
|
|
|
return p.EOFErr()
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
if !hasNext {
|
|
|
|
p.eof = true
|
|
|
|
break // EOF
|
|
|
|
}
|
2015-07-07 22:38:48 -06:00
|
|
|
if !expectingAnother && p.isNewLine() {
|
|
|
|
break
|
|
|
|
}
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *parser) blockContents() error {
|
|
|
|
errOpenCurlyBrace := p.openCurlyBrace()
|
|
|
|
if errOpenCurlyBrace != nil {
|
|
|
|
// single-server configs don't need curly braces
|
|
|
|
p.cursor--
|
|
|
|
}
|
|
|
|
|
|
|
|
err := p.directives()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only look for close curly brace if there was an opening
|
|
|
|
if errOpenCurlyBrace == nil {
|
|
|
|
err = p.closeCurlyBrace()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// directives parses through all the lines for directives
|
|
|
|
// and it expects the next token to be the first
|
|
|
|
// directive. It goes until EOF or closing curly brace
|
|
|
|
// which ends the server block.
|
|
|
|
func (p *parser) directives() error {
|
|
|
|
for p.Next() {
|
|
|
|
// end of server block
|
|
|
|
if p.Val() == "}" {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// special case: import directive replaces tokens during parse-time
|
|
|
|
if p.Val() == "import" {
|
|
|
|
err := p.doImport()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-07-07 22:38:48 -06:00
|
|
|
p.cursor-- // cursor is advanced when we continue, so roll back one more
|
2015-05-04 11:04:17 -06:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// normal case: parse a directive on this line
|
|
|
|
if err := p.directive(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// doImport swaps out the import directive and its argument
|
|
|
|
// (a total of 2 tokens) with the tokens in the file specified.
|
|
|
|
// When the function returns, the cursor is on the token before
|
|
|
|
// where the import directive was. In other words, call Next()
|
|
|
|
// to access the first token that was imported.
|
|
|
|
func (p *parser) doImport() error {
|
|
|
|
if !p.NextArg() {
|
|
|
|
return p.ArgErr()
|
|
|
|
}
|
|
|
|
importFile := p.Val()
|
|
|
|
if p.NextArg() {
|
|
|
|
return p.Err("Import allows only one file to import")
|
|
|
|
}
|
|
|
|
|
|
|
|
file, err := os.Open(importFile)
|
|
|
|
if err != nil {
|
|
|
|
return p.Errf("Could not import %s - %v", importFile, err)
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
importedTokens := allTokens(file)
|
|
|
|
|
2015-07-07 22:38:48 -06:00
|
|
|
// Tack the filename onto these tokens so any errors show the imported file's name
|
|
|
|
for i := 0; i < len(importedTokens); i++ {
|
|
|
|
importedTokens[i].file = filepath.Base(importFile)
|
|
|
|
}
|
|
|
|
|
2015-05-04 11:04:17 -06:00
|
|
|
// Splice out the import directive and its argument (2 tokens total)
|
2015-07-07 22:38:48 -06:00
|
|
|
// and insert the imported tokens in their place.
|
2015-05-04 11:04:17 -06:00
|
|
|
tokensBefore := p.tokens[:p.cursor-1]
|
|
|
|
tokensAfter := p.tokens[p.cursor+1:]
|
|
|
|
p.tokens = append(tokensBefore, append(importedTokens, tokensAfter...)...)
|
2015-07-07 22:38:48 -06:00
|
|
|
p.cursor-- // cursor was advanced one position to read the filename; rewind it
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// directive collects tokens until the directive's scope
|
|
|
|
// closes (either end of line or end of curly brace block).
|
|
|
|
// It expects the currently-loaded token to be a directive
|
|
|
|
// (or } that ends a server block). The collected tokens
|
|
|
|
// are loaded into the current server block for later use
|
|
|
|
// by directive setup functions.
|
|
|
|
func (p *parser) directive() error {
|
|
|
|
dir := p.Val()
|
|
|
|
nesting := 0
|
|
|
|
|
|
|
|
if _, ok := ValidDirectives[dir]; !ok {
|
|
|
|
return p.Errf("Unknown directive '%s'", dir)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The directive itself is appended as a relevant token
|
2015-08-01 13:08:31 -06:00
|
|
|
p.block.Tokens[dir] = append(p.block.Tokens[dir], p.tokens[p.cursor])
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
for p.Next() {
|
|
|
|
if p.Val() == "{" {
|
|
|
|
nesting++
|
2015-07-07 22:38:48 -06:00
|
|
|
} else if p.isNewLine() && nesting == 0 {
|
2015-05-04 11:04:17 -06:00
|
|
|
p.cursor-- // read too far
|
|
|
|
break
|
|
|
|
} else if p.Val() == "}" && nesting > 0 {
|
|
|
|
nesting--
|
|
|
|
} else if p.Val() == "}" && nesting == 0 {
|
|
|
|
return p.Err("Unexpected '}' because no matching opening brace")
|
|
|
|
}
|
2015-08-01 13:08:31 -06:00
|
|
|
p.block.Tokens[dir] = append(p.block.Tokens[dir], p.tokens[p.cursor])
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if nesting > 0 {
|
2015-10-09 18:35:34 -04:00
|
|
|
return p.EOFErr()
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
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
|
2015-07-07 22:38:48 -06:00
|
|
|
// a opening curly brace. It does NOT advance the token.
|
2015-05-04 11:04:17 -06:00
|
|
|
func (p *parser) openCurlyBrace() error {
|
|
|
|
if p.Val() != "{" {
|
|
|
|
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
|
2015-07-07 22:38:48 -06:00
|
|
|
// a closing curly brace. It does NOT advance the token.
|
2015-05-04 11:04:17 -06:00
|
|
|
func (p *parser) closeCurlyBrace() error {
|
|
|
|
if p.Val() != "}" {
|
|
|
|
return p.SyntaxErr("}")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// standardAddress turns the accepted host and port patterns
|
|
|
|
// into a format accepted by net.Dial.
|
|
|
|
func standardAddress(str string) (host, port string, err error) {
|
2015-05-21 16:31:01 -06:00
|
|
|
var schemePort, splitPort string
|
2015-05-04 11:04:17 -06:00
|
|
|
|
|
|
|
if strings.HasPrefix(str, "https://") {
|
|
|
|
schemePort = "https"
|
|
|
|
str = str[8:]
|
|
|
|
} else if strings.HasPrefix(str, "http://") {
|
|
|
|
schemePort = "http"
|
|
|
|
str = str[7:]
|
|
|
|
}
|
|
|
|
|
2015-05-21 16:31:01 -06:00
|
|
|
host, splitPort, err = net.SplitHostPort(str)
|
|
|
|
if err != nil {
|
|
|
|
host, splitPort, err = net.SplitHostPort(str + ":") // tack on empty port
|
2015-05-20 20:15:39 -06:00
|
|
|
}
|
2015-05-21 16:31:01 -06:00
|
|
|
if err != nil {
|
|
|
|
// ¯\_(ツ)_/¯
|
2015-05-04 11:04:17 -06:00
|
|
|
host = str
|
2015-05-21 16:31:01 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if splitPort != "" {
|
|
|
|
port = splitPort
|
|
|
|
} else {
|
|
|
|
port = schemePort
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
type (
|
2015-10-14 23:45:28 -06:00
|
|
|
// serverBlock associates tokens with a list of addresses
|
2015-08-01 13:08:31 -06:00
|
|
|
// and groups tokens by directive name.
|
2015-10-14 23:45:28 -06:00
|
|
|
serverBlock struct {
|
|
|
|
Addresses []address
|
2015-08-01 13:08:31 -06:00
|
|
|
Tokens map[string][]token
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
|
2015-10-14 23:45:28 -06:00
|
|
|
address struct {
|
2015-08-01 13:08:31 -06:00
|
|
|
Host, Port string
|
2015-05-04 11:04:17 -06:00
|
|
|
}
|
|
|
|
)
|
2015-10-15 23:34:54 -06:00
|
|
|
|
|
|
|
// HostList converts the list of addresses (hosts)
|
|
|
|
// that are associated with this server block into
|
|
|
|
// a slice of strings. Each string is a host:port
|
|
|
|
// combination.
|
|
|
|
func (sb serverBlock) HostList() []string {
|
|
|
|
sbHosts := make([]string, len(sb.Addresses))
|
|
|
|
for j, addr := range sb.Addresses {
|
|
|
|
sbHosts[j] = net.JoinHostPort(addr.Host, addr.Port)
|
|
|
|
}
|
|
|
|
return sbHosts
|
|
|
|
}
|