mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-14 06:46:27 +03:00
Implement templates handler; various minor cleanups and bug fixes
This commit is contained in:
parent
5137859e47
commit
6706c9225a
9 changed files with 1006 additions and 13 deletions
|
@ -16,6 +16,7 @@ import (
|
||||||
_ "github.com/caddyserver/caddy/modules/caddyhttp/requestbody"
|
_ "github.com/caddyserver/caddy/modules/caddyhttp/requestbody"
|
||||||
_ "github.com/caddyserver/caddy/modules/caddyhttp/reverseproxy"
|
_ "github.com/caddyserver/caddy/modules/caddyhttp/reverseproxy"
|
||||||
_ "github.com/caddyserver/caddy/modules/caddyhttp/rewrite"
|
_ "github.com/caddyserver/caddy/modules/caddyhttp/rewrite"
|
||||||
|
_ "github.com/caddyserver/caddy/modules/caddyhttp/templates"
|
||||||
_ "github.com/caddyserver/caddy/modules/caddytls"
|
_ "github.com/caddyserver/caddy/modules/caddytls"
|
||||||
_ "github.com/caddyserver/caddy/modules/caddytls/standardstek"
|
_ "github.com/caddyserver/caddy/modules/caddytls/standardstek"
|
||||||
)
|
)
|
|
@ -33,7 +33,7 @@ func init() {
|
||||||
type App struct {
|
type App struct {
|
||||||
HTTPPort int `json:"http_port,omitempty"`
|
HTTPPort int `json:"http_port,omitempty"`
|
||||||
HTTPSPort int `json:"https_port,omitempty"`
|
HTTPSPort int `json:"https_port,omitempty"`
|
||||||
GracePeriod caddy.Duration `json:"grace_period,omitempty"`
|
GracePeriod caddy.Duration `json:"grace_period,omitempty"`
|
||||||
Servers map[string]*Server `json:"servers,omitempty"`
|
Servers map[string]*Server `json:"servers,omitempty"`
|
||||||
|
|
||||||
servers []*http.Server
|
servers []*http.Server
|
||||||
|
|
|
@ -148,7 +148,7 @@ func (rw *responseWriter) Write(p []byte) (int, error) {
|
||||||
return n, err
|
return n, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// init should be called once we know we are writing an encoded response.
|
// init should be called before we write a response, if rw.buf is not nil.
|
||||||
func (rw *responseWriter) init() {
|
func (rw *responseWriter) init() {
|
||||||
if rw.Header().Get("Content-Encoding") == "" && rw.buf.Len() >= rw.config.MinLength {
|
if rw.Header().Get("Content-Encoding") == "" && rw.buf.Len() >= rw.config.MinLength {
|
||||||
rw.w = rw.config.writerPools[rw.encodingName].Get().(Encoder)
|
rw.w = rw.config.writerPools[rw.encodingName].Get().(Encoder)
|
||||||
|
@ -164,7 +164,13 @@ func (rw *responseWriter) init() {
|
||||||
// deallocates any active resources.
|
// deallocates any active resources.
|
||||||
func (rw *responseWriter) Close() error {
|
func (rw *responseWriter) Close() error {
|
||||||
var err error
|
var err error
|
||||||
if rw.buf != nil {
|
// only attempt to write the remaining buffered response
|
||||||
|
// if there are any bytes left to write; otherwise, if
|
||||||
|
// the handler above us returned an error without writing
|
||||||
|
// anything, we'd write to the response when we instead
|
||||||
|
// should simply let the error propagate back down; this
|
||||||
|
// is why the check for rw.buf.Len() > 0 is crucial
|
||||||
|
if rw.buf != nil && rw.buf.Len() > 0 {
|
||||||
rw.init()
|
rw.init()
|
||||||
p := rw.buf.Bytes()
|
p := rw.buf.Bytes()
|
||||||
defer func() {
|
defer func() {
|
||||||
|
@ -280,7 +286,7 @@ const defaultMinLength = 512
|
||||||
|
|
||||||
// Interface guards
|
// Interface guards
|
||||||
var (
|
var (
|
||||||
_ caddy.Provisioner = (*Encode)(nil)
|
_ caddy.Provisioner = (*Encode)(nil)
|
||||||
_ caddyhttp.MiddlewareHandler = (*Encode)(nil)
|
_ caddyhttp.MiddlewareHandler = (*Encode)(nil)
|
||||||
_ caddyhttp.HTTPInterfaces = (*responseWriter)(nil)
|
_ caddyhttp.HTTPInterfaces = (*responseWriter)(nil)
|
||||||
)
|
)
|
||||||
|
|
|
@ -186,17 +186,21 @@ type middlewareResponseWriter struct {
|
||||||
|
|
||||||
func (mrw middlewareResponseWriter) WriteHeader(statusCode int) {
|
func (mrw middlewareResponseWriter) WriteHeader(statusCode int) {
|
||||||
if !mrw.allowWrites {
|
if !mrw.allowWrites {
|
||||||
panic("WriteHeader: middleware cannot write to the response")
|
// technically, this is not true: middleware can write headers,
|
||||||
|
// but only after the responder handler has returned; either the
|
||||||
|
// responder did nothing with the response (sad face), or the
|
||||||
|
// middleware wrapped the response and deferred the write
|
||||||
|
panic("WriteHeader: middleware cannot write response headers")
|
||||||
}
|
}
|
||||||
mrw.ResponseWriterWrapper.WriteHeader(statusCode)
|
mrw.ResponseWriterWrapper.WriteHeader(statusCode)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mrw middlewareResponseWriter) Write(b []byte) (int, error) {
|
func (mrw middlewareResponseWriter) Write(b []byte) (int, error) {
|
||||||
if !mrw.allowWrites {
|
if !mrw.allowWrites {
|
||||||
panic("Write: middleware cannot write to the response")
|
panic("Write: middleware cannot write to the response before responder")
|
||||||
}
|
}
|
||||||
return mrw.ResponseWriterWrapper.Write(b)
|
return mrw.ResponseWriterWrapper.Write(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Interface guard
|
// Interface guard
|
||||||
var _ HTTPInterfaces = middlewareResponseWriter{}
|
var _ HTTPInterfaces = (*middlewareResponseWriter)(nil)
|
||||||
|
|
|
@ -15,10 +15,10 @@ import (
|
||||||
// Server is an HTTP server.
|
// Server is an HTTP server.
|
||||||
type Server struct {
|
type Server struct {
|
||||||
Listen []string `json:"listen,omitempty"`
|
Listen []string `json:"listen,omitempty"`
|
||||||
ReadTimeout caddy.Duration `json:"read_timeout,omitempty"`
|
ReadTimeout caddy.Duration `json:"read_timeout,omitempty"`
|
||||||
ReadHeaderTimeout caddy.Duration `json:"read_header_timeout,omitempty"`
|
ReadHeaderTimeout caddy.Duration `json:"read_header_timeout,omitempty"`
|
||||||
WriteTimeout caddy.Duration `json:"write_timeout,omitempty"`
|
WriteTimeout caddy.Duration `json:"write_timeout,omitempty"`
|
||||||
IdleTimeout caddy.Duration `json:"idle_timeout,omitempty"`
|
IdleTimeout caddy.Duration `json:"idle_timeout,omitempty"`
|
||||||
MaxHeaderBytes int `json:"max_header_bytes,omitempty"`
|
MaxHeaderBytes int `json:"max_header_bytes,omitempty"`
|
||||||
Routes RouteList `json:"routes,omitempty"`
|
Routes RouteList `json:"routes,omitempty"`
|
||||||
Errors *httpErrorConfig `json:"errors,omitempty"`
|
Errors *httpErrorConfig `json:"errors,omitempty"`
|
||||||
|
@ -40,6 +40,7 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
// set up the context for the request
|
// set up the context for the request
|
||||||
repl := caddy.NewReplacer()
|
repl := caddy.NewReplacer()
|
||||||
ctx := context.WithValue(r.Context(), caddy.ReplacerCtxKey, repl)
|
ctx := context.WithValue(r.Context(), caddy.ReplacerCtxKey, repl)
|
||||||
|
ctx = context.WithValue(ctx, ServerCtxKey, s)
|
||||||
ctx = context.WithValue(ctx, TableCtxKey, make(map[string]interface{})) // TODO: Implement this
|
ctx = context.WithValue(ctx, TableCtxKey, make(map[string]interface{})) // TODO: Implement this
|
||||||
r = r.WithContext(ctx)
|
r = r.WithContext(ctx)
|
||||||
|
|
||||||
|
@ -126,5 +127,7 @@ type httpErrorConfig struct {
|
||||||
// the logging configuration first.
|
// the logging configuration first.
|
||||||
}
|
}
|
||||||
|
|
||||||
// TableCtxKey is the context key for the request's variable table.
|
const ServerCtxKey caddy.CtxKey = "server"
|
||||||
|
|
||||||
|
// TableCtxKey is the context key for the request's variable table. TODO: implement this
|
||||||
const TableCtxKey caddy.CtxKey = "table"
|
const TableCtxKey caddy.CtxKey = "table"
|
||||||
|
|
146
modules/caddyhttp/templates/templates.go
Normal file
146
modules/caddyhttp/templates/templates.go
Normal file
|
@ -0,0 +1,146 @@
|
||||||
|
package templates
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
"text/template"
|
||||||
|
|
||||||
|
"github.com/caddyserver/caddy"
|
||||||
|
"github.com/caddyserver/caddy/modules/caddyhttp"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
caddy.RegisterModule(caddy.Module{
|
||||||
|
Name: "http.middleware.templates",
|
||||||
|
New: func() interface{} { return new(Templates) },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Templates is a middleware which execute response bodies as templates.
|
||||||
|
type Templates struct {
|
||||||
|
FileRoot string `json:"file_root,omitempty"`
|
||||||
|
Delimiters []string `json:"delimiters,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate ensures t has a valid configuration.
|
||||||
|
func (t *Templates) Validate() error {
|
||||||
|
if len(t.Delimiters) != 0 && len(t.Delimiters) != 2 {
|
||||||
|
return fmt.Errorf("delimiters must consist of exactly two elements: opening and closing")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Templates) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyhttp.Handler) error {
|
||||||
|
buf := bufPool.Get().(*bytes.Buffer)
|
||||||
|
buf.Reset()
|
||||||
|
defer bufPool.Put(buf)
|
||||||
|
|
||||||
|
wb := &responseBuffer{
|
||||||
|
ResponseWriterWrapper: &caddyhttp.ResponseWriterWrapper{ResponseWriter: w},
|
||||||
|
buf: buf,
|
||||||
|
}
|
||||||
|
|
||||||
|
err := next.ServeHTTP(wb, r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = t.executeTemplate(wb, r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Length", strconv.Itoa(wb.buf.Len()))
|
||||||
|
w.Header().Del("Accept-Ranges") // we don't know ranges for dynamically-created content
|
||||||
|
w.Header().Del("Etag") // don't know a way to quickly generate etag for dynamic content
|
||||||
|
w.Header().Del("Last-Modified") // useless for dynamic content since it's always changing
|
||||||
|
|
||||||
|
w.WriteHeader(wb.statusCode)
|
||||||
|
io.Copy(w, wb.buf)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// executeTemplate executes the template contianed
|
||||||
|
// in wb.buf and replaces it with the results.
|
||||||
|
func (t *Templates) executeTemplate(wb *responseBuffer, r *http.Request) error {
|
||||||
|
tpl := template.New(r.URL.Path)
|
||||||
|
|
||||||
|
if len(t.Delimiters) == 2 {
|
||||||
|
tpl.Delims(t.Delimiters[0], t.Delimiters[1])
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedTpl, err := tpl.Parse(wb.buf.String())
|
||||||
|
if err != nil {
|
||||||
|
return caddyhttp.Error(http.StatusInternalServerError, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var fs http.FileSystem
|
||||||
|
if t.FileRoot != "" {
|
||||||
|
fs = http.Dir(t.FileRoot)
|
||||||
|
}
|
||||||
|
ctx := &templateContext{
|
||||||
|
Root: fs,
|
||||||
|
Req: r,
|
||||||
|
RespHeader: tplWrappedHeader{wb.Header()},
|
||||||
|
}
|
||||||
|
|
||||||
|
wb.buf.Reset() // reuse buffer for output
|
||||||
|
err = parsedTpl.Execute(wb.buf, ctx)
|
||||||
|
if err != nil {
|
||||||
|
return caddyhttp.Error(http.StatusInternalServerError, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// responseBuffer buffers the response so that it can be
|
||||||
|
// executed as a template.
|
||||||
|
type responseBuffer struct {
|
||||||
|
*caddyhttp.ResponseWriterWrapper
|
||||||
|
wroteHeader bool
|
||||||
|
statusCode int
|
||||||
|
buf *bytes.Buffer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rb *responseBuffer) WriteHeader(statusCode int) {
|
||||||
|
if rb.wroteHeader {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rb.statusCode = statusCode
|
||||||
|
rb.wroteHeader = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rb *responseBuffer) Write(data []byte) (int, error) {
|
||||||
|
rb.WriteHeader(http.StatusOK)
|
||||||
|
return rb.buf.Write(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// virtualResponseWriter is used in virtualized HTTP requests.
|
||||||
|
type virtualResponseWriter struct {
|
||||||
|
status int
|
||||||
|
header http.Header
|
||||||
|
body *bytes.Buffer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vrw *virtualResponseWriter) Header() http.Header {
|
||||||
|
return vrw.header
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vrw *virtualResponseWriter) WriteHeader(statusCode int) {
|
||||||
|
vrw.status = statusCode
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vrw *virtualResponseWriter) Write(data []byte) (int, error) {
|
||||||
|
return vrw.body.Write(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface guards
|
||||||
|
var (
|
||||||
|
_ caddy.Validator = (*Templates)(nil)
|
||||||
|
_ caddyhttp.MiddlewareHandler = (*Templates)(nil)
|
||||||
|
_ caddyhttp.HTTPInterfaces = (*responseBuffer)(nil)
|
||||||
|
)
|
413
modules/caddyhttp/templates/tplcontext.go
Normal file
413
modules/caddyhttp/templates/tplcontext.go
Normal file
|
@ -0,0 +1,413 @@
|
||||||
|
package templates
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"crypto/rand"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
weakrand "math/rand"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"text/template"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/caddyserver/caddy/modules/caddyhttp"
|
||||||
|
"gopkg.in/russross/blackfriday.v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
// templateContext is the templateContext with which HTTP templates are executed.
|
||||||
|
type templateContext struct {
|
||||||
|
Root http.FileSystem
|
||||||
|
Req *http.Request
|
||||||
|
Args []interface{} // defined by arguments to .Include
|
||||||
|
RespHeader tplWrappedHeader
|
||||||
|
server http.Handler
|
||||||
|
}
|
||||||
|
|
||||||
|
// Include returns the contents of filename relative to the site root.
|
||||||
|
func (c templateContext) Include(filename string, args ...interface{}) (string, error) {
|
||||||
|
if c.Root == nil {
|
||||||
|
return "", fmt.Errorf("root file system not specified")
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := c.Root.Open(filename)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
bodyBuf := bufPool.Get().(*bytes.Buffer)
|
||||||
|
bodyBuf.Reset()
|
||||||
|
defer bufPool.Put(bodyBuf)
|
||||||
|
|
||||||
|
_, err = io.Copy(bodyBuf, file)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
c.Args = args
|
||||||
|
|
||||||
|
return c.executeTemplate(filename, bodyBuf.Bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPInclude returns the body of a virtual (lightweight) request
|
||||||
|
// to the given URI on the same server.
|
||||||
|
func (c templateContext) HTTPInclude(uri string) (string, error) {
|
||||||
|
if c.Req.Header.Get(recursionPreventionHeader) == "1" {
|
||||||
|
return "", fmt.Errorf("virtual include cycle")
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := bufPool.Get().(*bytes.Buffer)
|
||||||
|
buf.Reset()
|
||||||
|
defer bufPool.Put(buf)
|
||||||
|
|
||||||
|
virtReq, err := http.NewRequest("GET", uri, nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
virtReq.Header.Set(recursionPreventionHeader, "1")
|
||||||
|
|
||||||
|
vrw := &virtualResponseWriter{body: buf, header: make(http.Header)}
|
||||||
|
server := c.Req.Context().Value(caddyhttp.ServerCtxKey).(http.Handler)
|
||||||
|
|
||||||
|
server.ServeHTTP(vrw, virtReq)
|
||||||
|
if vrw.status >= 400 {
|
||||||
|
return "", fmt.Errorf("http %d", vrw.status)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.executeTemplate(uri, buf.Bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c templateContext) executeTemplate(tplName string, body []byte) (string, error) {
|
||||||
|
tpl, err := template.New(tplName).Parse(string(body))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := bufPool.Get().(*bytes.Buffer)
|
||||||
|
buf.Reset()
|
||||||
|
defer bufPool.Put(buf)
|
||||||
|
|
||||||
|
err = tpl.Execute(buf, c)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.String(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now returns the current timestamp.
|
||||||
|
func (c templateContext) Now() time.Time {
|
||||||
|
return time.Now()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cookie gets the value of a cookie with name name.
|
||||||
|
func (c templateContext) Cookie(name string) string {
|
||||||
|
cookies := c.Req.Cookies()
|
||||||
|
for _, cookie := range cookies {
|
||||||
|
if cookie.Name == name {
|
||||||
|
return cookie.Value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReqHeader gets the value of a request header with field name.
|
||||||
|
func (c templateContext) ReqHeader(name string) string {
|
||||||
|
return c.Req.Header.Get(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hostname gets the (remote) hostname of the client making the request.
|
||||||
|
func (c templateContext) Hostname() string {
|
||||||
|
ip := c.IP()
|
||||||
|
|
||||||
|
hostnameList, err := net.LookupAddr(ip)
|
||||||
|
if err != nil || len(hostnameList) == 0 {
|
||||||
|
return c.Req.RemoteAddr
|
||||||
|
}
|
||||||
|
|
||||||
|
return hostnameList[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Env gets a map of the environment variables.
|
||||||
|
func (c templateContext) Env() map[string]string {
|
||||||
|
osEnv := os.Environ()
|
||||||
|
envVars := make(map[string]string, len(osEnv))
|
||||||
|
for _, env := range osEnv {
|
||||||
|
data := strings.SplitN(env, "=", 2)
|
||||||
|
if len(data) == 2 && len(data[0]) > 0 {
|
||||||
|
envVars[data[0]] = data[1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return envVars
|
||||||
|
}
|
||||||
|
|
||||||
|
// IP gets the (remote) IP address of the client making the request.
|
||||||
|
func (c templateContext) IP() string {
|
||||||
|
ip, _, err := net.SplitHostPort(c.Req.RemoteAddr)
|
||||||
|
if err != nil {
|
||||||
|
return c.Req.RemoteAddr
|
||||||
|
}
|
||||||
|
return ip
|
||||||
|
}
|
||||||
|
|
||||||
|
// Host returns the hostname portion of the Host header
|
||||||
|
// from the HTTP request.
|
||||||
|
func (c templateContext) Host() (string, error) {
|
||||||
|
host, _, err := net.SplitHostPort(c.Req.Host)
|
||||||
|
if err != nil {
|
||||||
|
if !strings.Contains(c.Req.Host, ":") {
|
||||||
|
// common with sites served on the default port 80
|
||||||
|
return c.Req.Host, nil
|
||||||
|
}
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return host, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Truncate truncates the input string to the given length.
|
||||||
|
// If length is negative, it returns that many characters
|
||||||
|
// starting from the end of the string. If the absolute value
|
||||||
|
// of length is greater than len(input), the whole input is
|
||||||
|
// returned.
|
||||||
|
func (c templateContext) Truncate(input string, length int) string {
|
||||||
|
if length < 0 && len(input)+length > 0 {
|
||||||
|
return input[len(input)+length:]
|
||||||
|
}
|
||||||
|
if length >= 0 && len(input) > length {
|
||||||
|
return input[:length]
|
||||||
|
}
|
||||||
|
return input
|
||||||
|
}
|
||||||
|
|
||||||
|
// StripHTML returns s without HTML tags. It is fairly naive
|
||||||
|
// but works with most valid HTML inputs.
|
||||||
|
func (c templateContext) StripHTML(s string) string {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
var inTag, inQuotes bool
|
||||||
|
var tagStart int
|
||||||
|
for i, ch := range s {
|
||||||
|
if inTag {
|
||||||
|
if ch == '>' && !inQuotes {
|
||||||
|
inTag = false
|
||||||
|
} else if ch == '<' && !inQuotes {
|
||||||
|
// false start
|
||||||
|
buf.WriteString(s[tagStart:i])
|
||||||
|
tagStart = i
|
||||||
|
} else if ch == '"' {
|
||||||
|
inQuotes = !inQuotes
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if ch == '<' {
|
||||||
|
inTag = true
|
||||||
|
tagStart = i
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
buf.WriteRune(ch)
|
||||||
|
}
|
||||||
|
if inTag {
|
||||||
|
// false start
|
||||||
|
buf.WriteString(s[tagStart:])
|
||||||
|
}
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Markdown renders the markdown body as HTML.
|
||||||
|
func (c templateContext) Markdown(body string) string {
|
||||||
|
return string(blackfriday.Run([]byte(body)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ext returns the suffix beginning at the final dot in the final
|
||||||
|
// slash-separated element of the pathStr (or in other words, the
|
||||||
|
// file extension).
|
||||||
|
func (c templateContext) Ext(pathStr string) string {
|
||||||
|
return path.Ext(pathStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StripExt returns the input string without the extension,
|
||||||
|
// which is the suffix starting with the final '.' character
|
||||||
|
// but not before the final path separator ('/') character.
|
||||||
|
// If there is no extension, the whole input is returned.
|
||||||
|
func (c templateContext) StripExt(path string) string {
|
||||||
|
for i := len(path) - 1; i >= 0 && path[i] != '/'; i-- {
|
||||||
|
if path[i] == '.' {
|
||||||
|
return path[:i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
|
||||||
|
// Replace replaces instances of find in input with replacement.
|
||||||
|
func (c templateContext) Replace(input, find, replacement string) string {
|
||||||
|
return strings.Replace(input, find, replacement, -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HasPrefix returns true if s starts with prefix.
|
||||||
|
func (c templateContext) HasPrefix(s, prefix string) bool {
|
||||||
|
return strings.HasPrefix(s, prefix)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToLower will convert the given string to lower case.
|
||||||
|
func (c templateContext) ToLower(s string) string {
|
||||||
|
return strings.ToLower(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToUpper will convert the given string to upper case.
|
||||||
|
func (c templateContext) ToUpper(s string) string {
|
||||||
|
return strings.ToUpper(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Split is a pass-through to strings.Split. It will split
|
||||||
|
// the first argument at each instance of the separator and
|
||||||
|
// return a slice of strings.
|
||||||
|
func (c templateContext) Split(s string, sep string) []string {
|
||||||
|
return strings.Split(s, sep)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Join is a pass-through to strings.Join. It will join the
|
||||||
|
// first argument slice with the separator in the second
|
||||||
|
// argument and return the result.
|
||||||
|
func (c templateContext) Join(a []string, sep string) string {
|
||||||
|
return strings.Join(a, sep)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Slice will convert the given arguments into a slice.
|
||||||
|
func (c templateContext) Slice(elems ...interface{}) []interface{} {
|
||||||
|
return elems
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dict will convert the arguments into a dictionary (map). It expects
|
||||||
|
// alternating keys and values of string types. This is useful since you
|
||||||
|
// cannot express map literals directly in Go templates.
|
||||||
|
func (c templateContext) Dict(values ...interface{}) (map[string]interface{}, error) {
|
||||||
|
if len(values)%2 != 0 {
|
||||||
|
return nil, fmt.Errorf("expected even number of arguments")
|
||||||
|
}
|
||||||
|
dict := make(map[string]interface{}, len(values)/2)
|
||||||
|
for i := 0; i < len(values); i += 2 {
|
||||||
|
key, ok := values[i].(string)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("argument %d: map keys must be strings", i)
|
||||||
|
}
|
||||||
|
dict[key] = values[i+1]
|
||||||
|
}
|
||||||
|
return dict, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListFiles reads and returns a slice of names from the given
|
||||||
|
// directory relative to the root of c.
|
||||||
|
func (c templateContext) ListFiles(name string) ([]string, error) {
|
||||||
|
if c.Root == nil {
|
||||||
|
return nil, fmt.Errorf("root file system not specified")
|
||||||
|
}
|
||||||
|
|
||||||
|
dir, err := c.Root.Open(path.Clean(name))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer dir.Close()
|
||||||
|
|
||||||
|
stat, err := dir.Stat()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if !stat.IsDir() {
|
||||||
|
return nil, fmt.Errorf("%v is not a directory", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
dirInfo, err := dir.Readdir(0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
names := make([]string, len(dirInfo))
|
||||||
|
for i, fileInfo := range dirInfo {
|
||||||
|
names[i] = fileInfo.Name()
|
||||||
|
}
|
||||||
|
|
||||||
|
return names, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RandomString generates a random string of random length given
|
||||||
|
// length bounds. Thanks to http://stackoverflow.com/a/35615565/1048862
|
||||||
|
// for the clever technique that is fairly fast, secure, and maintains
|
||||||
|
// proper distributions over the dictionary.
|
||||||
|
func (c templateContext) RandomString(minLen, maxLen int) string {
|
||||||
|
const (
|
||||||
|
letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"
|
||||||
|
letterIdxBits = 6 // 6 bits to represent 64 possibilities (indexes)
|
||||||
|
letterIdxMask = 1<<letterIdxBits - 1 // all 1-bits, as many as letterIdxBits
|
||||||
|
)
|
||||||
|
|
||||||
|
if minLen < 0 || maxLen < 0 || maxLen < minLen {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
n := weakrand.Intn(maxLen-minLen+1) + minLen // choose actual length
|
||||||
|
|
||||||
|
// secureRandomBytes returns a number of bytes using crypto/rand.
|
||||||
|
secureRandomBytes := func(numBytes int) []byte {
|
||||||
|
randomBytes := make([]byte, numBytes)
|
||||||
|
if _, err := rand.Read(randomBytes); err != nil {
|
||||||
|
// TODO: what to do with the logs (throughout whole file) (could return as error? might get rendered though...)
|
||||||
|
log.Println("[ERROR] failed to read bytes: ", err)
|
||||||
|
}
|
||||||
|
return randomBytes
|
||||||
|
}
|
||||||
|
|
||||||
|
result := make([]byte, n)
|
||||||
|
bufferSize := int(float64(n) * 1.3)
|
||||||
|
for i, j, randomBytes := 0, 0, []byte{}; i < n; j++ {
|
||||||
|
if j%bufferSize == 0 {
|
||||||
|
randomBytes = secureRandomBytes(bufferSize)
|
||||||
|
}
|
||||||
|
if idx := int(randomBytes[j%n] & letterIdxMask); idx < len(letterBytes) {
|
||||||
|
result[i] = letterBytes[idx]
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
// tplWrappedHeader wraps niladic functions so that they
|
||||||
|
// can be used in templates. (Template functions must
|
||||||
|
// return a value.)
|
||||||
|
type tplWrappedHeader struct{ http.Header }
|
||||||
|
|
||||||
|
// Add adds a header field value, appending val to
|
||||||
|
// existing values for that field. It returns an
|
||||||
|
// empty string.
|
||||||
|
func (h tplWrappedHeader) Add(field, val string) string {
|
||||||
|
h.Header.Add(field, val)
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set sets a header field value, overwriting any
|
||||||
|
// other values for that field. It returns an
|
||||||
|
// empty string.
|
||||||
|
func (h tplWrappedHeader) Set(field, val string) string {
|
||||||
|
h.Header.Set(field, val)
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Del deletes a header field. It returns an empty string.
|
||||||
|
func (h tplWrappedHeader) Del(field string) string {
|
||||||
|
h.Header.Del(field)
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
var bufPool = sync.Pool{
|
||||||
|
New: func() interface{} {
|
||||||
|
return new(bytes.Buffer)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const recursionPreventionHeader = "Caddy-Templates-Include"
|
420
modules/caddyhttp/templates/tplcontext_test.go
Normal file
420
modules/caddyhttp/templates/tplcontext_test.go
Normal file
|
@ -0,0 +1,420 @@
|
||||||
|
// Copyright 2015 Light Code Labs, LLC
|
||||||
|
//
|
||||||
|
// 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 templates
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"reflect"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMarkdown(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
|
||||||
|
for i, test := range []struct {
|
||||||
|
body string
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
body: "- str1\n- str2\n",
|
||||||
|
expect: "<ul>\n<li>str1</li>\n<li>str2</li>\n</ul>\n",
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
result := context.Markdown(test.body)
|
||||||
|
if result != test.expect {
|
||||||
|
t.Errorf("Test %d: expected '%s' but got '%s'", i, test.expect, result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCookie(t *testing.T) {
|
||||||
|
for i, test := range []struct {
|
||||||
|
cookie *http.Cookie
|
||||||
|
cookieName string
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// happy path
|
||||||
|
cookie: &http.Cookie{Name: "cookieName", Value: "cookieValue"},
|
||||||
|
cookieName: "cookieName",
|
||||||
|
expect: "cookieValue",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// try to get a non-existing cookie
|
||||||
|
cookie: &http.Cookie{Name: "cookieName", Value: "cookieValue"},
|
||||||
|
cookieName: "notExisting",
|
||||||
|
expect: "",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// partial name match
|
||||||
|
cookie: &http.Cookie{Name: "cookie", Value: "cookieValue"},
|
||||||
|
cookieName: "cook",
|
||||||
|
expect: "",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// cookie with optional fields
|
||||||
|
cookie: &http.Cookie{Name: "cookie", Value: "cookieValue", Path: "/path", Domain: "https://localhost", Expires: (time.Now().Add(10 * time.Minute)), MaxAge: 120},
|
||||||
|
cookieName: "cookie",
|
||||||
|
expect: "cookieValue",
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
context.Req.AddCookie(test.cookie)
|
||||||
|
actual := context.Cookie(test.cookieName)
|
||||||
|
if actual != test.expect {
|
||||||
|
t.Errorf("Test %d: Expected cookie value '%s' but got '%s' for cookie with name '%s'",
|
||||||
|
i, test.expect, actual, test.cookieName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCookieMultipleCookies(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
|
||||||
|
cookieNameBase, cookieValueBase := "cookieName", "cookieValue"
|
||||||
|
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
|
context.Req.AddCookie(&http.Cookie{
|
||||||
|
Name: fmt.Sprintf("%s%d", cookieNameBase, i),
|
||||||
|
Value: fmt.Sprintf("%s%d", cookieValueBase, i),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
|
expectedCookieVal := fmt.Sprintf("%s%d", cookieValueBase, i)
|
||||||
|
actualCookieVal := context.Cookie(fmt.Sprintf("%s%d", cookieNameBase, i))
|
||||||
|
if actualCookieVal != expectedCookieVal {
|
||||||
|
t.Errorf("Expected cookie value %s, found %s", expectedCookieVal, actualCookieVal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEnv(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
|
||||||
|
name := "ENV_TEST_NAME"
|
||||||
|
testValue := "TEST_VALUE"
|
||||||
|
os.Setenv(name, testValue)
|
||||||
|
|
||||||
|
notExisting := "ENV_TEST_NOT_EXISTING"
|
||||||
|
os.Unsetenv(notExisting)
|
||||||
|
|
||||||
|
invalidName := "ENV_TEST_INVALID_NAME"
|
||||||
|
os.Setenv("="+invalidName, testValue)
|
||||||
|
|
||||||
|
env := context.Env()
|
||||||
|
if value := env[name]; value != testValue {
|
||||||
|
t.Errorf("Expected env-variable %s value '%s', found '%s'",
|
||||||
|
name, testValue, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
if value, ok := env[notExisting]; ok {
|
||||||
|
t.Errorf("Expected empty env-variable %s, found '%s'",
|
||||||
|
notExisting, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
for k, v := range env {
|
||||||
|
if strings.Contains(k, invalidName) {
|
||||||
|
t.Errorf("Expected invalid name not to be included in Env %s, found in key '%s'", invalidName, k)
|
||||||
|
}
|
||||||
|
if strings.Contains(v, invalidName) {
|
||||||
|
t.Errorf("Expected invalid name not be be included in Env %s, found in value '%s'", invalidName, v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
os.Unsetenv("=" + invalidName)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIP(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
for i, test := range []struct {
|
||||||
|
inputRemoteAddr string
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{"1.1.1.1:1111", "1.1.1.1"},
|
||||||
|
{"1.1.1.1", "1.1.1.1"},
|
||||||
|
{"[::1]:11", "::1"},
|
||||||
|
{"[2001:db8:a0b:12f0::1]", "[2001:db8:a0b:12f0::1]"},
|
||||||
|
{`[fe80:1::3%eth0]:44`, `fe80:1::3%eth0`},
|
||||||
|
} {
|
||||||
|
context.Req.RemoteAddr = test.inputRemoteAddr
|
||||||
|
if actual := context.IP(); actual != test.expect {
|
||||||
|
t.Errorf("Test %d: Expected %s but got %s", i, test.expect, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTruncate(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
|
||||||
|
for i, test := range []struct {
|
||||||
|
input string
|
||||||
|
length int
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: 1,
|
||||||
|
expect: "s",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: 6,
|
||||||
|
expect: "string",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: 10,
|
||||||
|
expect: "string",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: 0,
|
||||||
|
expect: "",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: -5,
|
||||||
|
expect: "tring",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: -6,
|
||||||
|
expect: "string",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "string",
|
||||||
|
length: -7,
|
||||||
|
expect: "string",
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
actual := context.Truncate(test.input, test.length)
|
||||||
|
if actual != test.expect {
|
||||||
|
t.Errorf("Test %d: Expected '%s' but got '%s'", i, test.expect, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStripHTML(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
|
||||||
|
for i, test := range []struct {
|
||||||
|
input string
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// no tags
|
||||||
|
input: `h1`,
|
||||||
|
expect: `h1`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// happy path
|
||||||
|
input: `<h1>h1</h1>`,
|
||||||
|
expect: `h1`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// tag in quotes
|
||||||
|
input: `<h1">">h1</h1>`,
|
||||||
|
expect: `h1`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// multiple tags
|
||||||
|
input: `<h1><b>h1</b></h1>`,
|
||||||
|
expect: `h1`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// tags not closed
|
||||||
|
input: `<h1`,
|
||||||
|
expect: `<h1`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// false start
|
||||||
|
input: `<h1<b>hi`,
|
||||||
|
expect: `<h1hi`,
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
actual := context.StripHTML(test.input)
|
||||||
|
if actual != test.expect {
|
||||||
|
t.Errorf("Test %d: Expected %s, found %s. Input was StripHTML(%s)", i, test.expect, actual, test.input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStripExt(t *testing.T) {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
tests := []struct {
|
||||||
|
input string
|
||||||
|
expect string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
input: "",
|
||||||
|
expect: "",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "file.ext",
|
||||||
|
expect: "file",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "file",
|
||||||
|
expect: "file",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "/file",
|
||||||
|
expect: "/file",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "/file.ext",
|
||||||
|
expect: "/file",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "/dir.ext/",
|
||||||
|
expect: "/dir.ext/",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: "/dir.ext/file.ext",
|
||||||
|
expect: "/dir.ext/file",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, test := range tests {
|
||||||
|
actual := context.StripExt(test.input)
|
||||||
|
if actual != test.expect {
|
||||||
|
t.Errorf("Test %d: Expected %s but got %s", i, test.expect, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFileListing(t *testing.T) {
|
||||||
|
for i, test := range []struct {
|
||||||
|
fileNames []string
|
||||||
|
inputBase string
|
||||||
|
shouldErr bool
|
||||||
|
verifyErr func(error) bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// directory and files exist
|
||||||
|
fileNames: []string{"file1", "file2"},
|
||||||
|
shouldErr: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// directory exists, no files
|
||||||
|
fileNames: []string{},
|
||||||
|
shouldErr: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// file or directory does not exist
|
||||||
|
fileNames: nil,
|
||||||
|
inputBase: "doesNotExist",
|
||||||
|
shouldErr: true,
|
||||||
|
verifyErr: os.IsNotExist,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// directory and files exist, but path to a file
|
||||||
|
fileNames: []string{"file1", "file2"},
|
||||||
|
inputBase: "file1",
|
||||||
|
shouldErr: true,
|
||||||
|
verifyErr: func(err error) bool {
|
||||||
|
return strings.HasSuffix(err.Error(), "is not a directory")
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// try to escape Context Root
|
||||||
|
fileNames: nil,
|
||||||
|
inputBase: filepath.Join("..", "..", "..", "..", "..", "etc"),
|
||||||
|
shouldErr: true,
|
||||||
|
verifyErr: os.IsNotExist,
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
context := getContextOrFail(t)
|
||||||
|
var dirPath string
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// create files for test case
|
||||||
|
if test.fileNames != nil {
|
||||||
|
dirPath, err = ioutil.TempDir(fmt.Sprintf("%s", context.Root), "caddy_ctxtest")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Test %d: Expected no error creating directory, got: '%s'", i, err.Error())
|
||||||
|
}
|
||||||
|
for _, name := range test.fileNames {
|
||||||
|
absFilePath := filepath.Join(dirPath, name)
|
||||||
|
if err = ioutil.WriteFile(absFilePath, []byte(""), os.ModePerm); err != nil {
|
||||||
|
os.RemoveAll(dirPath)
|
||||||
|
t.Fatalf("Test %d: Expected no error creating file, got: '%s'", i, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// perform test
|
||||||
|
input := filepath.ToSlash(filepath.Join(filepath.Base(dirPath), test.inputBase))
|
||||||
|
actual, err := context.ListFiles(input)
|
||||||
|
if err != nil {
|
||||||
|
if !test.shouldErr {
|
||||||
|
t.Errorf("Test %d: Expected no error, got: '%s'", i, err)
|
||||||
|
} else if !test.verifyErr(err) {
|
||||||
|
t.Errorf("Test %d: Could not verify error content, got: '%s'", i, err)
|
||||||
|
}
|
||||||
|
} else if test.shouldErr {
|
||||||
|
t.Errorf("Test %d: Expected error but had none", i)
|
||||||
|
} else {
|
||||||
|
numFiles := len(test.fileNames)
|
||||||
|
// reflect.DeepEqual does not consider two empty slices to be equal
|
||||||
|
if numFiles == 0 && len(actual) != 0 {
|
||||||
|
t.Errorf("Test %d: Expected files %v, got: %v",
|
||||||
|
i, test.fileNames, actual)
|
||||||
|
} else {
|
||||||
|
sort.Strings(actual)
|
||||||
|
if numFiles > 0 && !reflect.DeepEqual(test.fileNames, actual) {
|
||||||
|
t.Errorf("Test %d: Expected files %v, got: %v",
|
||||||
|
i, test.fileNames, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if dirPath != "" {
|
||||||
|
if err := os.RemoveAll(dirPath); err != nil && !os.IsNotExist(err) {
|
||||||
|
t.Fatalf("Test %d: Expected no error removing temporary test directory, got: %v", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getContextOrFail(t *testing.T) templateContext {
|
||||||
|
context, err := initTestContext()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("failed to prepare test context: %v", err)
|
||||||
|
}
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
|
func initTestContext() (templateContext, error) {
|
||||||
|
body := bytes.NewBufferString("request body")
|
||||||
|
request, err := http.NewRequest("GET", "https://example.com/foo/bar", body)
|
||||||
|
if err != nil {
|
||||||
|
return templateContext{}, err
|
||||||
|
}
|
||||||
|
return templateContext{
|
||||||
|
Root: http.Dir(os.TempDir()),
|
||||||
|
Req: request,
|
||||||
|
RespHeader: tplWrappedHeader{make(http.Header)},
|
||||||
|
}, nil
|
||||||
|
}
|
|
@ -28,4 +28,4 @@ func (m MatchServerName) Match(hello *tls.ClientHelloInfo) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Interface guard
|
// Interface guard
|
||||||
var _ ConnectionMatcher = MatchServerName{}
|
var _ ConnectionMatcher = (*MatchServerName)(nil)
|
||||||
|
|
Loading…
Reference in a new issue