markup/goldmark: Improve attributes vs options
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 26 Feb 2022 16:24:10 +0000 (17:24 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 26 Feb 2022 20:54:36 +0000 (21:54 +0100)
Fixes #9571

hugolib/site.go
markup/converter/hooks/hooks.go
markup/goldmark/codeblocks/integration_test.go
markup/goldmark/codeblocks/render.go
markup/highlight/highlight.go
markup/internal/attributes/attributes.go

index ebda29f463bcc614df3246641359485c6f049c32..c76bdc141bc26739a31cfde874ab58951e230a47 100644 (file)
@@ -1798,6 +1798,10 @@ func (hr hookRendererTemplate) ResolvePosition(ctx interface{}) text.Position {
        return hr.resolvePosition(ctx)
 }
 
+func (hr hookRendererTemplate) IsDefaultCodeBlockRenderer() bool {
+       return false
+}
+
 func (s *Site) renderForTemplate(name, outputFormat string, d interface{}, w io.Writer, templ tpl.Template) (err error) {
        if templ == nil {
                s.logMissingLayout(name, "", "", outputFormat)
index a570113ff34c87f80180ba3467b93e74b4b85216..e81689f04355f4f42b5b8a6a09b489716c3b16d3 100644 (file)
@@ -90,11 +90,11 @@ type HeadingRenderer interface {
        identity.Provider
 }
 
-// ElementPositionRevolver provides a way to resolve the start Position
+// ElementPositionResolver provides a way to resolve the start Position
 // of a markdown element in the original source document.
 // This may be both slow and aproximate, so should only be
 // used for error logging.
-type ElementPositionRevolver interface {
+type ElementPositionResolver interface {
        ResolvePosition(ctx interface{}) text.Position
 }
 
index 68f6b809e2d138182efda8c20eccad76dcae57b2..c33ea31e3846b405d044063d8c79cd2733641e3b 100644 (file)
@@ -14,6 +14,7 @@
 package codeblocks_test
 
 import (
+       "strings"
        "testing"
 
        "github.com/gohugoio/hugo/hugolib"
@@ -176,7 +177,7 @@ Position: {{ .Position | safeHTML }}
 }
 
 // Issue 9571
-func TestOptionsNonChroma(t *testing.T) {
+func TestAttributesChroma(t *testing.T) {
        t.Parallel()
 
        files := `
@@ -188,23 +189,27 @@ title: "p1"
 
 ##   Code
 
-§§§bash {style=monokai}
+§§§LANGUAGE {style=monokai}
 echo "p1";
 §§§
 -- layouts/_default/single.html --
 {{ .Content }}
 -- layouts/_default/_markup/render-codeblock.html --
-Style: {{ .Attributes }}|
+Attributes: {{ .Attributes }}|Options: {{ .Options }}|
 
 
 `
-
-       b := hugolib.NewIntegrationTestBuilder(
-               hugolib.IntegrationTestConfig{
-                       T:           t,
-                       TxtarString: files,
-               },
-       ).Build()
-
-       b.AssertFileContent("public/p1/index.html", "asdfadf")
+       testLanguage := func(language, expect string) {
+               b := hugolib.NewIntegrationTestBuilder(
+                       hugolib.IntegrationTestConfig{
+                               T:           t,
+                               TxtarString: strings.ReplaceAll(files, "LANGUAGE", language),
+                       },
+               ).Build()
+
+               b.AssertFileContent("public/p1/index.html", expect)
+       }
+
+       testLanguage("bash", "Attributes: map[]|Options: map[style:monokai]|")
+       testLanguage("hugo", "Attributes: map[style:monokai]|Options: map[]|")
 }
index bbf15bef30a6bcca7fff4bb62168239d1635fc7a..27252fc27cf02fd8dd3609f19738d18cac1de6f5 100644 (file)
@@ -18,7 +18,7 @@ import (
        "fmt"
        "sync"
 
-       "github.com/gohugoio/hugo/common/herrors"
+       "github.com/alecthomas/chroma/lexers"
        htext "github.com/gohugoio/hugo/common/text"
        "github.com/gohugoio/hugo/markup/converter/hooks"
        "github.com/gohugoio/hugo/markup/goldmark/internal/render"
@@ -61,8 +61,6 @@ func (r *htmlRenderer) RegisterFuncs(reg renderer.NodeRendererFuncRegisterer) {
 }
 
 func (r *htmlRenderer) renderCodeBlock(w util.BufWriter, src []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
-       defer herrors.Recover()
-
        ctx := w.(*render.Context)
 
        if entering {
@@ -92,17 +90,26 @@ func (r *htmlRenderer) renderCodeBlock(w util.BufWriter, src []byte, node ast.No
        if n.b.Info != nil {
                info = n.b.Info.Segment.Value(src)
        }
+
+       attrtp := attributes.AttributesOwnerCodeBlockCustom
+       if isd, ok := renderer.(hooks.IsDefaultCodeBlockRendererProvider); (ok && isd.IsDefaultCodeBlockRenderer()) || lexers.Get(lang) != nil {
+               // We say that this is a Chroma code block if it's the default code block renderer
+               // or if the language is supported by Chroma.
+               attrtp = attributes.AttributesOwnerCodeBlockChroma
+       }
+
+       // IsDefaultCodeBlockRendererProvider
        attrs := getAttributes(n.b, info)
        cbctx := &codeBlockContext{
                page:             ctx.DocumentContext().Document,
                lang:             lang,
                code:             s,
                ordinal:          ordinal,
-               AttributesHolder: attributes.New(attrs, attributes.AttributesOwnerCodeBlock),
+               AttributesHolder: attributes.New(attrs, attrtp),
        }
 
        cbctx.createPos = func() htext.Position {
-               if resolver, ok := renderer.(hooks.ElementPositionRevolver); ok {
+               if resolver, ok := renderer.(hooks.ElementPositionResolver); ok {
                        return resolver.ResolvePosition(cbctx)
                }
                return htext.Position{
index 156007d0ad7240ec6a0cbfc2d4703c2550900ca8..dac45e3121b987ec6ce997c464e652802f712635 100644 (file)
@@ -61,6 +61,7 @@ type Highlighter interface {
        Highlight(code, lang string, opts interface{}) (string, error)
        HighlightCodeBlock(ctx hooks.CodeblockContext, opts interface{}) (HightlightResult, error)
        hooks.CodeBlockRenderer
+       hooks.IsDefaultCodeBlockRendererProvider
 }
 
 type chromaHighlighter struct {
@@ -129,6 +130,10 @@ func (h chromaHighlighter) RenderCodeblock(w hugio.FlexiWriter, ctx hooks.Codebl
        return highlight(w, code, ctx.Lang(), attributes, cfg)
 }
 
+func (h chromaHighlighter) IsDefaultCodeBlockRenderer() bool {
+       return true
+}
+
 var id = identity.NewPathIdentity("chroma", "highlight")
 
 func (h chromaHighlighter) GetIdentity() identity.Identity {
index 1cce7edd1e02943da6773cafb5362149fd802c77..a20690c4cc63d4f6e1a3756893439fd5a14807fb 100644 (file)
@@ -50,7 +50,8 @@ type AttributesOwnerType int
 
 const (
        AttributesOwnerGeneral AttributesOwnerType = iota
-       AttributesOwnerCodeBlock
+       AttributesOwnerCodeBlockChroma
+       AttributesOwnerCodeBlockCustom
 )
 
 func New(astAttributes []ast.Attribute, ownerType AttributesOwnerType) *AttributesHolder {
@@ -99,7 +100,7 @@ func New(astAttributes []ast.Attribute, ownerType AttributesOwnerType) *Attribut
                        panic(fmt.Sprintf("not implemented: %T", vvv))
                }
 
-               if ownerType == AttributesOwnerCodeBlock && chromaHightlightProcessingAttributes[nameLower] {
+               if ownerType == AttributesOwnerCodeBlockChroma && chromaHightlightProcessingAttributes[nameLower] {
                        attr := Attribute{Name: string(v.Name), Value: vv}
                        opts = append(opts, attr)
                } else {