mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-21 20:46:30 -05:00
parent
51e178a6a2
commit
108314444b
4 changed files with 92 additions and 1 deletions
|
@ -18,6 +18,8 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
hconfig "github.com/gohugoio/hugo/config"
|
||||||
|
|
||||||
"golang.org/x/sync/semaphore"
|
"golang.org/x/sync/semaphore"
|
||||||
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
@ -58,7 +60,8 @@ type commandeerHugoState struct {
|
||||||
type commandeer struct {
|
type commandeer struct {
|
||||||
*commandeerHugoState
|
*commandeerHugoState
|
||||||
|
|
||||||
logger *loggers.Logger
|
logger *loggers.Logger
|
||||||
|
serverConfig *config.Server
|
||||||
|
|
||||||
// Currently only set when in "fast render mode". But it seems to
|
// 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.
|
// be fast enough that we could maybe just add it for all server modes.
|
||||||
|
@ -343,6 +346,7 @@ func (c *commandeer) loadConfig(mustHaveConfigFile, running bool) error {
|
||||||
|
|
||||||
cfg.Logger = logger
|
cfg.Logger = logger
|
||||||
c.logger = logger
|
c.logger = logger
|
||||||
|
c.serverConfig = hconfig.DecodeServer(cfg.Cfg)
|
||||||
|
|
||||||
createMemFs := config.GetBool("renderToMemory")
|
createMemFs := config.GetBool("renderToMemory")
|
||||||
|
|
||||||
|
|
|
@ -355,6 +355,10 @@ func (f *fileServer) createEndpoint(i int) (*http.ServeMux, string, string, erro
|
||||||
w.Header().Set("Pragma", "no-cache")
|
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 {
|
if f.c.fastRenderMode && f.c.buildErr == nil {
|
||||||
p := r.RequestURI
|
p := r.RequestURI
|
||||||
if strings.HasSuffix(p, "/") || strings.HasSuffix(p, "html") || strings.HasSuffix(p, "htm") {
|
if strings.HasSuffix(p, "/") || strings.HasSuffix(p, "html") || strings.HasSuffix(p, "htm") {
|
||||||
|
|
|
@ -14,8 +14,13 @@
|
||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/gohugoio/hugo/common/types"
|
||||||
|
|
||||||
|
"github.com/gobwas/glob"
|
||||||
"github.com/gohugoio/hugo/common/herrors"
|
"github.com/gohugoio/hugo/common/herrors"
|
||||||
"github.com/mitchellh/mapstructure"
|
"github.com/mitchellh/mapstructure"
|
||||||
"github.com/spf13/cast"
|
"github.com/spf13/cast"
|
||||||
|
@ -88,3 +93,57 @@ func DecodeSitemap(prototype Sitemap, input map[string]interface{}) Sitemap {
|
||||||
|
|
||||||
return prototype
|
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
|
||||||
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/common/herrors"
|
"github.com/gohugoio/hugo/common/herrors"
|
||||||
|
"github.com/gohugoio/hugo/common/types"
|
||||||
|
|
||||||
qt "github.com/frankban/quicktest"
|
qt "github.com/frankban/quicktest"
|
||||||
|
|
||||||
|
@ -58,3 +59,26 @@ func TestBuild(t *testing.T) {
|
||||||
c.Assert(b.UseResourceCache(nil), 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"}})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue