caddy/modules/caddyhttp/markdown/markdown.go
Matthew Holt 1c443beb9c caddyhttp: ResponseRecorder type for middlewares to buffer responses
Unfortunately, templates and markdown require buffering the full
response before it can be processed and written to the client
2019-06-20 21:49:45 -06:00

59 lines
1.3 KiB
Go

package markdown
import (
"bytes"
"net/http"
"strconv"
"sync"
"gopkg.in/russross/blackfriday.v2"
"github.com/caddyserver/caddy"
"github.com/caddyserver/caddy/modules/caddyhttp"
)
func init() {
caddy.RegisterModule(caddy.Module{
Name: "http.middleware.markdown",
New: func() interface{} { return new(Markdown) },
})
}
// Markdown is a middleware for rendering a Markdown response body.
type Markdown struct {
}
func (m Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyhttp.Handler) error {
buf := bufPool.Get().(*bytes.Buffer)
buf.Reset()
defer bufPool.Put(buf)
rr := caddyhttp.NewResponseRecorder(w, buf)
err := next.ServeHTTP(rr, r)
if err != nil {
return err
}
output := blackfriday.Run(buf.Bytes())
w.Header().Set("Content-Length", strconv.Itoa(len(output)))
w.Header().Set("Content-Type", "text/html; charset=utf-8")
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(rr.Status())
w.Write(output)
return nil
}
var bufPool = sync.Pool{
New: func() interface{} {
return new(bytes.Buffer)
},
}
// Interface guard
var _ caddyhttp.MiddlewareHandler = (*Markdown)(nil)