792b4dba2c
* update github.com/blevesearch/bleve v2.0.2 -> v2.0.3 * github.com/denisenkom/go-mssqldb v0.9.0 -> v0.10.0 * github.com/editorconfig/editorconfig-core-go v2.4.1 -> v2.4.2 * github.com/go-chi/cors v1.1.1 -> v1.2.0 * github.com/go-git/go-billy v5.0.0 -> v5.1.0 * github.com/go-git/go-git v5.2.0 -> v5.3.0 * github.com/go-ldap/ldap v3.2.4 -> v3.3.0 * github.com/go-redis/redis v8.6.0 -> v8.8.2 * github.com/go-sql-driver/mysql v1.5.0 -> v1.6.0 * github.com/go-swagger/go-swagger v0.26.1 -> v0.27.0 * github.com/lib/pq v1.9.0 -> v1.10.1 * github.com/mattn/go-sqlite3 v1.14.6 -> v1.14.7 * github.com/go-testfixtures/testfixtures v3.5.0 -> v3.6.0 * github.com/issue9/identicon v1.0.1 -> v1.2.0 * github.com/klauspost/compress v1.11.8 -> v1.12.1 * github.com/mgechev/revive v1.0.3 -> v1.0.6 * github.com/microcosm-cc/bluemonday v1.0.7 -> v1.0.8 * github.com/niklasfasching/go-org v1.4.0 -> v1.5.0 * github.com/olivere/elastic v7.0.22 -> v7.0.24 * github.com/pelletier/go-toml v1.8.1 -> v1.9.0 * github.com/prometheus/client_golang v1.9.0 -> v1.10.0 * github.com/xanzy/go-gitlab v0.44.0 -> v0.48.0 * github.com/yuin/goldmark v1.3.3 -> v1.3.5 * github.com/6543/go-version v1.2.4 -> v1.3.1 * do github.com/lib/pq v1.10.0 -> v1.10.1 again ...
115 lines
3 KiB
Go
Vendored
115 lines
3 KiB
Go
Vendored
package parser
|
|
|
|
import (
|
|
"bytes"
|
|
|
|
"github.com/yuin/goldmark/ast"
|
|
"github.com/yuin/goldmark/text"
|
|
"github.com/yuin/goldmark/util"
|
|
)
|
|
|
|
type fencedCodeBlockParser struct {
|
|
}
|
|
|
|
var defaultFencedCodeBlockParser = &fencedCodeBlockParser{}
|
|
|
|
// NewFencedCodeBlockParser returns a new BlockParser that
|
|
// parses fenced code blocks.
|
|
func NewFencedCodeBlockParser() BlockParser {
|
|
return defaultFencedCodeBlockParser
|
|
}
|
|
|
|
type fenceData struct {
|
|
char byte
|
|
indent int
|
|
length int
|
|
node ast.Node
|
|
}
|
|
|
|
var fencedCodeBlockInfoKey = NewContextKey()
|
|
|
|
func (b *fencedCodeBlockParser) Trigger() []byte {
|
|
return []byte{'~', '`'}
|
|
}
|
|
|
|
func (b *fencedCodeBlockParser) Open(parent ast.Node, reader text.Reader, pc Context) (ast.Node, State) {
|
|
line, segment := reader.PeekLine()
|
|
pos := pc.BlockOffset()
|
|
if pos < 0 || (line[pos] != '`' && line[pos] != '~') {
|
|
return nil, NoChildren
|
|
}
|
|
findent := pos
|
|
fenceChar := line[pos]
|
|
i := pos
|
|
for ; i < len(line) && line[i] == fenceChar; i++ {
|
|
}
|
|
oFenceLength := i - pos
|
|
if oFenceLength < 3 {
|
|
return nil, NoChildren
|
|
}
|
|
var info *ast.Text
|
|
if i < len(line)-1 {
|
|
rest := line[i:]
|
|
left := util.TrimLeftSpaceLength(rest)
|
|
right := util.TrimRightSpaceLength(rest)
|
|
if left < len(rest)-right {
|
|
infoStart, infoStop := segment.Start-segment.Padding+i+left, segment.Stop-right
|
|
value := rest[left : len(rest)-right]
|
|
if fenceChar == '`' && bytes.IndexByte(value, '`') > -1 {
|
|
return nil, NoChildren
|
|
} else if infoStart != infoStop {
|
|
info = ast.NewTextSegment(text.NewSegment(infoStart, infoStop))
|
|
}
|
|
}
|
|
}
|
|
node := ast.NewFencedCodeBlock(info)
|
|
pc.Set(fencedCodeBlockInfoKey, &fenceData{fenceChar, findent, oFenceLength, node})
|
|
return node, NoChildren
|
|
|
|
}
|
|
|
|
func (b *fencedCodeBlockParser) Continue(node ast.Node, reader text.Reader, pc Context) State {
|
|
line, segment := reader.PeekLine()
|
|
fdata := pc.Get(fencedCodeBlockInfoKey).(*fenceData)
|
|
|
|
// if code block line starts with a tab, keep a tab as it is.
|
|
if segment.Padding != 0 {
|
|
preserveLeadingTabInCodeBlock(&segment, reader, fdata.indent)
|
|
}
|
|
w, pos := util.IndentWidth(line, reader.LineOffset())
|
|
if w < 4 {
|
|
i := pos
|
|
for ; i < len(line) && line[i] == fdata.char; i++ {
|
|
}
|
|
length := i - pos
|
|
if length >= fdata.length && util.IsBlank(line[i:]) {
|
|
newline := 1
|
|
if line[len(line)-1] != '\n' {
|
|
newline = 0
|
|
}
|
|
reader.Advance(segment.Stop - segment.Start - newline - segment.Padding)
|
|
return Close
|
|
}
|
|
}
|
|
pos, padding := util.DedentPositionPadding(line, reader.LineOffset(), segment.Padding, fdata.indent)
|
|
|
|
seg := text.NewSegmentPadding(segment.Start+pos, segment.Stop, padding)
|
|
node.Lines().Append(seg)
|
|
reader.AdvanceAndSetPadding(segment.Stop-segment.Start-pos-1, padding)
|
|
return Continue | NoChildren
|
|
}
|
|
|
|
func (b *fencedCodeBlockParser) Close(node ast.Node, reader text.Reader, pc Context) {
|
|
fdata := pc.Get(fencedCodeBlockInfoKey).(*fenceData)
|
|
if fdata.node == node {
|
|
pc.Set(fencedCodeBlockInfoKey, nil)
|
|
}
|
|
}
|
|
|
|
func (b *fencedCodeBlockParser) CanInterruptParagraph() bool {
|
|
return true
|
|
}
|
|
|
|
func (b *fencedCodeBlockParser) CanAcceptIndentedLine() bool {
|
|
return false
|
|
}
|