Remove the trailing new line in .Code
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 26 Feb 2022 09:42:21 +0000 (10:42 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 26 Feb 2022 20:54:36 +0000 (21:54 +0100)
Fixes #9572

common/text/transform.go
common/text/transform_test.go
hugolib/integrationtest_builder.go
markup/goldmark/codeblocks/integration_test.go
markup/goldmark/codeblocks/render.go
markup/highlight/highlight.go
tpl/strings/strings.go

index f5957780363cebd96b2b3fc36ebe0f772b48d69a..66a67d8bc1af708374187670debe6cf96cab4660 100644 (file)
@@ -14,6 +14,7 @@
 package text
 
 import (
+       "strings"
        "sync"
        "unicode"
 
@@ -45,3 +46,18 @@ func RemoveAccentsString(s string) string {
        accentTransformerPool.Put(t)
        return s
 }
+
+// Chomp removes trailing newline characters from s.
+func Chomp(s string) string {
+       return strings.TrimRightFunc(s, func(r rune) bool {
+               return r == '\n' || r == '\r'
+       })
+}
+
+// Puts adds a trailing \n none found.
+func Puts(s string) string {
+       if s == "" || s[len(s)-1] == '\n' {
+               return s
+       }
+       return s + "\n"
+}
index 08265f976b503534446cf7eb8814d8ea018e45a6..992dd524c80542fe796bd4e7c47d4566c52e9285 100644 (file)
@@ -26,3 +26,18 @@ func TestRemoveAccents(t *testing.T) {
        c.Assert(string(RemoveAccents([]byte("Hugo Rocks!"))), qt.Equals, "Hugo Rocks!")
        c.Assert(string(RemoveAccentsString("Resumé")), qt.Equals, "Resume")
 }
+
+func TestChomp(t *testing.T) {
+       c := qt.New(t)
+
+       c.Assert(Chomp("\nA\n"), qt.Equals, "\nA")
+       c.Assert(Chomp("A\r\n"), qt.Equals, "A")
+}
+
+func TestPuts(t *testing.T) {
+       c := qt.New(t)
+
+       c.Assert(Puts("A"), qt.Equals, "A\n")
+       c.Assert(Puts("\nA\n"), qt.Equals, "\nA\n")
+       c.Assert(Puts(""), qt.Equals, "")
+}
index ed68783a1650b8169fb05d8387ef6b726c7bdffa..b7b43624f37260b8af5b37b679b0c1d32bce6747 100644 (file)
@@ -28,6 +28,9 @@ import (
 )
 
 func NewIntegrationTestBuilder(conf IntegrationTestConfig) *IntegrationTestBuilder {
+       // Code fences.
+       conf.TxtarString = strings.ReplaceAll(conf.TxtarString, "§§§", "```")
+
        data := txtar.Parse([]byte(conf.TxtarString))
 
        c, ok := conf.T.(*qt.C)
index d662b39118e0085b19831e57b61e8f21681fcab6..0c9039cec68b3649f761b291ec177e9b74850df0 100644 (file)
@@ -113,3 +113,34 @@ Go Language: golang|
                "<h2 id=\"bash-code\">Bash Code</h2>\n<div class=\"highlight blue\"><pre tabindex=\"0\" class=\"chroma\"><code class=\"language-bash\" data-lang=\"bash\"><span class=\"line\"><span class=\"ln\">32</span><span class=\"cl\"><span class=\"nb\">echo</span> <span class=\"s2\">&#34;l1&#34;</span><span class=\"p\">;</span>\n</span></span><span class=\"line hl\"><span class=\"ln\">33</span>",
        )
 }
+
+func TestCodeChomp(t *testing.T) {
+       t.Parallel()
+
+       files := `
+-- config.toml --
+-- content/p1.md --
+---
+title: "p1"
+---
+
+§§§bash
+echo "p1";
+§§§
+-- layouts/_default/single.html --
+{{ .Content }}
+-- layouts/_default/_markup/render-codeblock.html --
+|{{ .Code | safeHTML }}|
+
+`
+
+       b := hugolib.NewIntegrationTestBuilder(
+               hugolib.IntegrationTestConfig{
+                       T:           t,
+                       TxtarString: files,
+                       NeedsOsFS:   false,
+               },
+       ).Build()
+
+       b.AssertFileContent("public/p1/index.html", "|echo \"p1\";|")
+}
index b598763d360a135c98ca4e9e2ae7921d252ec190..6cc43128be9b4f7327bfb9ad01f2764b3662de27 100644 (file)
@@ -17,6 +17,7 @@ import (
        "bytes"
        "fmt"
 
+       htext "github.com/gohugoio/hugo/common/text"
        "github.com/gohugoio/hugo/markup/converter/hooks"
        "github.com/gohugoio/hugo/markup/goldmark/internal/render"
        "github.com/gohugoio/hugo/markup/internal/attributes"
@@ -75,7 +76,8 @@ func (r *htmlRenderer) renderCodeBlock(w util.BufWriter, src []byte, node ast.No
                line := n.b.Lines().At(i)
                buff.Write(line.Value(src))
        }
-       text := buff.String()
+
+       text := htext.Chomp(buff.String())
 
        var info []byte
        if n.b.Info != nil {
index 6013ba1c8b1fc003f8c6ff86eb57815b1c200ca7..156007d0ad7240ec6a0cbfc2d4703c2550900ca8 100644 (file)
@@ -25,6 +25,7 @@ import (
        "github.com/alecthomas/chroma/lexers"
        "github.com/alecthomas/chroma/styles"
        "github.com/gohugoio/hugo/common/hugio"
+       "github.com/gohugoio/hugo/common/text"
        "github.com/gohugoio/hugo/identity"
        "github.com/gohugoio/hugo/markup/converter/hooks"
        "github.com/gohugoio/hugo/markup/internal/attributes"
@@ -123,7 +124,9 @@ func (h chromaHighlighter) RenderCodeblock(w hugio.FlexiWriter, ctx hooks.Codebl
                return err
        }
 
-       return highlight(w, ctx.Code(), ctx.Lang(), attributes, cfg)
+       code := text.Puts(ctx.Code())
+
+       return highlight(w, code, ctx.Lang(), attributes, cfg)
 }
 
 var id = identity.NewPathIdentity("chroma", "highlight")
index ac2defed50d6b7e74578ff4124f5f2d0865156c1..2575b2feee352e72c38c539635cd27103f0403a1 100644 (file)
@@ -21,6 +21,7 @@ import (
        "strings"
        "unicode/utf8"
 
+       "github.com/gohugoio/hugo/common/text"
        "github.com/gohugoio/hugo/deps"
        "github.com/gohugoio/hugo/helpers"
 
@@ -119,7 +120,7 @@ func (ns *Namespace) Chomp(s interface{}) (interface{}, error) {
                return "", err
        }
 
-       res := strings.TrimRight(ss, "\r\n")
+       res := text.Chomp(ss)
        switch s.(type) {
        case template.HTML:
                return template.HTML(res), nil