permalink string
relPermalink string
+ layoutDescriptor output.LayoutDescriptor
+
scratch *Scratch
// It would be tempting to use the language set on the Site, but in they way we do
}
return p.s.layoutHandler.For(
- p.createLayoutDescriptor(),
+ p.layoutDescriptor,
layoutOverride,
output.HTMLType)
}
p.permalink = p.s.permalink(rel)
rel = p.s.PathSpec.PrependBasePath(rel)
p.relPermalink = rel
+ p.layoutDescriptor = p.createLayoutDescriptor()
return nil
}
func (p *Page) RSSlink() template.URL {
// TODO(bep) we cannot have two of these
// Remove in Hugo 0.20
- helpers.Deprecated(".Page", "Use RSSlink", "RSSLink", true)
+ helpers.Deprecated(".Page", "RSSlink", "Use RSSLink", true)
return p.RSSLink
}
}
func (s *Site) layouts(p *PageOutput) []string {
- return s.layoutHandler.For(p.createLayoutDescriptor(), "", p.outputFormat)
+ return s.layoutHandler.For(p.layoutDescriptor, "", p.outputFormat)
}
func (s *Site) preparePages() error {
"fmt"
"path"
"strings"
+ "sync"
)
// LayoutDescriptor describes how a layout should be chosen. This is
// TODO(bep) output improve names
type LayoutHandler struct {
hasTheme bool
+
+ mu sync.RWMutex
+ cache map[layoutCacheKey][]string
+}
+
+type layoutCacheKey struct {
+ d LayoutDescriptor
+ layoutOverride string
+ f Format
}
func NewLayoutHandler(hasTheme bool) *LayoutHandler {
- return &LayoutHandler{hasTheme: hasTheme}
+ return &LayoutHandler{hasTheme: hasTheme, cache: make(map[layoutCacheKey][]string)}
}
const (
)
func (l *LayoutHandler) For(d LayoutDescriptor, layoutOverride string, f Format) []string {
+
+ // We will get lots of requests for the same layouts, so avoid recalculations.
+ key := layoutCacheKey{d, layoutOverride, f}
+ l.mu.RLock()
+ if cacheVal, found := l.cache[key]; found {
+ l.mu.RUnlock()
+ return cacheVal
+ }
+ l.mu.RUnlock()
+
var layouts []string
layout := d.Layout
return layoutsWithThemeLayouts
}
+ l.mu.Lock()
+ l.cache[key] = layouts
+ l.mu.Unlock()
+
return layouts
}
}
})
}
+
+}
+
+func BenchmarkLayout(b *testing.B) {
+ descriptor := LayoutDescriptor{Kind: "taxonomyTerm", Section: "categories"}
+ l := NewLayoutHandler(false)
+
+ for i := 0; i < b.N; i++ {
+ layouts := l.For(descriptor, "", HTMLType)
+ require.NotEmpty(b, layouts)
+ }
}