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 page_generate
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"reflect"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/common/maps"
|
|
|
|
|
|
|
|
"github.com/gohugoio/hugo/codegen"
|
|
|
|
"github.com/gohugoio/hugo/resources/page"
|
|
|
|
"github.com/gohugoio/hugo/source"
|
|
|
|
)
|
|
|
|
|
|
|
|
const header = `// 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.
|
|
|
|
|
|
|
|
// This file is autogenerated.
|
|
|
|
`
|
|
|
|
|
|
|
|
var (
|
|
|
|
fileInterfaceDeprecated = reflect.TypeOf((*source.FileWithoutOverlap)(nil)).Elem()
|
|
|
|
pageInterfaceDeprecated = reflect.TypeOf((*page.DeprecatedWarningPageMethods)(nil)).Elem()
|
|
|
|
pageInterface = reflect.TypeOf((*page.Page)(nil)).Elem()
|
|
|
|
|
|
|
|
packageDir = filepath.FromSlash("resources/page")
|
|
|
|
)
|
|
|
|
|
|
|
|
func Generate(c *codegen.Inspector) error {
|
|
|
|
if err := generateMarshalJSON(c); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to generate JSON marshaler")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := generateDeprecatedWrappers(c); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to generate deprecate wrappers")
|
|
|
|
}
|
|
|
|
|
2019-03-25 13:18:34 -04:00
|
|
|
if err := generateFileIsZeroWrappers(c); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to generate file wrappers")
|
|
|
|
}
|
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateMarshalJSON(c *codegen.Inspector) error {
|
|
|
|
filename := filepath.Join(c.ProjectRootDir, packageDir, "page_marshaljson.autogen.go")
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
includes := []reflect.Type{pageInterface}
|
|
|
|
|
|
|
|
// Exclude these methods
|
|
|
|
excludes := []reflect.Type{
|
|
|
|
// We need to eveluate the deprecated vs JSON in the future,
|
|
|
|
// but leave them out for now.
|
|
|
|
pageInterfaceDeprecated,
|
|
|
|
|
|
|
|
// Leave this out for now. We need to revisit the author issue.
|
|
|
|
reflect.TypeOf((*page.AuthorProvider)(nil)).Elem(),
|
|
|
|
|
|
|
|
// navigation.PageMenus
|
|
|
|
|
|
|
|
// Prevent loops.
|
|
|
|
reflect.TypeOf((*page.SitesProvider)(nil)).Elem(),
|
|
|
|
reflect.TypeOf((*page.Positioner)(nil)).Elem(),
|
|
|
|
|
|
|
|
reflect.TypeOf((*page.ChildCareProvider)(nil)).Elem(),
|
|
|
|
reflect.TypeOf((*page.TreeProvider)(nil)).Elem(),
|
|
|
|
reflect.TypeOf((*page.InSectionPositioner)(nil)).Elem(),
|
|
|
|
reflect.TypeOf((*page.PaginatorProvider)(nil)).Elem(),
|
|
|
|
reflect.TypeOf((*maps.Scratcher)(nil)).Elem(),
|
|
|
|
}
|
|
|
|
|
|
|
|
methods := c.MethodsFromTypes(
|
|
|
|
includes,
|
|
|
|
excludes)
|
|
|
|
|
|
|
|
if len(methods) == 0 {
|
|
|
|
return errors.New("no methods found")
|
|
|
|
}
|
|
|
|
|
2019-03-25 02:54:10 -04:00
|
|
|
marshalJSON, pkgImports := methods.ToMarshalJSON(
|
|
|
|
"Page",
|
|
|
|
"github.com/gohugoio/hugo/resources/page",
|
|
|
|
// Exclusion regexps. Matches method names.
|
|
|
|
`\bPage\b`,
|
|
|
|
)
|
2019-01-02 06:33:26 -05:00
|
|
|
|
|
|
|
fmt.Fprintf(f, `%s
|
|
|
|
|
|
|
|
package page
|
|
|
|
|
|
|
|
%s
|
|
|
|
|
|
|
|
|
|
|
|
%s
|
|
|
|
|
|
|
|
|
|
|
|
`, header, importsString(pkgImports), marshalJSON)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateDeprecatedWrappers(c *codegen.Inspector) error {
|
|
|
|
filename := filepath.Join(c.ProjectRootDir, packageDir, "page_wrappers.autogen.go")
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
// Generate a wrapper for deprecated page methods
|
|
|
|
|
|
|
|
reasons := map[string]string{
|
|
|
|
"IsDraft": "Use .Draft.",
|
|
|
|
"Hugo": "Use the global hugo function.",
|
|
|
|
"LanguagePrefix": "Use .Site.LanguagePrefix.",
|
|
|
|
"GetParam": "Use .Param or .Params.myParam.",
|
|
|
|
"RSSLink": `Use the Output Format's link, e.g. something like:
|
2019-04-09 10:47:29 -04:00
|
|
|
{{ with .OutputFormats.Get "RSS" }}{{ .RelPermalink }}{{ end }}`,
|
2019-01-02 06:33:26 -05:00
|
|
|
"URL": "Use .Permalink or .RelPermalink. If what you want is the front matter URL value, use .Params.url",
|
|
|
|
}
|
|
|
|
|
|
|
|
deprecated := func(name string, tp reflect.Type) string {
|
|
|
|
var alternative string
|
|
|
|
if tp == fileInterfaceDeprecated {
|
|
|
|
alternative = "Use .File." + name
|
|
|
|
} else {
|
|
|
|
var found bool
|
|
|
|
alternative, found = reasons[name]
|
|
|
|
if !found {
|
|
|
|
panic(fmt.Sprintf("no deprecated reason found for %q", name))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-25 14:04:49 -05:00
|
|
|
return fmt.Sprintf("helpers.Deprecated(%q, %q, false)", "Page."+name, alternative)
|
2019-01-02 06:33:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
var buff bytes.Buffer
|
|
|
|
|
|
|
|
methods := c.MethodsFromTypes([]reflect.Type{fileInterfaceDeprecated, pageInterfaceDeprecated}, nil)
|
|
|
|
|
|
|
|
for _, m := range methods {
|
|
|
|
fmt.Fprint(&buff, m.Declaration("*pageDeprecated"))
|
|
|
|
fmt.Fprintln(&buff, " {")
|
|
|
|
fmt.Fprintf(&buff, "\t%s\n", deprecated(m.Name, m.Owner))
|
|
|
|
fmt.Fprintf(&buff, "\t%s\n}\n", m.Delegate("p", "p"))
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pkgImports := append(methods.Imports(), "github.com/gohugoio/hugo/helpers")
|
|
|
|
|
|
|
|
fmt.Fprintf(f, `%s
|
|
|
|
|
|
|
|
package page
|
|
|
|
|
|
|
|
%s
|
|
|
|
// NewDeprecatedWarningPage adds deprecation warnings to the given implementation.
|
|
|
|
func NewDeprecatedWarningPage(p DeprecatedWarningPageMethods) DeprecatedWarningPageMethods {
|
|
|
|
return &pageDeprecated{p: p}
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageDeprecated struct {
|
|
|
|
p DeprecatedWarningPageMethods
|
|
|
|
}
|
|
|
|
|
|
|
|
%s
|
|
|
|
|
|
|
|
`, header, importsString(pkgImports), buff.String())
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-03-25 13:18:34 -04:00
|
|
|
func generateFileIsZeroWrappers(c *codegen.Inspector) error {
|
|
|
|
filename := filepath.Join(c.ProjectRootDir, packageDir, "zero_file.autogen.go")
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
// Generate warnings for zero file access
|
|
|
|
|
|
|
|
warning := func(name string, tp reflect.Type) string {
|
|
|
|
msg := fmt.Sprintf(".File.%s on zero object. Wrap it in if or with: {{ with .File }}{{ .%s }}{{ end }}", name, name)
|
|
|
|
|
|
|
|
return fmt.Sprintf("z.log.Println(%q)", msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
var buff bytes.Buffer
|
|
|
|
|
|
|
|
methods := c.MethodsFromTypes([]reflect.Type{reflect.TypeOf((*source.File)(nil)).Elem()}, nil)
|
|
|
|
|
|
|
|
for _, m := range methods {
|
|
|
|
if m.Name == "IsZero" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fmt.Fprint(&buff, m.DeclarationNamed("zeroFile"))
|
|
|
|
fmt.Fprintln(&buff, " {")
|
|
|
|
fmt.Fprintf(&buff, "\t%s\n", warning(m.Name, m.Owner))
|
|
|
|
if len(m.Out) > 0 {
|
|
|
|
fmt.Fprintln(&buff, "\treturn")
|
|
|
|
}
|
|
|
|
fmt.Fprintln(&buff, "}")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pkgImports := append(methods.Imports(), "github.com/gohugoio/hugo/helpers", "github.com/gohugoio/hugo/source")
|
|
|
|
|
|
|
|
fmt.Fprintf(f, `%s
|
|
|
|
|
|
|
|
package page
|
|
|
|
|
|
|
|
%s
|
|
|
|
|
|
|
|
// ZeroFile represents a zero value of source.File with warnings if invoked.
|
|
|
|
type zeroFile struct {
|
|
|
|
log *helpers.DistinctLogger
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewZeroFile(log *helpers.DistinctLogger) source.File {
|
|
|
|
return zeroFile{log: log}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (zeroFile) IsZero() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
%s
|
|
|
|
|
|
|
|
`, header, importsString(pkgImports), buff.String())
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-02 06:33:26 -05:00
|
|
|
func importsString(imps []string) string {
|
|
|
|
if len(imps) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(imps) == 1 {
|
|
|
|
return fmt.Sprintf("import %q", imps[0])
|
|
|
|
}
|
|
|
|
|
|
|
|
impsStr := "import (\n"
|
|
|
|
for _, imp := range imps {
|
|
|
|
impsStr += fmt.Sprintf("%q\n", imp)
|
|
|
|
}
|
|
|
|
|
|
|
|
return impsStr + ")"
|
|
|
|
}
|