markup/goldmark: Default to https for linkify
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 9 Mar 2022 17:26:32 +0000 (18:26 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 9 Mar 2022 21:30:10 +0000 (22:30 +0100)
Fixes #9639

markup/goldmark/convert.go
markup/goldmark/goldmark_config/config.go
markup/goldmark/integration_test.go
markup/goldmark/render_hooks.go

index 4c1641a0b56e151809c99db2067a25a631c1604a..9442ee9e7ac101d1253fb046d0c976bdbf6e0afd 100644 (file)
@@ -94,7 +94,7 @@ func newMarkdown(pcfg converter.ProviderConfig) goldmark.Markdown {
 
        var (
                extensions = []goldmark.Extender{
-                       newLinks(),
+                       newLinks(cfg),
                        newTocExtension(rendererOptions),
                }
                parserOptions []parser.Option
index 82b8d96301c92655fa3ab13a44d8d50bd6090ab3..a3238091b18185d85ae7af915aeafbeceee7a5d8 100644 (file)
@@ -23,13 +23,14 @@ const (
 // DefaultConfig holds the default Goldmark configuration.
 var Default = Config{
        Extensions: Extensions{
-               Typographer:    true,
-               Footnote:       true,
-               DefinitionList: true,
-               Table:          true,
-               Strikethrough:  true,
-               Linkify:        true,
-               TaskList:       true,
+               Typographer:     true,
+               Footnote:        true,
+               DefinitionList:  true,
+               Table:           true,
+               Strikethrough:   true,
+               Linkify:         true,
+               LinkifyProtocol: "https",
+               TaskList:        true,
        },
        Renderer: Renderer{
                Unsafe: false,
@@ -57,10 +58,11 @@ type Extensions struct {
        DefinitionList bool
 
        // GitHub flavored markdown
-       Table         bool
-       Strikethrough bool
-       Linkify       bool
-       TaskList      bool
+       Table           bool
+       Strikethrough   bool
+       Linkify         bool
+       LinkifyProtocol string
+       TaskList        bool
 }
 
 type Renderer struct {
index 89cd5bbb64464964d6c805c73debed7a0af26cc2..d8f218b31ffa7dcc266684974d711dc6ec88a737 100644 (file)
@@ -423,3 +423,70 @@ title: "p1"
                <img src="b.jpg" alt="&quot;a&quot;">
        `)
 }
+
+func TestLinkifyProtocol(t *testing.T) {
+       t.Parallel()
+
+       runTest := func(protocol string, withHook bool) *hugolib.IntegrationTestBuilder {
+
+               files := `
+-- config.toml --
+[markup.goldmark]
+[markup.goldmark.extensions]
+linkify = true
+linkifyProtocol = "PROTOCOL"
+-- content/p1.md --
+---
+title: "p1"
+---
+Link no procol: www.example.org
+Link http procol: http://www.example.org
+Link https procol: https://www.example.org
+
+-- layouts/_default/single.html --
+{{ .Content }}
+`
+               files = strings.ReplaceAll(files, "PROTOCOL", protocol)
+
+               if withHook {
+                       files += `-- layouts/_default/_markup/render-link.html --
+<a href="{{ .Destination | safeURL }}">{{ .Text | safeHTML }}</a>`
+               }
+
+               return hugolib.NewIntegrationTestBuilder(
+                       hugolib.IntegrationTestConfig{
+                               T:           t,
+                               TxtarString: files,
+                       },
+               ).Build()
+
+       }
+
+       for _, withHook := range []bool{false, true} {
+
+               b := runTest("https", withHook)
+
+               b.AssertFileContent("public/p1/index.html",
+                       "Link no procol: <a href=\"https://www.example.org\">www.example.org</a>",
+                       "Link http procol: <a href=\"http://www.example.org\">http://www.example.org</a>",
+                       "Link https procol: <a href=\"https://www.example.org\">https://www.example.org</a></p>",
+               )
+
+               b = runTest("http", withHook)
+
+               b.AssertFileContent("public/p1/index.html",
+                       "Link no procol: <a href=\"http://www.example.org\">www.example.org</a>",
+                       "Link http procol: <a href=\"http://www.example.org\">http://www.example.org</a>",
+                       "Link https procol: <a href=\"https://www.example.org\">https://www.example.org</a></p>",
+               )
+
+               b = runTest("gopher", withHook)
+
+               b.AssertFileContent("public/p1/index.html",
+                       "Link no procol: <a href=\"gopher://www.example.org\">www.example.org</a>",
+                       "Link http procol: <a href=\"http://www.example.org\">http://www.example.org</a>",
+                       "Link https procol: <a href=\"https://www.example.org\">https://www.example.org</a></p>",
+               )
+
+       }
+}
index 138a60d263b06fd079c34c0e6b5f23fb87bba805..a22030f545c39ef8c9e7b16581a4d836829a2b4e 100644 (file)
@@ -18,6 +18,7 @@ import (
        "strings"
 
        "github.com/gohugoio/hugo/markup/converter/hooks"
+       "github.com/gohugoio/hugo/markup/goldmark/goldmark_config"
        "github.com/gohugoio/hugo/markup/goldmark/internal/render"
        "github.com/gohugoio/hugo/markup/internal/attributes"
 
@@ -30,8 +31,9 @@ import (
 
 var _ renderer.SetOptioner = (*hookedRenderer)(nil)
 
-func newLinkRenderer() renderer.NodeRenderer {
+func newLinkRenderer(cfg goldmark_config.Config) renderer.NodeRenderer {
        r := &hookedRenderer{
+               linkifyProtocol: []byte(cfg.Extensions.LinkifyProtocol),
                Config: html.Config{
                        Writer: html.DefaultWriter,
                },
@@ -39,8 +41,8 @@ func newLinkRenderer() renderer.NodeRenderer {
        return r
 }
 
-func newLinks() goldmark.Extender {
-       return &links{}
+func newLinks(cfg goldmark_config.Config) goldmark.Extender {
+       return &links{cfg: cfg}
 }
 
 type linkContext struct {
@@ -105,6 +107,7 @@ func (ctx headingContext) PlainText() string {
 }
 
 type hookedRenderer struct {
+       linkifyProtocol []byte
        html.Config
 }
 
@@ -279,7 +282,7 @@ func (r *hookedRenderer) renderAutoLink(w util.BufWriter, source []byte, node as
                return r.renderAutoLinkDefault(w, source, node, entering)
        }
 
-       url := string(n.URL(source))
+       url := string(r.autoLinkURL(n, source))
        label := string(n.Label(source))
        if n.AutoLinkType == ast.AutoLinkEmail && !strings.HasPrefix(strings.ToLower(url), "mailto:") {
                url = "mailto:" + url
@@ -310,8 +313,9 @@ func (r *hookedRenderer) renderAutoLinkDefault(w util.BufWriter, source []byte,
        if !entering {
                return ast.WalkContinue, nil
        }
+
        _, _ = w.WriteString(`<a href="`)
-       url := n.URL(source)
+       url := r.autoLinkURL(n, source)
        label := n.Label(source)
        if n.AutoLinkType == ast.AutoLinkEmail && !bytes.HasPrefix(bytes.ToLower(url), []byte("mailto:")) {
                _, _ = w.WriteString("mailto:")
@@ -329,6 +333,17 @@ func (r *hookedRenderer) renderAutoLinkDefault(w util.BufWriter, source []byte,
        return ast.WalkContinue, nil
 }
 
+func (r *hookedRenderer) autoLinkURL(n *ast.AutoLink, source []byte) []byte {
+       url := n.URL(source)
+       if len(n.Protocol) > 0 && !bytes.Equal(n.Protocol, r.linkifyProtocol) {
+               // The CommonMark spec says "http" is the correct protocol for links,
+               // but this doesn't make much sense (the fact that they should care about the rendered output).
+               // Note that n.Protocol is not set if protocol is provided by user.
+               url = append(r.linkifyProtocol, url[len(n.Protocol):]...)
+       }
+       return url
+}
+
 func (r *hookedRenderer) renderHeading(w util.BufWriter, source []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
        n := node.(*ast.Heading)
        var hr hooks.HeadingRenderer
@@ -394,11 +409,13 @@ func (r *hookedRenderer) renderHeadingDefault(w util.BufWriter, source []byte, n
        return ast.WalkContinue, nil
 }
 
-type links struct{}
+type links struct {
+       cfg goldmark_config.Config
+}
 
 // Extend implements goldmark.Extender.
 func (e *links) Extend(m goldmark.Markdown) {
        m.Renderer().AddOptions(renderer.WithNodeRenderers(
-               util.Prioritized(newLinkRenderer(), 100),
+               util.Prioritized(newLinkRenderer(e.cfg), 100),
        ))
 }