Skip to content

Commit

Permalink
add backslash as escape for alias:
Browse files Browse the repository at this point in the history
such that \|foo| is not replaced, while |foo| is.
  • Loading branch information
mgoonde committed Nov 23, 2024
1 parent 68afe6e commit af7f127
Show file tree
Hide file tree
Showing 3 changed files with 35 additions and 5 deletions.
7 changes: 6 additions & 1 deletion docs/user_guide/writing_documentation.rst
Original file line number Diff line number Diff line change
Expand Up @@ -243,7 +243,7 @@ There are three predefined macros:
- ``|media|``: the (absolute) path to the `media directory <option-media_dir>`
- ``|page|``: the `static page directory <option-page_dir>`

You can defined additional custom aliases with the `alias
You can define additional custom aliases with the `alias
<option-alias>` option.

.. note::
Expand All @@ -262,6 +262,11 @@ You can defined additional custom aliases with the `alias
This avoids clashes between the syntax for the two features.


.. note::
An alias can be escaped with a prepended baskslash, such that `|foo|` will
be replaced with its alias, while `\|foo|` will be rendered as `|foo|`.


.. _writing-links:

Links
Expand Down
15 changes: 12 additions & 3 deletions ford/_markdown.py
Original file line number Diff line number Diff line change
Expand Up @@ -131,9 +131,14 @@ def convert(

class AliasPreprocessor(Preprocessor):
"""Substitute text aliases of the form ``|foo|`` from a dictionary
of aliases and their replacements"""
of aliases and their replacements.
The backslash ``\`` acts as an escape character, aliases of the
form ``\|foo|`` will not be replaced, but instead copied verbatim
without the preceding backslash.
"""

ALIAS_RE = re.compile(r"\|([^ ].*?[^ ]?)\|")
# pattern to match alias only if not preceded by `\`
ALIAS_RE = re.compile(r"(?<!\\)\|([^ ].*?[^ ]?)\|")

def __init__(self, md: Markdown, aliases: Dict[str, str]):
self.aliases = aliases
Expand All @@ -149,7 +154,11 @@ def _lookup(self, m: re.Match):

def run(self, lines: List[str]) -> List[str]:
for line_num, line in enumerate(lines):
lines[line_num] = self.ALIAS_RE.sub(self._lookup, line)
# replace the real aliases
line = self.ALIAS_RE.sub(self._lookup, line)
# replace the escaped aliases verbatim, without the preceding `\`
line = re.sub(r"\\(\|([^ ].*?[^ ]?)\|)", r"\g<1>",line)
lines[line_num]=line
return lines


Expand Down
18 changes: 17 additions & 1 deletion test/test_markdown.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,25 @@

def test_sub_alias():
result = MetaMarkdown(aliases={"a": "b"}).convert("|a|")

assert result == "<p>b</p>"

result = MetaMarkdown(aliases={"a": "b"}).convert("|undefined|")
assert result == "<p>|undefined|</p>"


def test_sub_alias_escape():
def_alias={"a":"b"}

result = MetaMarkdown(aliases=def_alias).convert("\|a|")
assert result == "<p>|a|</p>"

result = MetaMarkdown(aliases=def_alias).convert("*|a|")
assert result == "<p>*b</p>"

result = MetaMarkdown(aliases=def_alias).convert("\|undefined|")
assert result == "<p>|undefined|</p>"



def test_sub_alias_with_equals():
result = MetaMarkdown(aliases={"a": "b=c"}).convert("|a|")
Expand Down

0 comments on commit af7f127

Please sign in to comment.