diff --git a/commands/gen.go b/commands/gen.go
index 83b4d637c66..fad392578d6 100644
--- a/commands/gen.go
+++ b/commands/gen.go
@@ -27,6 +27,7 @@ import (
"github.com/alecthomas/chroma/v2/formatters/html"
"github.com/alecthomas/chroma/v2/styles"
"github.com/bep/simplecobra"
+ "github.com/goccy/go-yaml"
"github.com/gohugoio/hugo/common/hugo"
"github.com/gohugoio/hugo/docshelper"
"github.com/gohugoio/hugo/helpers"
@@ -35,7 +36,6 @@ import (
"github.com/gohugoio/hugo/parser"
"github.com/spf13/cobra"
"github.com/spf13/cobra/doc"
- "gopkg.in/yaml.v2"
)
func newGenCommand() *genCommand {
diff --git a/commands/server.go b/commands/server.go
index c2fee68b239..219d97acd28 100644
--- a/commands/server.go
+++ b/commands/server.go
@@ -1160,7 +1160,6 @@ func chmodFilter(dst, src os.FileInfo) bool {
}
func cleanErrorLog(content string) string {
- content = strings.ReplaceAll(content, "\n", " ")
content = logReplacer.Replace(content)
content = logDuplicateTemplateExecuteRe.ReplaceAllString(content, "")
content = logDuplicateTemplateParseRe.ReplaceAllString(content, "")
diff --git a/common/herrors/file_error.go b/common/herrors/file_error.go
index 007a06b4807..e8d7b268d33 100644
--- a/common/herrors/file_error.go
+++ b/common/herrors/file_error.go
@@ -110,11 +110,11 @@ func (fe *fileError) UpdateContent(r io.Reader, linematcher LineMatcherFn) FileE
fe.errorContext = ectx
- if ectx.Position.LineNumber > 0 {
+ if ectx.Position.LineNumber > 0 && ectx.Position.LineNumber > fe.position.LineNumber {
fe.position.LineNumber = ectx.Position.LineNumber
}
- if ectx.Position.ColumnNumber > 0 {
+ if ectx.Position.ColumnNumber > 0 && ectx.Position.ColumnNumber > fe.position.ColumnNumber {
fe.position.ColumnNumber = ectx.Position.ColumnNumber
}
@@ -177,6 +177,7 @@ func NewFileErrorFromName(err error, name string) FileError {
// Filetype is used to determine the Chroma lexer to use.
fileType, pos := extractFileTypePos(err)
pos.Filename = name
+
if fileType == "" {
_, fileType = paths.FileAndExtNoDelimiter(filepath.Clean(name))
}
@@ -234,7 +235,9 @@ func NewFileErrorFromFile(err error, filename string, fs afero.Fs, linematcher L
return NewFileErrorFromName(err, realFilename)
}
defer f.Close()
- return NewFileErrorFromName(err, realFilename).UpdateContent(f, linematcher)
+ fe := NewFileErrorFromName(err, realFilename)
+ fe = fe.UpdateContent(f, linematcher)
+ return fe
}
func openFile(filename string, fs afero.Fs) (afero.File, string, error) {
@@ -302,13 +305,9 @@ func extractFileTypePos(err error) (string, text.Position) {
}
// Look in the error message for the line number.
- for _, handle := range lineNumberExtractors {
- lno, col := handle(err)
- if lno > 0 {
- pos.ColumnNumber = col
- pos.LineNumber = lno
- break
- }
+ if lno, col := commonLineNumberExtractor(err); lno > 0 {
+ pos.ColumnNumber = col
+ pos.LineNumber = lno
}
if fileType == "" && pos.Filename != "" {
diff --git a/common/herrors/line_number_extractors.go b/common/herrors/line_number_extractors.go
index f70a2691fc3..121506bb07a 100644
--- a/common/herrors/line_number_extractors.go
+++ b/common/herrors/line_number_extractors.go
@@ -19,17 +19,27 @@ import (
)
var lineNumberExtractors = []lineNumberExtractor{
+ // YAML parse errors.
+ newLineNumberErrHandlerFromRegexp(`\[(\d+):(\d+)\]`),
+
// Template/shortcode parse errors
newLineNumberErrHandlerFromRegexp(`:(\d+):(\d*):`),
newLineNumberErrHandlerFromRegexp(`:(\d+):`),
- // YAML parse errors
- newLineNumberErrHandlerFromRegexp(`line (\d+):`),
-
// i18n bundle errors
newLineNumberErrHandlerFromRegexp(`\((\d+),\s(\d*)`),
}
+func commonLineNumberExtractor(e error) (int, int) {
+ for _, handler := range lineNumberExtractors {
+ lno, col := handler(e)
+ if lno > 0 {
+ return lno, col
+ }
+ }
+ return 0, 0
+}
+
type lineNumberExtractor func(e error) (int, int)
func newLineNumberErrHandlerFromRegexp(expression string) lineNumberExtractor {
diff --git a/common/herrors/line_number_extractors_test.go b/common/herrors/line_number_extractors_test.go
new file mode 100644
index 00000000000..7209ac9aa07
--- /dev/null
+++ b/common/herrors/line_number_extractors_test.go
@@ -0,0 +1,31 @@
+// Copyright 2024 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 herrors
+
+import (
+ "errors"
+ "testing"
+
+ qt "github.com/frankban/quicktest"
+)
+
+func TestCommonLineNumberExtractor(t *testing.T) {
+ t.Parallel()
+
+ c := qt.New(t)
+
+ lno, col := commonLineNumberExtractor(errors.New("[4:9] value is not allowed in this context"))
+ c.Assert(lno, qt.Equals, 4)
+ c.Assert(col, qt.Equals, 9)
+}
diff --git a/go.mod b/go.mod
index a82ecb5f8cc..60a70cc7626 100644
--- a/go.mod
+++ b/go.mod
@@ -125,6 +125,7 @@ require (
github.com/go-logr/stdr v1.2.2 // indirect
github.com/go-openapi/jsonpointer v0.20.2 // indirect
github.com/go-openapi/swag v0.22.8 // indirect
+ github.com/goccy/go-yaml v1.15.3 // indirect
github.com/golang-jwt/jwt/v5 v5.2.1 // indirect
github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect
github.com/google/s2a-go v0.1.8 // indirect
diff --git a/go.sum b/go.sum
index e60627e688c..b1dd00009e1 100644
--- a/go.sum
+++ b/go.sum
@@ -223,6 +223,8 @@ github.com/gobuffalo/flect v1.0.3 h1:xeWBM2nui+qnVvNM4S3foBhCAL2XgPU+a7FdpelbTq4
github.com/gobuffalo/flect v1.0.3/go.mod h1:A5msMlrHtLqh9umBSnvabjsMrCcCpAyzglnDvkbYKHs=
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
+github.com/goccy/go-yaml v1.15.3 h1:wQ4UwLFkgbSazdi+i9AVmZE3vKTktlNlI2kQqXo5L+I=
+github.com/goccy/go-yaml v1.15.3/go.mod h1:XBurs7gK8ATbW4ZPGKgcbrY1Br56PdM69F7LkFRi1kA=
github.com/gohugoio/go-i18n/v2 v2.1.3-0.20230805085216-e63c13218d0e h1:QArsSubW7eDh8APMXkByjQWvuljwPGAGQpJEFn0F0wY=
github.com/gohugoio/go-i18n/v2 v2.1.3-0.20230805085216-e63c13218d0e/go.mod h1:3Ltoo9Banwq0gOtcOwxuHG6omk+AwsQPADyw2vQYOJQ=
github.com/gohugoio/hashstructure v0.1.0 h1:kBSTMLMyTXbrJVAxaKI+wv30MMJJxn9Q8kfQtJaZ400=
diff --git a/hugolib/frontmatter_test.go b/hugolib/frontmatter_test.go
index 3a2080b0ec1..c4cbfa72ea7 100644
--- a/hugolib/frontmatter_test.go
+++ b/hugolib/frontmatter_test.go
@@ -40,7 +40,7 @@ Strings: {{ printf "%T" .Params.strings }} {{ range .Params.strings }}Strings: {
b.Build()
- b.AssertFileContent("public/post/one/index.html", "Ints: []interface {} Int: 1 (int)|Int: 2 (int)|Int: 3 (int)|")
- b.AssertFileContent("public/post/one/index.html", "Mixed: []interface {} Mixed: 1 (string)|Mixed: 2 (int)|Mixed: 3 (int)|")
+ b.AssertFileContent("public/post/one/index.html", "Ints: []interface {} Int: 1 (uint64)|Int: 2 (uint64)|Int: 3 (uint64)|")
+ b.AssertFileContent("public/post/one/index.html", "Mixed: []interface {} Mixed: 1 (string)|Mixed: 2 (uint64)|Mixed: 3 (uint64)|")
b.AssertFileContent("public/post/one/index.html", "Strings: []string Strings: 1 (string)|Strings: 2 (string)|Strings: 3 (string)|")
}
diff --git a/hugolib/hugo_sites_build_errors_test.go b/hugolib/hugo_sites_build_errors_test.go
index 71afe676772..27ab990760c 100644
--- a/hugolib/hugo_sites_build_errors_test.go
+++ b/hugolib/hugo_sites_build_errors_test.go
@@ -476,7 +476,7 @@ line 5
errors := herrors.UnwrapFileErrorsWithErrorContext(err)
b.Assert(errors, qt.HasLen, 3)
- b.Assert(errors[0].Error(), qt.Contains, filepath.FromSlash(`"/content/_index.md:1:1": "/layouts/_default/_markup/render-heading.html:2:5": execute of template failed`))
+ b.Assert(errors[0].Error(), qt.Contains, filepath.FromSlash(`"/content/_index.md:2:5": "/layouts/_default/_markup/render-heading.html:2:5": execute of template failed`))
}
func TestErrorRenderHookCodeblock(t *testing.T) {
@@ -645,3 +645,35 @@ Home.
b.Assert(err.Error(), qt.Contains, filepath.FromSlash(`/layouts/index.html:2:3`))
b.Assert(err.Error(), qt.Contains, `can't evaluate field ThisDoesNotExist`)
}
+
+func TestErrorFrontmatterYAMLSyntax(t *testing.T) {
+ t.Parallel()
+
+ files := `
+-- hugo.toml --
+-- content/_index.md --
+
+
+
+
+
+---
+line1: 'value1'
+x
+line2: 'value2'
+line3: 'value3'
+---
+`
+
+ b, err := TestE(t, files)
+
+ b.Assert(err, qt.Not(qt.IsNil))
+ b.Assert(err.Error(), qt.Contains, "> 3 |")
+ fe := herrors.UnwrapFileError(err)
+ b.Assert(fe, qt.Not(qt.IsNil))
+ pos := fe.Position()
+ b.Assert(pos.Filename, qt.Contains, filepath.FromSlash("content/_index.md"))
+ b.Assert(fe.ErrorContext(), qt.Not(qt.IsNil))
+ b.Assert(pos.LineNumber, qt.Equals, 9)
+ b.Assert(pos.ColumnNumber, qt.Equals, 1)
+}
diff --git a/hugolib/page__content.go b/hugolib/page__content.go
index 4ec91f7b5cd..b6bfe76e88f 100644
--- a/hugolib/page__content.go
+++ b/hugolib/page__content.go
@@ -283,23 +283,20 @@ func (c *contentParseInfo) parseFrontMatter(it pageparser.Item, iter *pageparser
var err error
c.frontMatter, err = metadecoders.Default.UnmarshalToMap(it.Val(source), f)
if err != nil {
- if fe, ok := err.(herrors.FileError); ok {
- pos := fe.Position()
+ fe := herrors.UnwrapFileError(err)
+ if fe == nil {
+ fe = herrors.NewFileError(err)
+ }
+ pos := fe.Position()
- // Offset the starting position of front matter.
- offset := iter.LineNumber(source) - 1
- if f == metadecoders.YAML {
- offset -= 1
- }
- pos.LineNumber += offset
+ // Offset the starting position of front matter.
+ offset := iter.LineNumber(source) - 1
- fe.UpdatePosition(pos)
- fe.SetFilename("") // It will be set later.
+ pos.LineNumber += offset
- return fe
- } else {
- return err
- }
+ fe.UpdatePosition(pos)
+ fe.SetFilename("") // It will be set later.
+ return fe
}
return nil
diff --git a/hugolib/pages_capture.go b/hugolib/pages_capture.go
index 96c2c0f96e2..0b6234d3b07 100644
--- a/hugolib/pages_capture.go
+++ b/hugolib/pages_capture.go
@@ -123,7 +123,7 @@ func (c *pagesCollector) Collect() (collectErr error) {
Handle: func(ctx context.Context, fi hugofs.FileMetaInfo) error {
numPages, numResources, err := c.m.AddFi(fi, c.buildConfig)
if err != nil {
- return hugofs.AddFileInfoToError(err, fi, c.fs)
+ return hugofs.AddFileInfoToError(err, fi, c.h.SourceFs)
}
numFilesProcessedTotal.Add(1)
numPagesProcessedTotal.Add(numPages)
diff --git a/langs/i18n/translationProvider.go b/langs/i18n/translationProvider.go
index 9ede538d233..78a134708d5 100644
--- a/langs/i18n/translationProvider.go
+++ b/langs/i18n/translationProvider.go
@@ -20,9 +20,9 @@ import (
"github.com/gohugoio/hugo/common/paths"
+ yaml "github.com/goccy/go-yaml"
"github.com/gohugoio/hugo/common/herrors"
"golang.org/x/text/language"
- yaml "gopkg.in/yaml.v2"
"github.com/gohugoio/go-i18n/v2/i18n"
"github.com/gohugoio/hugo/helpers"
diff --git a/parser/frontmatter.go b/parser/frontmatter.go
index 18e55f9ad4f..398aecc3005 100644
--- a/parser/frontmatter.go
+++ b/parser/frontmatter.go
@@ -22,8 +22,6 @@ import (
toml "github.com/pelletier/go-toml/v2"
- yaml "gopkg.in/yaml.v2"
-
xml "github.com/clbanning/mxj/v2"
)
@@ -39,7 +37,7 @@ func InterfaceToConfig(in any, format metadecoders.Format, w io.Writer) error {
switch format {
case metadecoders.YAML:
- b, err := yaml.Marshal(in)
+ b, err := metadecoders.MarshalYAML(in)
if err != nil {
return err
}
diff --git a/parser/metadecoders/decoder.go b/parser/metadecoders/decoder.go
index 5dac23f0328..145cc72fc8e 100644
--- a/parser/metadecoders/decoder.go
+++ b/parser/metadecoders/decoder.go
@@ -1,4 +1,4 @@
-// Copyright 2018 The Hugo Authors. All rights reserved.
+// Copyright 2024 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.
@@ -28,10 +28,10 @@ import (
"github.com/niklasfasching/go-org/org"
xml "github.com/clbanning/mxj/v2"
+ yaml "github.com/goccy/go-yaml"
toml "github.com/pelletier/go-toml/v2"
"github.com/spf13/afero"
"github.com/spf13/cast"
- yaml "gopkg.in/yaml.v2"
)
// Decoder provides some configuration options for the decoders.
@@ -164,35 +164,7 @@ func (d Decoder) UnmarshalTo(data []byte, f Format, v any) error {
case TOML:
err = toml.Unmarshal(data, v)
case YAML:
- err = yaml.Unmarshal(data, v)
- if err != nil {
- return toFileError(f, data, fmt.Errorf("failed to unmarshal YAML: %w", err))
- }
-
- // To support boolean keys, the YAML package unmarshals maps to
- // map[interface{}]interface{}. Here we recurse through the result
- // and change all maps to map[string]interface{} like we would've
- // gotten from `json`.
- var ptr any
- switch vv := v.(type) {
- case *map[string]any:
- ptr = *vv
- case *any:
- ptr = *vv
- default:
- // Not a map.
- }
-
- if ptr != nil {
- if mm, changed := stringifyMapKeys(ptr); changed {
- switch vv := v.(type) {
- case *map[string]any:
- *vv = mm.(map[string]any)
- case *any:
- *vv = mm
- }
- }
- }
+ return yaml.Unmarshal(data, v)
case CSV:
return d.unmarshalCSV(data, v)
@@ -269,50 +241,3 @@ func (d Decoder) unmarshalORG(data []byte, v any) error {
func toFileError(f Format, data []byte, err error) error {
return herrors.NewFileErrorFromName(err, fmt.Sprintf("_stream.%s", f)).UpdateContent(bytes.NewReader(data), nil)
}
-
-// stringifyMapKeys recurses into in and changes all instances of
-// map[interface{}]interface{} to map[string]interface{}. This is useful to
-// work around the impedance mismatch between JSON and YAML unmarshaling that's
-// described here: https://github.com/go-yaml/yaml/issues/139
-//
-// Inspired by https://github.com/stripe/stripe-mock, MIT licensed
-func stringifyMapKeys(in any) (any, bool) {
- switch in := in.(type) {
- case []any:
- for i, v := range in {
- if vv, replaced := stringifyMapKeys(v); replaced {
- in[i] = vv
- }
- }
- case map[string]any:
- for k, v := range in {
- if vv, changed := stringifyMapKeys(v); changed {
- in[k] = vv
- }
- }
- case map[any]any:
- res := make(map[string]any)
- var (
- ok bool
- err error
- )
- for k, v := range in {
- var ks string
-
- if ks, ok = k.(string); !ok {
- ks, err = cast.ToStringE(k)
- if err != nil {
- ks = fmt.Sprintf("%v", k)
- }
- }
- if vv, replaced := stringifyMapKeys(v); replaced {
- res[ks] = vv
- } else {
- res[ks] = v
- }
- }
- return res, true
- }
-
- return nil, false
-}
diff --git a/parser/metadecoders/decoder_test.go b/parser/metadecoders/decoder_test.go
index 49f7868cc18..f43bb209bc5 100644
--- a/parser/metadecoders/decoder_test.go
+++ b/parser/metadecoders/decoder_test.go
@@ -14,7 +14,6 @@
package metadecoders
import (
- "reflect"
"testing"
qt "github.com/frankban/quicktest"
@@ -91,8 +90,8 @@ func TestUnmarshalToMap(t *testing.T) {
{`a = "b"`, TOML, expect},
{`a: "b"`, YAML, expect},
// Make sure we get all string keys, even for YAML
- {"a: Easy!\nb:\n c: 2\n d: [3, 4]", YAML, map[string]any{"a": "Easy!", "b": map[string]any{"c": 2, "d": []any{3, 4}}}},
- {"a:\n true: 1\n false: 2", YAML, map[string]any{"a": map[string]any{"true": 1, "false": 2}}},
+ {"a: Easy!\nb:\n c: 2\n d: [3, 4]", YAML, map[string]any{"a": "Easy!", "b": map[string]any{"c": uint64(2), "d": []any{uint64(3), uint64(4)}}}},
+ {"a:\n true: 1\n false: 2", YAML, map[string]any{"a": map[string]any{"true": uint64(1), "false": uint64(2)}}},
{`{ "a": "b" }`, JSON, expect},
{`b`, XML, expect},
{`#+a: b`, ORG, expect},
@@ -137,7 +136,7 @@ func TestUnmarshalToInterface(t *testing.T) {
{[]byte(`a: "b"`), YAML, expect},
{[]byte(`b`), XML, expect},
{[]byte(`a,b,c`), CSV, [][]string{{"a", "b", "c"}}},
- {[]byte("a: Easy!\nb:\n c: 2\n d: [3, 4]"), YAML, map[string]any{"a": "Easy!", "b": map[string]any{"c": 2, "d": []any{3, 4}}}},
+ {[]byte("a: Easy!\nb:\n c: 2\n d: [3, 4]"), YAML, map[string]any{"a": "Easy!", "b": map[string]any{"c": uint64(2), "d": []any{uint64(3), uint64(4)}}}},
// errors
{[]byte(`a = "`), TOML, false},
} {
@@ -170,7 +169,7 @@ func TestUnmarshalStringTo(t *testing.T) {
{"32", int64(1234), int64(32)},
{"32", int(1234), int(32)},
{"3.14159", float64(1), float64(3.14159)},
- {"[3,7,9]", []any{}, []any{3, 7, 9}},
+ {"[3,7,9]", []any{}, []any{uint64(3), uint64(7), uint64(9)}},
{"[3.1,7.2,9.3]", []any{}, []any{3.1, 7.2, 9.3}},
} {
msg := qt.Commentf("%d: %T", i, test.to)
@@ -185,128 +184,6 @@ func TestUnmarshalStringTo(t *testing.T) {
}
}
-func TestStringifyYAMLMapKeys(t *testing.T) {
- cases := []struct {
- input any
- want any
- replaced bool
- }{
- {
- map[any]any{"a": 1, "b": 2},
- map[string]any{"a": 1, "b": 2},
- true,
- },
- {
- map[any]any{"a": []any{1, map[any]any{"b": 2}}},
- map[string]any{"a": []any{1, map[string]any{"b": 2}}},
- true,
- },
- {
- map[any]any{true: 1, "b": false},
- map[string]any{"true": 1, "b": false},
- true,
- },
- {
- map[any]any{1: "a", 2: "b"},
- map[string]any{"1": "a", "2": "b"},
- true,
- },
- {
- map[any]any{"a": map[any]any{"b": 1}},
- map[string]any{"a": map[string]any{"b": 1}},
- true,
- },
- {
- map[string]any{"a": map[string]any{"b": 1}},
- map[string]any{"a": map[string]any{"b": 1}},
- false,
- },
- {
- []any{map[any]any{1: "a", 2: "b"}},
- []any{map[string]any{"1": "a", "2": "b"}},
- false,
- },
- }
-
- for i, c := range cases {
- res, replaced := stringifyMapKeys(c.input)
-
- if c.replaced != replaced {
- t.Fatalf("[%d] Replaced mismatch: %t", i, replaced)
- }
- if !c.replaced {
- res = c.input
- }
- if !reflect.DeepEqual(res, c.want) {
- t.Errorf("[%d] given %q\nwant: %q\n got: %q", i, c.input, c.want, res)
- }
- }
-}
-
-func BenchmarkStringifyMapKeysStringsOnlyInterfaceMaps(b *testing.B) {
- maps := make([]map[any]any, b.N)
- for i := 0; i < b.N; i++ {
- maps[i] = map[any]any{
- "a": map[any]any{
- "b": 32,
- "c": 43,
- "d": map[any]any{
- "b": 32,
- "c": 43,
- },
- },
- "b": []any{"a", "b"},
- "c": "d",
- }
- }
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- stringifyMapKeys(maps[i])
- }
-}
-
-func BenchmarkStringifyMapKeysStringsOnlyStringMaps(b *testing.B) {
- m := map[string]any{
- "a": map[string]any{
- "b": 32,
- "c": 43,
- "d": map[string]any{
- "b": 32,
- "c": 43,
- },
- },
- "b": []any{"a", "b"},
- "c": "d",
- }
-
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- stringifyMapKeys(m)
- }
-}
-
-func BenchmarkStringifyMapKeysIntegers(b *testing.B) {
- maps := make([]map[any]any, b.N)
- for i := 0; i < b.N; i++ {
- maps[i] = map[any]any{
- 1: map[any]any{
- 4: 32,
- 5: 43,
- 6: map[any]any{
- 7: 32,
- 8: 43,
- },
- },
- 2: []any{"a", "b"},
- 3: "d",
- }
- }
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- stringifyMapKeys(maps[i])
- }
-}
-
func BenchmarkDecodeYAMLToMap(b *testing.B) {
d := Default
diff --git a/parser/metadecoders/encoder.go b/parser/metadecoders/encoder.go
new file mode 100644
index 00000000000..a18da443a1f
--- /dev/null
+++ b/parser/metadecoders/encoder.go
@@ -0,0 +1,25 @@
+// Copyright 2024 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 metadecoders
+
+import yaml "github.com/goccy/go-yaml"
+
+var yamlEncodeOptions = []yaml.EncodeOption{
+ yaml.UseSingleQuote(true),
+}
+
+// MarshalYAML marshals the given value to YAML.
+var MarshalYAML = func(v any) ([]byte, error) {
+ return yaml.MarshalWithOptions(v, yamlEncodeOptions...)
+}
diff --git a/testscripts/commands/server__error_recovery_edit_content.txt b/testscripts/commands/server__error_recovery_edit_content.txt
index f5ea7e94baf..e640cd8f383 100644
--- a/testscripts/commands/server__error_recovery_edit_content.txt
+++ b/testscripts/commands/server__error_recovery_edit_content.txt
@@ -8,7 +8,7 @@ waitServer
httpget ${HUGOTEST_BASEURL_0}p1/ 'Title: P1'
replace $WORK/content/p1/index.md 'title:' 'titlecolon'
-httpget ${HUGOTEST_BASEURL_0}p1/ 'failed'
+httpget ${HUGOTEST_BASEURL_0}p1/ 'Error'
replace $WORK/content/p1/index.md 'titlecolon' 'title:'
httpget ${HUGOTEST_BASEURL_0}p1/ 'Title: P1'