| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 |
- // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
- // SPDX-License-Identifier: Apache-2.0
- package response // import "miniflux.app/v2/internal/http/response"
- import (
- "compress/flate"
- "compress/gzip"
- "fmt"
- "io"
- "log/slog"
- "maps"
- "mime"
- "net/http"
- "strings"
- "time"
- "github.com/andybalholm/brotli"
- )
- const compressionThreshold = 1024
- // Builder generates HTTP responses.
- type Builder struct {
- w http.ResponseWriter
- r *http.Request
- statusCode int
- headers http.Header
- enableCompression bool
- body any
- }
- // NewBuilder creates a new response builder.
- func NewBuilder(w http.ResponseWriter, r *http.Request) *Builder {
- return &Builder{w: w, r: r, statusCode: http.StatusOK, headers: make(http.Header), enableCompression: true}
- }
- // WithStatus uses the given status code to build the response.
- func (b *Builder) WithStatus(statusCode int) *Builder {
- b.statusCode = statusCode
- return b
- }
- // WithHeader adds the given HTTP header to the response.
- func (b *Builder) WithHeader(key, value string) *Builder {
- b.headers.Set(key, value)
- return b
- }
- // WithBodyAsBytes uses the given bytes to build the response.
- func (b *Builder) WithBodyAsBytes(body []byte) *Builder {
- b.body = body
- return b
- }
- // WithBodyAsString uses the given string to build the response.
- func (b *Builder) WithBodyAsString(body string) *Builder {
- b.body = body
- return b
- }
- // WithBodyAsReader uses the given reader to build the response.
- func (b *Builder) WithBodyAsReader(body io.Reader) *Builder {
- b.body = body
- return b
- }
- // WithAttachment forces the document to be downloaded by the web browser.
- func (b *Builder) WithAttachment(filename string) *Builder {
- b.headers.Set("Content-Disposition", formatContentDisposition("attachment", filename))
- return b
- }
- // WithInline suggests an inline filename for the current response.
- func (b *Builder) WithInline(filename string) *Builder {
- b.headers.Set("Content-Disposition", formatContentDisposition("inline", filename))
- return b
- }
- // WithoutCompression disables HTTP compression.
- func (b *Builder) WithoutCompression() *Builder {
- b.enableCompression = false
- return b
- }
- // WithCaching adds caching headers to the response.
- func (b *Builder) WithCaching(etag string, duration time.Duration, callback func(*Builder)) {
- etag = normalizeETag(etag)
- b.headers.Set("ETag", etag)
- // max-age is required for the "immutable" directive to take effect: without
- // it, browsers still revalidate content-hashed assets on every reload.
- b.headers.Set("Cache-Control", fmt.Sprintf("public, max-age=%d, immutable", int64(duration.Seconds())))
- b.headers.Set("Expires", time.Now().Add(duration).UTC().Format(http.TimeFormat))
- if ifNoneMatch(b.r.Header.Get("If-None-Match"), etag) {
- b.statusCode = http.StatusNotModified
- b.body = nil
- b.Write()
- } else {
- callback(b)
- }
- }
- // Write generates the HTTP response.
- func (b *Builder) Write() {
- if b.body == nil {
- b.writeHeaders()
- return
- }
- switch v := b.body.(type) {
- case []byte:
- b.compress(v)
- case string:
- b.compress([]byte(v))
- case io.Reader:
- // Compression not implemented in this case
- b.writeHeaders()
- _, err := io.Copy(b.w, v)
- if err != nil {
- slog.Error("Unable to write response body", slog.Any("error", err))
- }
- }
- }
- func (b *Builder) writeHeaders() {
- b.headers.Set("X-Content-Type-Options", "nosniff")
- b.headers.Set("X-Frame-Options", "DENY")
- b.headers.Set("Referrer-Policy", "no-referrer")
- maps.Copy(b.w.Header(), b.headers)
- b.w.WriteHeader(b.statusCode)
- }
- func (b *Builder) compress(data []byte) {
- if b.enableCompression && len(data) > compressionThreshold {
- b.headers.Set("Vary", "Accept-Encoding")
- acceptEncoding := b.r.Header.Get("Accept-Encoding")
- switch {
- case strings.Contains(acceptEncoding, "br"):
- b.headers.Set("Content-Encoding", "br")
- b.writeHeaders()
- brotliWriter := brotli.NewWriterV2(b.w, brotli.DefaultCompression)
- brotliWriter.Write(data)
- brotliWriter.Close()
- return
- case strings.Contains(acceptEncoding, "gzip"):
- b.headers.Set("Content-Encoding", "gzip")
- b.writeHeaders()
- gzipWriter := gzip.NewWriter(b.w)
- gzipWriter.Write(data)
- gzipWriter.Close()
- return
- case strings.Contains(acceptEncoding, "deflate"):
- b.headers.Set("Content-Encoding", "deflate")
- b.writeHeaders()
- flateWriter, _ := flate.NewWriter(b.w, -1)
- flateWriter.Write(data)
- flateWriter.Close()
- return
- }
- }
- b.writeHeaders()
- b.w.Write(data)
- }
- func normalizeETag(etag string) string {
- etag = strings.TrimSpace(etag)
- if etag == "" {
- return ""
- }
- if strings.HasPrefix(etag, `"`) || strings.HasPrefix(etag, `W/"`) {
- return etag
- }
- return `"` + etag + `"`
- }
- func ifNoneMatch(headerValue, etag string) bool {
- if headerValue == "" || etag == "" {
- return false
- }
- if strings.TrimSpace(headerValue) == "*" {
- return true
- }
- // Weak ETag comparison: the opaque-tag (quoted string without W/ prefix) must match.
- return strings.Contains(headerValue, strings.TrimPrefix(etag, `W/`))
- }
- func formatContentDisposition(dispositionType, filename string) string {
- if filename == "" {
- return dispositionType
- }
- if value := mime.FormatMediaType(dispositionType, map[string]string{"filename": filename}); value != "" {
- return value
- }
- return dispositionType
- }
|