`)
}
+// Issue #588
+func TestIncludeCachedRecursionShortcode(t *testing.T) {
+ t.Parallel()
+
+ files := `
+-- config.toml --
+baseURL = 'http://example.com/'
+-- content/_index.md --
+---
+title: "Index"
+---
+{{< short >}}
+-- layouts/index.html --
+{{ partials.IncludeCached "p1.html" . }}
+-- layouts/partials/p1.html --
+{{ .Content }}
+{{ partials.IncludeCached "p2.html" . }}
+-- layouts/partials/p2.html --
+-- layouts/shortcodes/short.html --
+SHORT
+{{ partials.IncludeCached "p2.html" . }}
+P2
+
+ `
+
+ b := hugolib.NewIntegrationTestBuilder(
+ hugolib.IntegrationTestConfig{
+ T: t,
+ TxtarString: files,
+ },
+ ).Build()
+
+ b.AssertFileContent("public/index.html", `
+SHORT
+P2
+`)
+}
+
func TestIncludeCacheHints(t *testing.T) {
t.Parallel()
}
}()
- // We may already have a write lock.
- hasLock := tpl.GetHasLockFromContext(ctx)
-
- if !hasLock {
- ns.cachedPartials.RLock()
- }
+ ns.cachedPartials.RLock()
p, ok := ns.cachedPartials.p[key]
- if !hasLock {
- ns.cachedPartials.RUnlock()
- }
+ ns.cachedPartials.RUnlock()
if ok {
if ns.deps.Metrics != nil {
ns.deps.Metrics.TrackValue(key.templateName(), p, true)
- // The templates that gets executed is measued in Execute.
+ // The templates that gets executed is measured in Execute.
// We need to track the time spent in the cache to
// get the totals correct.
ns.deps.Metrics.MeasureSince(key.templateName(), start)
return p, nil
}
- if !hasLock {
- ns.cachedPartials.Lock()
- defer ns.cachedPartials.Unlock()
- ctx = tpl.SetHasLockInContext(ctx, true)
- }
-
- var name string
- name, p, err = ns.include(ctx, key.name, context)
+ // This needs to be done outside the lock.
+ // See #9588
+ _, p, err = ns.include(ctx, key.name, context)
if err != nil {
return nil, err
}
+ ns.cachedPartials.Lock()
+ defer ns.cachedPartials.Unlock()
+ // Double-check.
+ if p2, ok := ns.cachedPartials.p[key]; ok {
+ if ns.deps.Metrics != nil {
+ ns.deps.Metrics.TrackValue(key.templateName(), p, true)
+ ns.deps.Metrics.MeasureSince(key.templateName(), start)
+ }
+ return p2, nil
+
+ }
if ns.deps.Metrics != nil {
- ns.deps.Metrics.TrackValue(name, p, false)
+ ns.deps.Metrics.TrackValue(key.templateName(), p, false)
}
+
ns.cachedPartials.p[key] = p
return p, nil