caddy/caddyconfig/caddyfile/formatter.go

220 lines
4.6 KiB
Go
Raw Permalink Normal View History

// Copyright 2015 Matthew Holt and The Caddy Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package caddyfile
import (
"bytes"
"io"
"unicode"
)
2020-03-26 03:45:54 +03:00
// Format formats the input Caddyfile to a standard, nice-looking
// appearance. It works by reading each rune of the input and taking
// control over all the bracing and whitespace that is written; otherwise,
// words, comments, placeholders, and escaped characters are all treated
// literally and written as they appear in the input.
func Format(input []byte) []byte {
input = bytes.TrimSpace(input)
out := new(bytes.Buffer)
rdr := bytes.NewReader(input)
var (
2020-03-26 03:45:54 +03:00
last rune // the last character that was written to the result
space = true // whether current/previous character was whitespace (beginning of input counts as space)
beginningOfLine = true // whether we are at beginning of line
2020-03-26 03:45:54 +03:00
openBrace bool // whether current word/token is or started with open curly brace
openBraceWritten bool // if openBrace, whether that brace was written or not
openBraceSpace bool // whether there was a non-newline space before open brace
2020-03-26 03:45:54 +03:00
newLines int // count of newlines consumed
2020-03-26 03:45:54 +03:00
comment bool // whether we're in a comment
quoted bool // whether we're in a quoted segment
escaped bool // whether current char is escaped
2020-03-26 03:45:54 +03:00
nesting int // indentation level
)
2020-03-26 03:45:54 +03:00
write := func(ch rune) {
out.WriteRune(ch)
last = ch
}
2020-03-26 03:45:54 +03:00
indent := func() {
for tabs := nesting; tabs > 0; tabs-- {
write('\t')
}
2020-03-26 03:45:54 +03:00
}
2020-03-26 03:45:54 +03:00
nextLine := func() {
write('\n')
beginningOfLine = true
}
for {
ch, _, err := rdr.ReadRune()
if err != nil {
if err == io.EOF {
2020-03-26 03:45:54 +03:00
break
}
panic(err)
}
if comment {
if ch == '\n' {
comment = false
space = true
2020-12-17 01:22:16 +03:00
nextLine()
continue
} else {
2020-03-26 03:45:54 +03:00
write(ch)
continue
}
}
if !escaped && ch == '\\' {
if space {
write(' ')
space = false
}
2020-03-26 03:45:54 +03:00
write(ch)
escaped = true
continue
}
2020-03-26 03:45:54 +03:00
if escaped {
write(ch)
escaped = false
continue
}
if quoted {
2020-03-26 03:45:54 +03:00
if ch == '"' {
quoted = false
}
2020-03-26 03:45:54 +03:00
write(ch)
continue
}
if space && ch == '"' {
quoted = true
}
if unicode.IsSpace(ch) {
space = true
if ch == '\n' {
newLines++
}
2020-03-26 03:45:54 +03:00
continue
}
spacePrior := space
space = false
//////////////////////////////////////////////////////////
// I find it helpful to think of the formatting loop in two
// main sections; by the time we reach this point, we
// know we are in a "regular" part of the file: we know
// the character is not a space, not in a literal segment
// like a comment or quoted, it's not escaped, etc.
//////////////////////////////////////////////////////////
if ch == '#' {
comment = true
}
if openBrace && spacePrior && !openBraceWritten {
if nesting == 0 && last == '}' {
nextLine()
nextLine()
}
2020-03-26 03:45:54 +03:00
openBrace = false
if beginningOfLine {
indent()
} else if !openBraceSpace {
2020-03-26 03:45:54 +03:00
write(' ')
}
2020-03-26 03:45:54 +03:00
write('{')
openBraceWritten = true
2020-03-26 03:45:54 +03:00
nextLine()
newLines = 0
2022-09-15 23:13:58 +03:00
// prevent infinite nesting from ridiculous inputs (issue #4169)
if nesting < 10 {
nesting++
}
2020-03-26 03:45:54 +03:00
}
switch {
case ch == '{':
openBrace = true
openBraceWritten = false
openBraceSpace = spacePrior && !beginningOfLine
if openBraceSpace {
write(' ')
}
2020-03-26 03:45:54 +03:00
continue
case ch == '}' && (spacePrior || !openBrace):
if last != '\n' {
nextLine()
}
2020-03-26 03:45:54 +03:00
if nesting > 0 {
nesting--
}
2020-03-26 03:45:54 +03:00
indent()
write('}')
newLines = 0
continue
}
2020-03-26 03:45:54 +03:00
if newLines > 2 {
newLines = 2
}
for i := 0; i < newLines; i++ {
nextLine()
}
newLines = 0
if beginningOfLine {
indent()
}
if nesting == 0 && last == '}' && beginningOfLine {
2020-03-26 03:45:54 +03:00
nextLine()
nextLine()
}
2020-03-26 03:45:54 +03:00
if !beginningOfLine && spacePrior {
write(' ')
}
if openBrace && !openBraceWritten {
write('{')
openBraceWritten = true
}
write(ch)
beginningOfLine = false
}
2020-03-26 03:45:54 +03:00
// the Caddyfile does not need any leading or trailing spaces, but...
trimmedResult := bytes.TrimSpace(out.Bytes())
// ...Caddyfiles should, however, end with a newline because
// newlines are significant to the syntax of the file
return append(trimmedResult, '\n')
}