Fix surprise OutputFormat.Rel overwriting
authorPaul Gottschling <paul.gottschling@gmail.com>
Mon, 3 Jan 2022 16:17:51 +0000 (11:17 -0500)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 4 Jan 2022 09:38:38 +0000 (10:38 +0100)
In page.NewOutputFormat, we take an output.Format f and use it to
create a page.OutputFormat. If the format is canonical, we assign
the final OutputFormat's Rel to "canonical" rather than using
f.Rel. However, this leads to unexpected behavior for custom
output formats, where a user can define a "rel" for a format
via the config file.

For example, the standard for "humans.txt" files requires using
rel="author" in HTML "link" elements. Meanwhile, humans.txt is
usually the only format used for its content. As a result, for
Hugo configurations that define a humans.txt custom output format,
Hugo will render "link" elements to content in this format with
rel="canonical," rather than "author" as required by the standard.

This commit changes page.NewOutputFormat to check whether a given
format is user defined and, if so, skips assigning Rel to
"canonical," even if isCanonical is true.

Fixes #8030

hugolib/site_output_test.go
resources/page/page_outputformat.go

index f3455f3692c6de19f9777f07693b99db4f2b597e..815625ff10dc716119dd6ca7ae003124461c5b23 100644 (file)
@@ -324,6 +324,36 @@ baseName = "customdelimbase"
        c.Assert(outputs.Get("CUS").RelPermalink(), qt.Equals, "/blog/customdelimbase_del")
 }
 
+// Issue 8030
+func TestGetOutputFormatRel(t *testing.T) {
+       b := newTestSitesBuilder(t).
+               WithSimpleConfigFileAndSettings(map[string]interface{}{
+                       "outputFormats": map[string]interface{}{
+                               "humansTXT": map[string]interface{}{
+                                       "name":        "HUMANS",
+                                       "mediaType":   "text/plain",
+                                       "baseName":    "humans",
+                                       "isPlainText": true,
+                                       "rel":         "author",
+                               },
+                       },
+               }).WithTemplates("index.html", `
+{{- with ($.Site.GetPage "humans").OutputFormats.Get "humans" -}}
+<link rel="{{ .Rel }}" type="{{ .MediaType.String }}" href="{{ .Permalink }}">
+{{- end -}}
+`).WithContent("humans.md", `---
+outputs:
+- HUMANS
+---
+This is my content.
+`)
+
+       b.Build(BuildCfg{})
+       b.AssertFileContent("public/index.html", `
+<link rel="author" type="text/plain" href="/humans.txt">
+`)
+}
+
 func TestCreateSiteOutputFormats(t *testing.T) {
        t.Run("Basic", func(t *testing.T) {
                c := qt.New(t)
index 9eed8241e6ddd14918d838c2d301097893c228f3..44f290025170217e83d2e0f6345fa4b509b82e72 100644 (file)
@@ -66,8 +66,18 @@ func (o OutputFormat) RelPermalink() string {
 }
 
 func NewOutputFormat(relPermalink, permalink string, isCanonical bool, f output.Format) OutputFormat {
+       isUserConfigured := true
+       for _, d := range output.DefaultFormats {
+               if strings.EqualFold(d.Name, f.Name) {
+                       isUserConfigured = false
+               }
+       }
        rel := f.Rel
-       if isCanonical {
+       // If the output format is the canonical format for the content, we want
+       // to specify this in the "rel" attribute of an HTML "link" element.
+       // However, for custom output formats, we don't want to surprise users by
+       // overwriting "rel"
+       if isCanonical && !isUserConfigured {
                rel = "canonical"
        }
        return OutputFormat{Rel: rel, Format: f, relPermalink: relPermalink, permalink: permalink}