ref: 108314444b510bfc330ccac745dce7beccd52c91
parent: 51e178a6a28a3f305d89ebb489675743f80862ee
author: Bjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
date: Sun Mar 8 12:33:15 EDT 2020
Add HTTP header support for the dev server Fixes #7031
--- a/commands/commandeer.go
+++ b/commands/commandeer.go
@@ -18,6 +18,8 @@
"errors"
"sync"
+ hconfig "github.com/gohugoio/hugo/config"
+
"golang.org/x/sync/semaphore"
"io/ioutil"
@@ -58,7 +60,8 @@
type commandeer struct {*commandeerHugoState
- logger *loggers.Logger
+ logger *loggers.Logger
+ serverConfig *config.Server
// Currently only set when in "fast render mode". But it seems to
// be fast enough that we could maybe just add it for all server modes.
@@ -343,6 +346,7 @@
cfg.Logger = logger
c.logger = logger
+ c.serverConfig = hconfig.DecodeServer(cfg.Cfg)
createMemFs := config.GetBool("renderToMemory")--- a/commands/server.go
+++ b/commands/server.go
@@ -355,6 +355,10 @@
w.Header().Set("Pragma", "no-cache")}
+ for _, header := range f.c.serverConfig.Match(r.RequestURI) {+ w.Header().Set(header.Key, header.Value)
+ }
+
if f.c.fastRenderMode && f.c.buildErr == nil {p := r.RequestURI
if strings.HasSuffix(p, "/") || strings.HasSuffix(p, "html") || strings.HasSuffix(p, "htm") {--- a/config/commonConfig.go
+++ b/config/commonConfig.go
@@ -14,8 +14,13 @@
package config
import (
+ "sort"
"strings"
+ "sync"
+ "github.com/gohugoio/hugo/common/types"
+
+ "github.com/gobwas/glob"
"github.com/gohugoio/hugo/common/herrors"
"github.com/mitchellh/mapstructure"
"github.com/spf13/cast"
@@ -87,4 +92,58 @@
}
return prototype
+}
+
+// Config for the dev server.
+type Server struct {+ Headers []Headers
+
+ compiledInit sync.Once
+ compiled []glob.Glob
+}
+
+func (s *Server) Match(pattern string) []types.KeyValueStr {+ s.compiledInit.Do(func() {+ for _, h := range s.Headers {+ s.compiled = append(s.compiled, glob.MustCompile(h.For))
+ }
+ })
+
+ if s.compiled == nil {+ return nil
+ }
+
+ var matches []types.KeyValueStr
+
+ for i, g := range s.compiled {+ if g.Match(pattern) {+ h := s.Headers[i]
+ for k, v := range h.Values {+ matches = append(matches, types.KeyValueStr{Key: k, Value: cast.ToString(v)})+ }
+ }
+ }
+
+ sort.Slice(matches, func(i, j int) bool {+ return matches[i].Key < matches[j].Key
+ })
+
+ return matches
+
+}
+
+type Headers struct {+ For string
+ Values map[string]interface{}+}
+
+func DecodeServer(cfg Provider) *Server {+ m := cfg.GetStringMap("server")+ s := &Server{}+ if m == nil {+ return s
+ }
+
+ _ = mapstructure.WeakDecode(m, s)
+ return s
}
--- a/config/commonConfig_test.go
+++ b/config/commonConfig_test.go
@@ -18,6 +18,7 @@
"testing"
"github.com/gohugoio/hugo/common/herrors"
+ "github.com/gohugoio/hugo/common/types"
qt "github.com/frankban/quicktest"
@@ -56,5 +57,28 @@
c.Assert(b.UseResourceCache(herrors.ErrFeatureNotAvailable), qt.Equals, false)
c.Assert(b.UseResourceCache(errors.New("err")), qt.Equals, false)c.Assert(b.UseResourceCache(nil), qt.Equals, false)
+
+}
+
+func TestServer(t *testing.T) {+ c := qt.New(t)
+
+ cfg, err := FromConfigString(`[[server.headers]]
+for = "/*.jpg"
+
+[server.headers.values]
+X-Frame-Options = "DENY"
+X-XSS-Protection = "1; mode=block"
+X-Content-Type-Options = "nosniff"
+`, "toml")
+
+ c.Assert(err, qt.IsNil)
+
+ s := DecodeServer(cfg)
+
+ c.Assert(s.Match("/foo.jpg"), qt.DeepEquals, []types.KeyValueStr{+ {Key: "X-Content-Type-Options", Value: "nosniff"},+ {Key: "X-Frame-Options", Value: "DENY"},+ {Key: "X-XSS-Protection", Value: "1; mode=block"}})}
--
⑨