mirror of
https://github.com/gohugoio/hugo.git
synced 2024-12-01 13:08:47 -05:00
322d19a81f
The motivation behind this is not to make it easier to publish Markdown files, as that sounds unusual. This is mainly meant for shortcodes that produces Markdown to be inlined. You would do this by creating shortcodes with `*.md` suffix (e.g. `layouts/shortcodes/myshortcode.md`). This output format is defined as plain text, and will use Go's much more lenient text template parser. Updates #9821
412 lines
10 KiB
Go
412 lines
10 KiB
Go
// 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 output
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/mitchellh/mapstructure"
|
|
|
|
"github.com/gohugoio/hugo/media"
|
|
)
|
|
|
|
// Format represents an output representation, usually to a file on disk.
|
|
type Format struct {
|
|
// The Name is used as an identifier. Internal output formats (i.e. HTML and RSS)
|
|
// can be overridden by providing a new definition for those types.
|
|
Name string `json:"name"`
|
|
|
|
MediaType media.Type `json:"-"`
|
|
|
|
// Must be set to a value when there are two or more conflicting mediatype for the same resource.
|
|
Path string `json:"path"`
|
|
|
|
// The base output file name used when not using "ugly URLs", defaults to "index".
|
|
BaseName string `json:"baseName"`
|
|
|
|
// The value to use for rel links
|
|
//
|
|
// See https://www.w3schools.com/tags/att_link_rel.asp
|
|
//
|
|
// AMP has a special requirement in this department, see:
|
|
// https://www.ampproject.org/docs/guides/deploy/discovery
|
|
// I.e.:
|
|
// <link rel="amphtml" href="https://www.example.com/url/to/amp/document.html">
|
|
Rel string `json:"rel"`
|
|
|
|
// The protocol to use, i.e. "webcal://". Defaults to the protocol of the baseURL.
|
|
Protocol string `json:"protocol"`
|
|
|
|
// IsPlainText decides whether to use text/template or html/template
|
|
// as template parser.
|
|
IsPlainText bool `json:"isPlainText"`
|
|
|
|
// IsHTML returns whether this format is int the HTML family. This includes
|
|
// HTML, AMP etc. This is used to decide when to create alias redirects etc.
|
|
IsHTML bool `json:"isHTML"`
|
|
|
|
// Enable to ignore the global uglyURLs setting.
|
|
NoUgly bool `json:"noUgly"`
|
|
|
|
// Enable if it doesn't make sense to include this format in an alternative
|
|
// format listing, CSS being one good example.
|
|
// Note that we use the term "alternative" and not "alternate" here, as it
|
|
// does not necessarily replace the other format, it is an alternative representation.
|
|
NotAlternative bool `json:"notAlternative"`
|
|
|
|
// Setting this will make this output format control the value of
|
|
// .Permalink and .RelPermalink for a rendered Page.
|
|
// If not set, these values will point to the main (first) output format
|
|
// configured. That is probably the behaviour you want in most situations,
|
|
// as you probably don't want to link back to the RSS version of a page, as an
|
|
// example. AMP would, however, be a good example of an output format where this
|
|
// behaviour is wanted.
|
|
Permalinkable bool `json:"permalinkable"`
|
|
|
|
// Setting this to a non-zero value will be used as the first sort criteria.
|
|
Weight int `json:"weight"`
|
|
}
|
|
|
|
// An ordered list of built-in output formats.
|
|
var (
|
|
AMPFormat = Format{
|
|
Name: "AMP",
|
|
MediaType: media.HTMLType,
|
|
BaseName: "index",
|
|
Path: "amp",
|
|
Rel: "amphtml",
|
|
IsHTML: true,
|
|
Permalinkable: true,
|
|
// See https://www.ampproject.org/learn/overview/
|
|
}
|
|
|
|
CalendarFormat = Format{
|
|
Name: "Calendar",
|
|
MediaType: media.CalendarType,
|
|
IsPlainText: true,
|
|
Protocol: "webcal://",
|
|
BaseName: "index",
|
|
Rel: "alternate",
|
|
}
|
|
|
|
CSSFormat = Format{
|
|
Name: "CSS",
|
|
MediaType: media.CSSType,
|
|
BaseName: "styles",
|
|
IsPlainText: true,
|
|
Rel: "stylesheet",
|
|
NotAlternative: true,
|
|
}
|
|
CSVFormat = Format{
|
|
Name: "CSV",
|
|
MediaType: media.CSVType,
|
|
BaseName: "index",
|
|
IsPlainText: true,
|
|
Rel: "alternate",
|
|
}
|
|
|
|
HTMLFormat = Format{
|
|
Name: "HTML",
|
|
MediaType: media.HTMLType,
|
|
BaseName: "index",
|
|
Rel: "canonical",
|
|
IsHTML: true,
|
|
Permalinkable: true,
|
|
|
|
// Weight will be used as first sort criteria. HTML will, by default,
|
|
// be rendered first, but set it to 10 so it's easy to put one above it.
|
|
Weight: 10,
|
|
}
|
|
|
|
MarkdownFormat = Format{
|
|
Name: "MARKDOWN",
|
|
MediaType: media.MarkdownType,
|
|
BaseName: "index",
|
|
Rel: "alternate",
|
|
IsPlainText: true,
|
|
}
|
|
|
|
JSONFormat = Format{
|
|
Name: "JSON",
|
|
MediaType: media.JSONType,
|
|
BaseName: "index",
|
|
IsPlainText: true,
|
|
Rel: "alternate",
|
|
}
|
|
|
|
WebAppManifestFormat = Format{
|
|
Name: "WebAppManifest",
|
|
MediaType: media.WebAppManifestType,
|
|
BaseName: "manifest",
|
|
IsPlainText: true,
|
|
NotAlternative: true,
|
|
Rel: "manifest",
|
|
}
|
|
|
|
RobotsTxtFormat = Format{
|
|
Name: "ROBOTS",
|
|
MediaType: media.TextType,
|
|
BaseName: "robots",
|
|
IsPlainText: true,
|
|
Rel: "alternate",
|
|
}
|
|
|
|
RSSFormat = Format{
|
|
Name: "RSS",
|
|
MediaType: media.RSSType,
|
|
BaseName: "index",
|
|
NoUgly: true,
|
|
Rel: "alternate",
|
|
}
|
|
|
|
SitemapFormat = Format{
|
|
Name: "Sitemap",
|
|
MediaType: media.XMLType,
|
|
BaseName: "sitemap",
|
|
NoUgly: true,
|
|
Rel: "sitemap",
|
|
}
|
|
)
|
|
|
|
// DefaultFormats contains the default output formats supported by Hugo.
|
|
var DefaultFormats = Formats{
|
|
AMPFormat,
|
|
CalendarFormat,
|
|
CSSFormat,
|
|
CSVFormat,
|
|
HTMLFormat,
|
|
JSONFormat,
|
|
MarkdownFormat,
|
|
WebAppManifestFormat,
|
|
RobotsTxtFormat,
|
|
RSSFormat,
|
|
SitemapFormat,
|
|
}
|
|
|
|
func init() {
|
|
sort.Sort(DefaultFormats)
|
|
}
|
|
|
|
// Formats is a slice of Format.
|
|
type Formats []Format
|
|
|
|
func (formats Formats) Len() int { return len(formats) }
|
|
func (formats Formats) Swap(i, j int) { formats[i], formats[j] = formats[j], formats[i] }
|
|
func (formats Formats) Less(i, j int) bool {
|
|
fi, fj := formats[i], formats[j]
|
|
if fi.Weight == fj.Weight {
|
|
return fi.Name < fj.Name
|
|
}
|
|
|
|
if fj.Weight == 0 {
|
|
return true
|
|
}
|
|
|
|
return fi.Weight > 0 && fi.Weight < fj.Weight
|
|
}
|
|
|
|
// GetBySuffix gets a output format given as suffix, e.g. "html".
|
|
// It will return false if no format could be found, or if the suffix given
|
|
// is ambiguous.
|
|
// The lookup is case insensitive.
|
|
func (formats Formats) GetBySuffix(suffix string) (f Format, found bool) {
|
|
for _, ff := range formats {
|
|
for _, suffix2 := range ff.MediaType.Suffixes() {
|
|
if strings.EqualFold(suffix, suffix2) {
|
|
if found {
|
|
// ambiguous
|
|
found = false
|
|
return
|
|
}
|
|
f = ff
|
|
found = true
|
|
}
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// GetByName gets a format by its identifier name.
|
|
func (formats Formats) GetByName(name string) (f Format, found bool) {
|
|
for _, ff := range formats {
|
|
if strings.EqualFold(name, ff.Name) {
|
|
f = ff
|
|
found = true
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// GetByNames gets a list of formats given a list of identifiers.
|
|
func (formats Formats) GetByNames(names ...string) (Formats, error) {
|
|
var types []Format
|
|
|
|
for _, name := range names {
|
|
tpe, ok := formats.GetByName(name)
|
|
if !ok {
|
|
return types, fmt.Errorf("OutputFormat with key %q not found", name)
|
|
}
|
|
types = append(types, tpe)
|
|
}
|
|
return types, nil
|
|
}
|
|
|
|
// FromFilename gets a Format given a filename.
|
|
func (formats Formats) FromFilename(filename string) (f Format, found bool) {
|
|
// mytemplate.amp.html
|
|
// mytemplate.html
|
|
// mytemplate
|
|
var ext, outFormat string
|
|
|
|
parts := strings.Split(filename, ".")
|
|
if len(parts) > 2 {
|
|
outFormat = parts[1]
|
|
ext = parts[2]
|
|
} else if len(parts) > 1 {
|
|
ext = parts[1]
|
|
}
|
|
|
|
if outFormat != "" {
|
|
return formats.GetByName(outFormat)
|
|
}
|
|
|
|
if ext != "" {
|
|
f, found = formats.GetBySuffix(ext)
|
|
if !found && len(parts) == 2 {
|
|
// For extensionless output formats (e.g. Netlify's _redirects)
|
|
// we must fall back to using the extension as format lookup.
|
|
f, found = formats.GetByName(ext)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// DecodeFormats takes a list of output format configurations and merges those,
|
|
// in the order given, with the Hugo defaults as the last resort.
|
|
func DecodeFormats(mediaTypes media.Types, maps ...map[string]any) (Formats, error) {
|
|
f := make(Formats, len(DefaultFormats))
|
|
copy(f, DefaultFormats)
|
|
|
|
for _, m := range maps {
|
|
for k, v := range m {
|
|
found := false
|
|
for i, vv := range f {
|
|
if strings.EqualFold(k, vv.Name) {
|
|
// Merge it with the existing
|
|
if err := decode(mediaTypes, v, &f[i]); err != nil {
|
|
return f, err
|
|
}
|
|
found = true
|
|
}
|
|
}
|
|
if !found {
|
|
var newOutFormat Format
|
|
newOutFormat.Name = k
|
|
if err := decode(mediaTypes, v, &newOutFormat); err != nil {
|
|
return f, err
|
|
}
|
|
|
|
// We need values for these
|
|
if newOutFormat.BaseName == "" {
|
|
newOutFormat.BaseName = "index"
|
|
}
|
|
if newOutFormat.Rel == "" {
|
|
newOutFormat.Rel = "alternate"
|
|
}
|
|
|
|
f = append(f, newOutFormat)
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
sort.Sort(f)
|
|
|
|
return f, nil
|
|
}
|
|
|
|
func decode(mediaTypes media.Types, input any, output *Format) error {
|
|
config := &mapstructure.DecoderConfig{
|
|
Metadata: nil,
|
|
Result: output,
|
|
WeaklyTypedInput: true,
|
|
DecodeHook: func(a reflect.Type, b reflect.Type, c any) (any, error) {
|
|
if a.Kind() == reflect.Map {
|
|
dataVal := reflect.Indirect(reflect.ValueOf(c))
|
|
for _, key := range dataVal.MapKeys() {
|
|
keyStr, ok := key.Interface().(string)
|
|
if !ok {
|
|
// Not a string key
|
|
continue
|
|
}
|
|
if strings.EqualFold(keyStr, "mediaType") {
|
|
// If mediaType is a string, look it up and replace it
|
|
// in the map.
|
|
vv := dataVal.MapIndex(key)
|
|
vvi := vv.Interface()
|
|
|
|
switch vviv := vvi.(type) {
|
|
case media.Type:
|
|
// OK
|
|
case string:
|
|
mediaType, found := mediaTypes.GetByType(vviv)
|
|
if !found {
|
|
return c, fmt.Errorf("media type %q not found", vviv)
|
|
}
|
|
dataVal.SetMapIndex(key, reflect.ValueOf(mediaType))
|
|
default:
|
|
return nil, fmt.Errorf("invalid output format configuration; wrong type for media type, expected string (e.g. text/html), got %T", vvi)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return c, nil
|
|
},
|
|
}
|
|
|
|
decoder, err := mapstructure.NewDecoder(config)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = decoder.Decode(input); err != nil {
|
|
return fmt.Errorf("failed to decode output format configuration: %w", err)
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// BaseFilename returns the base filename of f including an extension (ie.
|
|
// "index.xml").
|
|
func (f Format) BaseFilename() string {
|
|
return f.BaseName + f.MediaType.FirstSuffix.FullSuffix
|
|
}
|
|
|
|
// MarshalJSON returns the JSON encoding of f.
|
|
func (f Format) MarshalJSON() ([]byte, error) {
|
|
type Alias Format
|
|
return json.Marshal(&struct {
|
|
MediaType string `json:"mediaType"`
|
|
Alias
|
|
}{
|
|
MediaType: f.MediaType.String(),
|
|
Alias: (Alias)(f),
|
|
})
|
|
}
|