2019-01-02 06:33:26 -05:00
|
|
|
// Copyright 2019 The Hugo Authors. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package hugolib
|
|
|
|
|
|
|
|
import (
|
|
|
|
"html/template"
|
|
|
|
"strings"
|
|
|
|
|
2022-02-17 07:04:00 -05:00
|
|
|
"go.uber.org/atomic"
|
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
"github.com/gohugoio/hugo/common/hugo"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/common/maps"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/output"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/lazy"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/resources/page"
|
|
|
|
)
|
|
|
|
|
|
|
|
func newPageBase(metaProvider *pageMeta) (*pageState, error) {
|
|
|
|
if metaProvider.s == nil {
|
|
|
|
panic("must provide a Site")
|
|
|
|
}
|
|
|
|
|
|
|
|
s := metaProvider.s
|
|
|
|
|
|
|
|
ps := &pageState{
|
2022-02-17 07:04:00 -05:00
|
|
|
pageOutput: nopPageOutput,
|
|
|
|
pageOutputTemplateVariationsState: atomic.NewUint32(0),
|
2019-01-02 06:33:26 -05:00
|
|
|
pageCommon: &pageCommon{
|
|
|
|
FileProvider: metaProvider,
|
|
|
|
AuthorProvider: metaProvider,
|
|
|
|
Scratcher: maps.NewScratcher(),
|
2022-02-22 08:42:33 -05:00
|
|
|
store: maps.NewScratch(),
|
2019-01-02 06:33:26 -05:00
|
|
|
Positioner: page.NopPage,
|
|
|
|
InSectionPositioner: page.NopPage,
|
|
|
|
ResourceMetaProvider: metaProvider,
|
|
|
|
ResourceParamsProvider: metaProvider,
|
|
|
|
PageMetaProvider: metaProvider,
|
|
|
|
RelatedKeywordsProvider: metaProvider,
|
|
|
|
OutputFormatsProvider: page.NopPage,
|
2020-02-25 15:40:02 -05:00
|
|
|
ResourceTypeProvider: pageTypesProvider,
|
|
|
|
MediaTypeProvider: pageTypesProvider,
|
2019-01-02 06:33:26 -05:00
|
|
|
RefProvider: page.NopPage,
|
|
|
|
ShortcodeInfoProvider: page.NopPage,
|
|
|
|
LanguageProvider: s,
|
2019-04-17 04:36:36 -04:00
|
|
|
pagePages: &pagePages{},
|
2019-01-02 06:33:26 -05:00
|
|
|
|
|
|
|
InternalDependencies: s,
|
|
|
|
init: lazy.New(),
|
|
|
|
m: metaProvider,
|
2019-09-10 05:26:34 -04:00
|
|
|
s: s,
|
|
|
|
},
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
siteAdapter := pageSiteAdapter{s: s, p: ps}
|
|
|
|
|
|
|
|
ps.pageMenus = &pageMenus{p: ps}
|
|
|
|
ps.PageMenusProvider = ps.pageMenus
|
|
|
|
ps.GetPageProvider = siteAdapter
|
|
|
|
ps.GitInfoProvider = ps
|
|
|
|
ps.TranslationsProvider = ps
|
|
|
|
ps.ResourceDataProvider = &pageData{pageState: ps}
|
|
|
|
ps.RawContentProvider = ps
|
|
|
|
ps.ChildCareProvider = ps
|
|
|
|
ps.TreeProvider = pageTree{p: ps}
|
|
|
|
ps.Eqer = ps
|
|
|
|
ps.TranslationKeyProvider = ps
|
|
|
|
ps.ShortcodeInfoProvider = ps
|
|
|
|
ps.AlternativeOutputFormatsProvider = ps
|
|
|
|
|
|
|
|
return ps, nil
|
|
|
|
}
|
|
|
|
|
2019-09-10 05:26:34 -04:00
|
|
|
func newPageBucket(p *pageState) *pagesMapBucket {
|
2020-03-22 08:42:39 -04:00
|
|
|
return &pagesMapBucket{owner: p, pagesMapBucketPages: &pagesMapBucketPages{}}
|
2019-09-10 05:26:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func newPageFromMeta(
|
|
|
|
n *contentNode,
|
|
|
|
parentBucket *pagesMapBucket,
|
2022-03-17 17:03:27 -04:00
|
|
|
meta map[string]any,
|
2019-09-10 05:26:34 -04:00
|
|
|
metaProvider *pageMeta) (*pageState, error) {
|
2019-03-25 13:18:34 -04:00
|
|
|
if metaProvider.f == nil {
|
2021-06-07 10:36:48 -04:00
|
|
|
metaProvider.f = page.NewZeroFile(metaProvider.s.LogDistinct)
|
2019-03-25 13:18:34 -04:00
|
|
|
}
|
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
ps, err := newPageBase(metaProvider)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-09-10 05:26:34 -04:00
|
|
|
bucket := parentBucket
|
2019-08-09 04:05:22 -04:00
|
|
|
|
2019-09-10 05:26:34 -04:00
|
|
|
if ps.IsNode() {
|
|
|
|
ps.bucket = newPageBucket(ps)
|
|
|
|
}
|
2019-08-09 04:05:22 -04:00
|
|
|
|
2019-09-10 05:26:34 -04:00
|
|
|
if meta != nil || parentBucket != nil {
|
|
|
|
if err := metaProvider.setMetadata(bucket, ps, meta); err != nil {
|
|
|
|
return nil, ps.wrapError(err)
|
|
|
|
}
|
2019-08-09 04:05:22 -04:00
|
|
|
}
|
|
|
|
|
2019-09-10 05:26:34 -04:00
|
|
|
if err := metaProvider.applyDefaultValues(n); err != nil {
|
|
|
|
return nil, err
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
|
|
|
|
2022-03-17 17:03:27 -04:00
|
|
|
ps.init.Add(func() (any, error) {
|
2019-01-02 06:33:26 -05:00
|
|
|
pp, err := newPagePaths(metaProvider.s, ps, metaProvider)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
makeOut := func(f output.Format, render bool) *pageOutput {
|
2019-11-27 07:42:36 -05:00
|
|
|
return newPageOutput(ps, pp, f, render)
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
|
|
|
|
2020-02-21 03:02:07 -05:00
|
|
|
shouldRenderPage := !ps.m.noRender()
|
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
if ps.m.standalone {
|
2020-02-21 03:02:07 -05:00
|
|
|
ps.pageOutput = makeOut(ps.m.outputFormats()[0], shouldRenderPage)
|
2019-01-02 06:33:26 -05:00
|
|
|
} else {
|
|
|
|
outputFormatsForPage := ps.m.outputFormats()
|
2019-09-10 05:26:34 -04:00
|
|
|
|
2020-02-21 03:02:07 -05:00
|
|
|
// Prepare output formats for all sites.
|
|
|
|
// We do this even if this page does not get rendered on
|
|
|
|
// its own. It may be referenced via .Site.GetPage and
|
|
|
|
// it will then need an output format.
|
|
|
|
ps.pageOutputs = make([]*pageOutput, len(ps.s.h.renderFormats))
|
|
|
|
created := make(map[string]*pageOutput)
|
|
|
|
for i, f := range ps.s.h.renderFormats {
|
|
|
|
po, found := created[f.Name]
|
|
|
|
if !found {
|
|
|
|
render := shouldRenderPage
|
|
|
|
if render {
|
|
|
|
_, render = outputFormatsForPage.GetByName(f.Name)
|
2019-09-10 05:26:34 -04:00
|
|
|
}
|
2020-02-21 03:02:07 -05:00
|
|
|
po = makeOut(f, render)
|
|
|
|
created[f.Name] = po
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
2020-02-21 03:02:07 -05:00
|
|
|
ps.pageOutputs[i] = po
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := ps.initCommonProviders(pp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return ps, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used by the legacy 404, sitemap and robots.txt rendering
|
|
|
|
func newPageStandalone(m *pageMeta, f output.Format) (*pageState, error) {
|
|
|
|
m.configuredOutputFormats = output.Formats{f}
|
|
|
|
m.standalone = true
|
2019-09-10 05:26:34 -04:00
|
|
|
p, err := newPageFromMeta(nil, nil, nil, m)
|
2019-01-02 06:33:26 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := p.initPage(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageDeprecatedWarning struct {
|
|
|
|
p *pageState
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *pageDeprecatedWarning) IsDraft() bool { return p.p.m.draft }
|
|
|
|
func (p *pageDeprecatedWarning) Hugo() hugo.Info { return p.p.s.Info.Hugo() }
|
|
|
|
func (p *pageDeprecatedWarning) LanguagePrefix() string { return p.p.s.Info.LanguagePrefix }
|
2022-03-17 17:03:27 -04:00
|
|
|
func (p *pageDeprecatedWarning) GetParam(key string) any {
|
2019-01-02 06:33:26 -05:00
|
|
|
return p.p.m.params[strings.ToLower(key)]
|
|
|
|
}
|
2020-12-02 07:23:25 -05:00
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
func (p *pageDeprecatedWarning) RSSLink() template.URL {
|
|
|
|
f := p.p.OutputFormats().Get("RSS")
|
|
|
|
if f == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return template.URL(f.Permalink())
|
|
|
|
}
|
2020-12-02 07:23:25 -05:00
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
func (p *pageDeprecatedWarning) URL() string {
|
|
|
|
if p.p.IsPage() && p.p.m.urlPaths.URL != "" {
|
|
|
|
// This is the url set in front matter
|
|
|
|
return p.p.m.urlPaths.URL
|
|
|
|
}
|
|
|
|
// Fall back to the relative permalink.
|
|
|
|
return p.p.RelPermalink()
|
|
|
|
}
|