From a542d8508eaa4aa263cbecfdf27c099a9c528735 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Mon, 5 Dec 2022 20:46:35 -0500 Subject: [PATCH 01/10] feat: start renaming implementation to admon --- .pre-commit-config.yaml | 3 + mdformat-admon/plugin.py | 20 ++-- pyproject.toml | 7 +- tests/fixtures.md | 231 ++++++++++++++++++++++++++++++++++++--- 4 files changed, 236 insertions(+), 25 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 0e5544b..dc82612 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -27,3 +27,6 @@ repos: - flake8-bugbear>=21.3.2 - flake8-builtins>=1.5.3 - flake8-comprehensions>=3.4.0 + args: [ + --ignore=E501 + ] diff --git a/mdformat-admon/plugin.py b/mdformat-admon/plugin.py index 8fff61c..62f0c68 100644 --- a/mdformat-admon/plugin.py +++ b/mdformat-admon/plugin.py @@ -3,22 +3,28 @@ from markdown_it import MarkdownIt from mdformat.renderer import RenderContext, RenderTreeNode from mdformat.renderer.typing import Render +from mdit_py_plugins.admon import admon_plugin def update_mdit(mdit: MarkdownIt) -> None: - """Update the parser, e.g. by adding a plugin: `mdit.use(myplugin)`""" - pass + """Update the parser,""" + mdit.use(admon_plugin) -def _render_table(node: RenderTreeNode, context: RenderContext) -> str: - """Render a `RenderTreeNode` of type "table". +def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: + """Render a `RenderTreeNode` of type `admon`. + + Based on: + + - https://github.com/hukkin/mdformat-gfm/blob/cf316a121b6cf35cbff7b0ad6e171f287803f8cb/src/mdformat_gfm/plugin.py + - https://github.com/hukkin/mdformat-toc/blob/42624b6f3468da4f793ec7425da872b030714774/mdformat_toc/plugin.py + - https://github.com/executablebooks/mdformat-footnote/blob/80852fc20cfba7fd0330b9ac7a1a4df983542942/mdformat_footnote/plugin.py - Change "table" to the name of the syntax you want to render. """ - return "" + return node.markup + "\n" + node.markup # FIXME: Implement! # A mapping from syntax tree node type to a function that renders it. # This can be used to overwrite renderer functions of existing syntax # or add support for new syntax. -RENDERERS: Mapping[str, Render] = {"table": _render_table} +RENDERERS: Mapping[str, Render] = {"admon": _render_admon} diff --git a/pyproject.toml b/pyproject.toml index baa15bb..597cd23 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -15,16 +15,15 @@ classifiers = [ "Topic :: Software Development :: Libraries :: Python Modules", ] keywords = ["mdformat", "markdown", "markdown-it"] -requires-python = ">=3.7" +requires-python = ">=3.7.2" dependencies = [ - "mdformat >=0.7.0,<0.8.0", + "mdformat >=0.7.16,<0.8.0", ] dynamic = ["version", "description"] [project.optional-dependencies] test = [ - "pytest~=6.0", - "coverage", + "pytest>=6.0", "pytest-cov", ] dev = ["pre-commit"] diff --git a/tests/fixtures.md b/tests/fixtures.md index 31d6be2..8ca0921 100644 --- a/tests/fixtures.md +++ b/tests/fixtures.md @@ -1,24 +1,227 @@ -a test + +Simple admonition +. +!!! note + *content* +. +!!! note + *content* +. + + +Could contain block elements too +. +!!! note + ### heading + + ----------- + +. +!!! note + ### heading + + ----------- + +. + + +Shows custom title +. +!!! note Custom title + + Some text + +. +!!! note Custom title + + Some text + +. + + +Shows no title +. +!!! note "" + Some text + . -This is the input Markdown test, -then below add the expected output. +!!! note "" + Some text + +. + + +Closes block after 2 empty lines +. +!!! note + Some text + + + A code block +. +!!! note + Some text + + + A code block +. + + +Nested blocks . -This is the input Markdown test, -then below add the expected output. +!!! note + !!! note + Some text + + code block . +!!! note + !!! note + Some text -another test + code block . -Some *markdown* -- a -- b -* c + +Consecutive admonitions . -Some *markdown* +!!! note -- a -- b +!!! warning +. +!!! note -* c +!!! warning +. + + +Marker may be indented up to 3 chars +. + !!! note + content +. + !!! note + content +. + + +But that's a code block +. + !!! note + content +. + !!! note + content +. + + +Some more indent checks +. + !!! note + not a code block + + code block +. + !!! note + not a code block + + code block +. + + +Type could be adjacent to marker +. +!!!note + xxx + +. +!!!note + xxx + +. + + +Type could be adjacent to marker and content may be shifted up to 3 chars +. +!!!note + xxx + +. +!!!note + xxx + +. + + +Or several spaces apart +. +!!! note + xxx +. +!!! note + xxx +. + + +Admonitions self-close at the end of the document +. +!!! note + xxx +. +!!! note + xxx +. + + +They could be nested in lists +. +- !!! note + - a + - b +- !!! warning + - c + - d +. +- !!! note + - a + - b +- !!! warning + - c + - d +. + + +Or in blockquotes +. +> !!! note +> xxx +> > yyy +> zzz +> +. +> !!! note +> xxx +> > yyy +> zzz +> +. + + +Renders unknown admonition type +. +!!! unknown title + content +. +!!! unknown title + content +. + + +Does not render +. +!!! + content +. +!!! + content . From fd213df510167c25096d84da491fd1ec9d3bf90b Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 20:52:16 -0500 Subject: [PATCH 02/10] build: ensure minimum mdit-py-plugins --- pyproject.toml | 1 + 1 file changed, 1 insertion(+) diff --git a/pyproject.toml b/pyproject.toml index 597cd23..45c28ed 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -18,6 +18,7 @@ keywords = ["mdformat", "markdown", "markdown-it"] requires-python = ">=3.7.2" dependencies = [ "mdformat >=0.7.16,<0.8.0", + "mdit-py-plugins >=0.3.2", ] dynamic = ["version", "description"] From 90e9a1b0ff80f7909964ad60db0002841e72f845 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 20:55:00 -0500 Subject: [PATCH 03/10] fix: folder name should have an underscore --- {mdformat-admon => mdformat_admon}/__init__.py | 0 {mdformat-admon => mdformat_admon}/plugin.py | 0 2 files changed, 0 insertions(+), 0 deletions(-) rename {mdformat-admon => mdformat_admon}/__init__.py (100%) rename {mdformat-admon => mdformat_admon}/plugin.py (100%) diff --git a/mdformat-admon/__init__.py b/mdformat_admon/__init__.py similarity index 100% rename from mdformat-admon/__init__.py rename to mdformat_admon/__init__.py diff --git a/mdformat-admon/plugin.py b/mdformat_admon/plugin.py similarity index 100% rename from mdformat-admon/plugin.py rename to mdformat_admon/plugin.py From 8b586bd3c9da47bab790391769f52ae651977a35 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 21:21:28 -0500 Subject: [PATCH 04/10] fix: resolve KeyError of admonition --- mdformat_admon/plugin.py | 4 ++-- pyproject.toml | 2 +- tests/test_fixtures.py | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/mdformat_admon/plugin.py b/mdformat_admon/plugin.py index 62f0c68..1528b9a 100644 --- a/mdformat_admon/plugin.py +++ b/mdformat_admon/plugin.py @@ -12,7 +12,7 @@ def update_mdit(mdit: MarkdownIt) -> None: def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: - """Render a `RenderTreeNode` of type `admon`. + """Render a `RenderTreeNode` of type `admonition`. Based on: @@ -27,4 +27,4 @@ def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: # A mapping from syntax tree node type to a function that renders it. # This can be used to overwrite renderer functions of existing syntax # or add support for new syntax. -RENDERERS: Mapping[str, Render] = {"admon": _render_admon} +RENDERERS: Mapping[str, Render] = {"admonition": _render_admon} diff --git a/pyproject.toml b/pyproject.toml index 45c28ed..fe1255b 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -33,7 +33,7 @@ dev = ["pre-commit"] Homepage = "https://github.com/executablebooks/mdformat-admon" [project.entry-points."mdformat.parser_extension"] -admon = "mdformat_admon" +admonition = "mdformat_admon" [tool.flit.sdist] include = [] diff --git a/tests/test_fixtures.py b/tests/test_fixtures.py index 5e19d41..cc17c2d 100644 --- a/tests/test_fixtures.py +++ b/tests/test_fixtures.py @@ -12,6 +12,6 @@ "line,title,text,expected", fixtures, ids=[f[1] for f in fixtures] ) def test_fixtures(line, title, text, expected): - output = mdformat.text(text, extensions={"admon"}) + output = mdformat.text(text, extensions={"admonition"}) print(output) assert output.rstrip() == expected.rstrip(), output From 1c76c6e137dfeadec4a4083f746e469bd3470807 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:28:16 -0500 Subject: [PATCH 05/10] feat: trying to figure this out --- mdformat_admon/plugin.py | 23 +++++++++++++++++++++-- tests/fixtures.md | 2 +- tests/test_fixtures.py | 3 +-- 3 files changed, 23 insertions(+), 5 deletions(-) diff --git a/mdformat_admon/plugin.py b/mdformat_admon/plugin.py index 1528b9a..d0a5088 100644 --- a/mdformat_admon/plugin.py +++ b/mdformat_admon/plugin.py @@ -1,3 +1,4 @@ +import textwrap from typing import Mapping from markdown_it import MarkdownIt @@ -14,14 +15,32 @@ def update_mdit(mdit: MarkdownIt) -> None: def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: """Render a `RenderTreeNode` of type `admonition`. - Based on: + Primarily based on: + + - https://github.com/executablebooks/mdformat-footnote/blob/80852fc20cfba7fd0330b9ac7a1a4df983542942/mdformat_footnote/plugin.py#LL24-L40C29 + + And based on: - https://github.com/hukkin/mdformat-gfm/blob/cf316a121b6cf35cbff7b0ad6e171f287803f8cb/src/mdformat_gfm/plugin.py - https://github.com/hukkin/mdformat-toc/blob/42624b6f3468da4f793ec7425da872b030714774/mdformat_toc/plugin.py - https://github.com/executablebooks/mdformat-footnote/blob/80852fc20cfba7fd0330b9ac7a1a4df983542942/mdformat_footnote/plugin.py """ - return node.markup + "\n" + node.markup # FIXME: Implement! + # FIXME: children + # > return "\n\n".join([child.markup for child in node.children]) + + print(node.pretty()) # FYI: Debugging + + separator = "\n" + indent = " " * 4 # FIXME: Make this configurable? + body = "" + with context.indented(len(indent)): # TODO: What is this for? + elements = [child.render(context) for child in node.children[1:]] + body += textwrap.indent((separator + separator).join(elements), indent) + first_line = node.children[0].markup + result = separator.join([first_line, body]) + breakpoint() + return result # A mapping from syntax tree node type to a function that renders it. diff --git a/tests/fixtures.md b/tests/fixtures.md index 8ca0921..fe2d2d9 100644 --- a/tests/fixtures.md +++ b/tests/fixtures.md @@ -20,7 +20,7 @@ Could contain block elements too !!! note ### heading - ----------- + ______________________________________________________________________ . diff --git a/tests/test_fixtures.py b/tests/test_fixtures.py index cc17c2d..313637b 100644 --- a/tests/test_fixtures.py +++ b/tests/test_fixtures.py @@ -13,5 +13,4 @@ ) def test_fixtures(line, title, text, expected): output = mdformat.text(text, extensions={"admonition"}) - print(output) - assert output.rstrip() == expected.rstrip(), output + assert output.rstrip() == expected.rstrip() From c6b209061e935d4871713de7689bd7d54572d51a Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:29:52 -0500 Subject: [PATCH 06/10] test: add arguments --- tox.ini | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tox.ini b/tox.ini index 0e7ed6f..933e78e 100644 --- a/tox.ini +++ b/tox.ini @@ -4,7 +4,7 @@ isolated_build = True [testenv:py{36,37,38,39}] extras = test -commands = pytest {posargs} +commands = pytest {posargs} --exitfirst --last-failed --new-first -vv --no-cov [testenv:py{36,37,38,39}-cov] extras = test From 30ab82a6983b507e65a61ff5c5eea7f0f4f91a4d Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:45:48 -0500 Subject: [PATCH 07/10] refactor: minor improvements --- mdformat_admon/plugin.py | 23 ++++++++--------------- 1 file changed, 8 insertions(+), 15 deletions(-) diff --git a/mdformat_admon/plugin.py b/mdformat_admon/plugin.py index d0a5088..daa704b 100644 --- a/mdformat_admon/plugin.py +++ b/mdformat_admon/plugin.py @@ -26,21 +26,14 @@ def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: - https://github.com/executablebooks/mdformat-footnote/blob/80852fc20cfba7fd0330b9ac7a1a4df983542942/mdformat_footnote/plugin.py """ - # FIXME: children - # > return "\n\n".join([child.markup for child in node.children]) - - print(node.pretty()) # FYI: Debugging - - separator = "\n" - indent = " " * 4 # FIXME: Make this configurable? - body = "" - with context.indented(len(indent)): # TODO: What is this for? - elements = [child.render(context) for child in node.children[1:]] - body += textwrap.indent((separator + separator).join(elements), indent) - first_line = node.children[0].markup - result = separator.join([first_line, body]) - breakpoint() - return result + separator = "\n\n" # TODO: Is this configurable? + indent = " " * 4 # FIXME: Is this configurable? + title = node.children[0].render(context) # or 'node.info.strip()'? + body = f"{node.markup} {title}{separator}" + with context.indented(len(indent)): # Modifies context.env['indent_width'] + elements = [child.render(context) for child in [*node.walk()][1:]] + body += textwrap.indent(separator.join(elements), indent) + return body # A mapping from syntax tree node type to a function that renders it. From 0e36620f2dca41da387e9bd5400df87901595251 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:48:17 -0500 Subject: [PATCH 08/10] Revert "test: add arguments" This reverts commit c6b209061e935d4871713de7689bd7d54572d51a. --- tox.ini | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tox.ini b/tox.ini index 933e78e..0e7ed6f 100644 --- a/tox.ini +++ b/tox.ini @@ -4,7 +4,7 @@ isolated_build = True [testenv:py{36,37,38,39}] extras = test -commands = pytest {posargs} --exitfirst --last-failed --new-first -vv --no-cov +commands = pytest {posargs} [testenv:py{36,37,38,39}-cov] extras = test From 6d308744513f21ad600963f981f6dfa434accda5 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:57:54 -0500 Subject: [PATCH 09/10] fix: handle admon_title from info --- mdformat_admon/plugin.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/mdformat_admon/plugin.py b/mdformat_admon/plugin.py index daa704b..ed52746 100644 --- a/mdformat_admon/plugin.py +++ b/mdformat_admon/plugin.py @@ -28,10 +28,10 @@ def _render_admon(node: RenderTreeNode, context: RenderContext) -> str: """ separator = "\n\n" # TODO: Is this configurable? indent = " " * 4 # FIXME: Is this configurable? - title = node.children[0].render(context) # or 'node.info.strip()'? + title = node.info.strip() body = f"{node.markup} {title}{separator}" with context.indented(len(indent)): # Modifies context.env['indent_width'] - elements = [child.render(context) for child in [*node.walk()][1:]] + elements = [child.render(context) for child in node.children[1:]] body += textwrap.indent(separator.join(elements), indent) return body From 58cef76d9a74b34e9070a13e529d79f2b2696c01 Mon Sep 17 00:00:00 2001 From: Kyle King Date: Tue, 13 Dec 2022 22:59:50 -0500 Subject: [PATCH 10/10] build: lower mdformat minimum --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index fe1255b..0d58574 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -17,7 +17,7 @@ classifiers = [ keywords = ["mdformat", "markdown", "markdown-it"] requires-python = ">=3.7.2" dependencies = [ - "mdformat >=0.7.16,<0.8.0", + "mdformat >=0.7.0,<0.8.0", "mdit-py-plugins >=0.3.2", ] dynamic = ["version", "description"]