Skip to content

Commit

Permalink
allow short-style rst references with symbols (nim-lang#17372)
Browse files Browse the repository at this point in the history
  • Loading branch information
a-mr authored and ringabout committed Mar 22, 2021
1 parent db593ca commit ea617e4
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 11 deletions.
41 changes: 30 additions & 11 deletions lib/packages/docutils/rst.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1011,14 +1011,21 @@ proc parseSmiley(p: var RstParser): PRstNode =
result.text = val
return

proc validRefnamePunct(x: string): bool =
## https://docutils.sourceforge.io/docs/ref/rst/restructuredtext.html#reference-names
x.len == 1 and x[0] in {'-', '_', '.', ':', '+'}

proc isUrl(p: RstParser, i: int): bool =
result = p.tok[i+1].symbol == ":" and p.tok[i+2].symbol == "//" and
p.tok[i+3].kind == tkWord and
p.tok[i].symbol in ["http", "https", "ftp", "telnet", "file"]

proc parseWordOrUrl(p: var RstParser, father: PRstNode) =
#if currentTok(p).symbol[strStart] == '<':
if isUrl(p, p.idx):
proc parseWordOrRef(p: var RstParser, father: PRstNode) =
## Parses a normal word or may be a reference or URL.
if nextTok(p).kind != tkPunct: # <- main path, a normal word
father.add newLeaf(p)
inc p.idx
elif isUrl(p, p.idx): # URL http://something
var n = newRstNode(rnStandaloneHyperlink)
while true:
case currentTok(p).kind
Expand All @@ -1031,10 +1038,26 @@ proc parseWordOrUrl(p: var RstParser, father: PRstNode) =
inc p.idx
father.add(n)
else:
var n = newLeaf(p)
# check for reference (probably, long one like some.ref.with.dots_ )
var saveIdx = p.idx
var isRef = false
inc p.idx
if currentTok(p).symbol == "_": n = parsePostfix(p, n)
father.add(n)
while currentTok(p).kind in {tkWord, tkPunct}:
if currentTok(p).kind == tkPunct:
if isInlineMarkupEnd(p, "_"):
isRef = true
break
if not validRefnamePunct(currentTok(p).symbol):
break
inc p.idx
if isRef:
let r = newRstNode(rnRef)
for i in saveIdx..p.idx-1: r.add newLeaf(p.tok[i].symbol)
father.add r
inc p.idx # skip final _
else: # 1 normal word
father.add newLeaf(p.tok[saveIdx].symbol)
p.idx = saveIdx + 1

proc parseBackslash(p: var RstParser, father: PRstNode) =
assert(currentTok(p).kind == tkPunct)
Expand Down Expand Up @@ -1154,10 +1177,6 @@ proc getFootnoteType(label: PRstNode): (FootnoteType, int) =
else:
result = (fnCitation, -1)

proc validRefnamePunct(x: string): bool =
## https://docutils.sourceforge.io/docs/ref/rst/restructuredtext.html#reference-names
x.len == 1 and x[0] in {'-', '_', '.', ':', '+'}

proc parseFootnoteName(p: var RstParser, reference: bool): PRstNode =
## parse footnote/citation label. Precondition: start at `[`.
## Label text should be valid ref. name symbol, otherwise nil is returned.
Expand Down Expand Up @@ -1258,7 +1277,7 @@ proc parseInline(p: var RstParser, father: PRstNode) =
if n != nil:
father.add(n)
return
parseWordOrUrl(p, father)
parseWordOrRef(p, father)
of tkAdornment, tkOther, tkWhite:
if roSupportMarkdown in p.s.options and currentTok(p).symbol == "```":
inc p.idx
Expand Down
25 changes: 25 additions & 0 deletions tests/stdlib/trstgen.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1184,6 +1184,31 @@ Test1
doAssert "Ref. <a class=\"reference internal\" " &
"href=\"#some-definition\">some definition</a>" in output1

test "RST references (additional symbols)":
# check that ., _, -, +, : are allowed symbols in references without ` `
let input1 = dedent """
sec.1
-----
2-other:sec+c_2
^^^^^^^^^^^^^^^
.. _link.1_2021:
Paragraph
Ref. sec.1_! and 2-other:sec+c_2_;and link.1_2021_.
"""
let output1 = input1.toHtml
doAssert "id=\"secdot1\"" in output1
doAssert "id=\"Z2minusothercolonsecplusc-2\"" in output1
doAssert "id=\"linkdot1-2021\"" in output1
let ref1 = "<a class=\"reference internal\" href=\"#secdot1\">sec.1</a>"
let ref2 = "<a class=\"reference internal\" href=\"#Z2minusothercolonsecplusc-2\">2-other:sec+c_2</a>"
let ref3 = "<a class=\"reference internal\" href=\"#linkdot1-2021\">link.1_2021</a>"
let refline = "Ref. " & ref1 & "! and " & ref2 & ";and " & ref3 & "."
doAssert refline in output1

suite "RST/Code highlight":
test "Basic Python code highlight":
let pythonCode = """
Expand Down

0 comments on commit ea617e4

Please sign in to comment.