mkdocs2confluence 0.9.7__tar.gz → 0.9.8__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/PKG-INFO +2 -1
  2. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/README.md +1 -0
  3. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/pyproject.toml +1 -1
  4. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/PKG-INFO +2 -1
  5. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/emitter/xhtml.py +8 -0
  6. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/__init__.py +3 -1
  7. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/nodes.py +11 -0
  8. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/parser/markdown.py +13 -0
  9. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_parser.py +38 -0
  10. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/LICENSE +0 -0
  11. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/setup.cfg +0 -0
  12. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/SOURCES.txt +0 -0
  13. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/dependency_links.txt +0 -0
  14. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/entry_points.txt +0 -0
  15. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/requires.txt +0 -0
  16. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/top_level.txt +0 -0
  17. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/__init__.py +0 -0
  18. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/cli.py +0 -0
  19. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/emitter/__init__.py +0 -0
  20. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/document.py +0 -0
  21. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/treeutil.py +0 -0
  22. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/__init__.py +0 -0
  23. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/config.py +0 -0
  24. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/extra_css.py +0 -0
  25. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/nav.py +0 -0
  26. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/page.py +0 -0
  27. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/parser/__init__.py +0 -0
  28. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/__init__.py +0 -0
  29. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/generator.py +0 -0
  30. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/render.py +0 -0
  31. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/__init__.py +0 -0
  32. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/abbrevs.py +0 -0
  33. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/fence.py +0 -0
  34. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/frontmatter.py +0 -0
  35. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/icons.py +0 -0
  36. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/includes.py +0 -0
  37. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/linkdefs.py +0 -0
  38. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/__init__.py +0 -0
  39. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/render.py +0 -0
  40. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/server.py +0 -0
  41. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/__init__.py +0 -0
  42. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/client.py +0 -0
  43. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/pipeline.py +0 -0
  44. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/__init__.py +0 -0
  45. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/anchoring.py +0 -0
  46. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/command.py +0 -0
  47. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/comments.py +0 -0
  48. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/github.py +0 -0
  49. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/platform.py +0 -0
  50. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/state.py +0 -0
  51. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/__init__.py +0 -0
  52. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/abbrevs.py +0 -0
  53. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/assets.py +0 -0
  54. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/editlink.py +0 -0
  55. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/footer.py +0 -0
  56. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/images.py +0 -0
  57. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/internallinks.py +0 -0
  58. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/mermaid.py +0 -0
  59. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_abbrevs.py +0 -0
  60. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_children_macro.py +0 -0
  61. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_cli.py +0 -0
  62. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_editlink.py +0 -0
  63. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_emitter.py +0 -0
  64. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_extra_css.py +0 -0
  65. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_footer.py +0 -0
  66. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_frontmatter.py +0 -0
  67. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_icons.py +0 -0
  68. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_images.py +0 -0
  69. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_internallinks.py +0 -0
  70. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_ir.py +0 -0
  71. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_linkdefs.py +0 -0
  72. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_loader.py +0 -0
  73. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_mermaid.py +0 -0
  74. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_page_loader.py +0 -0
  75. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_pdf.py +0 -0
  76. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_preprocess.py +0 -0
  77. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_preview.py +0 -0
  78. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_publish_client.py +0 -0
  79. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_publish_config.py +0 -0
  80. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_publish_pipeline.py +0 -0
  81. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_server.py +0 -0
  82. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_sync_anchoring.py +0 -0
  83. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_sync_command.py +0 -0
  84. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_sync_comments.py +0 -0
  85. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_sync_github.py +0 -0
  86. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_sync_state.py +0 -0
  87. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.8}/tests/test_treeutil.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: mkdocs2confluence
3
- Version: 0.9.7
3
+ Version: 0.9.8
4
4
  Summary: Publish MkDocs Material pages to Confluence Cloud — admonitions, Mermaid diagrams, tabs, page properties and more
5
5
  Author: Anders Hybertz
6
6
  License: GPL-3.0-or-later
@@ -393,6 +393,7 @@ confluence:
393
393
  | Content tabs `=== "Label"` | `expand` macros (one per tab) |
394
394
  | Details blocks `??? "title"` | `expand` macro |
395
395
  | Footnotes `[^1]` | Superscript anchor links + *Footnotes* section at page bottom |
396
+ | In-page anchors `<a id="...">` / `<a name="...">` | Confluence `anchor` macro; same-page links `[text](#target)` resolve correctly |
396
397
  | Mermaid diagrams | PNG via Kroki, uploaded as attachment (`<ac:image ac:align="center">`) |
397
398
  | Internal links `[text](page.md)` | Native Confluence page link; `#fragment` anchors preserved |
398
399
  | `awesome-pages` nav (`.pages` files) | Fully supported |
@@ -353,6 +353,7 @@ confluence:
353
353
  | Content tabs `=== "Label"` | `expand` macros (one per tab) |
354
354
  | Details blocks `??? "title"` | `expand` macro |
355
355
  | Footnotes `[^1]` | Superscript anchor links + *Footnotes* section at page bottom |
356
+ | In-page anchors `<a id="...">` / `<a name="...">` | Confluence `anchor` macro; same-page links `[text](#target)` resolve correctly |
356
357
  | Mermaid diagrams | PNG via Kroki, uploaded as attachment (`<ac:image ac:align="center">`) |
357
358
  | Internal links `[text](page.md)` | Native Confluence page link; `#fragment` anchors preserved |
358
359
  | `awesome-pages` nav (`.pages` files) | Fully supported |
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "mkdocs2confluence"
3
- version = "0.9.7"
3
+ version = "0.9.8"
4
4
  description = "Publish MkDocs Material pages to Confluence Cloud — admonitions, Mermaid diagrams, tabs, page properties and more"
5
5
  readme = "README.md"
6
6
  license = { text = "GPL-3.0-or-later" }
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: mkdocs2confluence
3
- Version: 0.9.7
3
+ Version: 0.9.8
4
4
  Summary: Publish MkDocs Material pages to Confluence Cloud — admonitions, Mermaid diagrams, tabs, page properties and more
5
5
  Author: Anders Hybertz
6
6
  License: GPL-3.0-or-later
@@ -393,6 +393,7 @@ confluence:
393
393
  | Content tabs `=== "Label"` | `expand` macros (one per tab) |
394
394
  | Details blocks `??? "title"` | `expand` macro |
395
395
  | Footnotes `[^1]` | Superscript anchor links + *Footnotes* section at page bottom |
396
+ | In-page anchors `<a id="...">` / `<a name="...">` | Confluence `anchor` macro; same-page links `[text](#target)` resolve correctly |
396
397
  | Mermaid diagrams | PNG via Kroki, uploaded as attachment (`<ac:image ac:align="center">`) |
397
398
  | Internal links `[text](page.md)` | Native Confluence page link; `#fragment` anchors preserved |
398
399
  | `awesome-pages` nav (`.pages` files) | Fully supported |
@@ -24,6 +24,7 @@ from mkdocs_to_confluence.ir.nodes import (
24
24
  AbbrevFootnoteNode,
25
25
  AbbrevGlossaryBlock,
26
26
  Admonition,
27
+ AnchorNode,
27
28
  BlockQuote,
28
29
  BoldNode,
29
30
  BulletList,
@@ -685,6 +686,13 @@ def _emit_inline(node: IRNode) -> str:
685
686
  return f"{open_tag}{_emit_inlines(node.children)}{close_tag}"
686
687
  if isinstance(node, RawInlineHtml):
687
688
  return node.html_str
689
+ if isinstance(node, AnchorNode):
690
+ name = html.escape(node.name)
691
+ return (
692
+ f'<ac:structured-macro ac:name="anchor">'
693
+ f"<ac:parameter ac:name=\"\"><![CDATA[{name}]]></ac:parameter>"
694
+ f"</ac:structured-macro>"
695
+ )
688
696
  # Fallback: emit unknown inline nodes as escaped repr
689
697
  return html.escape(repr(node))
690
698
 
@@ -13,6 +13,8 @@ from mkdocs_to_confluence.ir.document import Document, PageMeta, compute_sha
13
13
  from mkdocs_to_confluence.ir.nodes import (
14
14
  # Material extension nodes
15
15
  Admonition,
16
+ # Inline HTML nodes
17
+ AnchorNode,
16
18
  BlockQuote,
17
19
  BoldNode,
18
20
  # List nodes
@@ -31,7 +33,6 @@ from mkdocs_to_confluence.ir.nodes import (
31
33
  GridCards,
32
34
  HorizontalRule,
33
35
  ImageNode,
34
- # Inline HTML nodes
35
36
  InlineHtmlNode,
36
37
  InsertNode,
37
38
  # Traversal utility
@@ -83,6 +84,7 @@ __all__ = [
83
84
  "LineBreakNode",
84
85
  "InlineHtmlNode",
85
86
  "RawInlineHtml",
87
+ "AnchorNode",
86
88
  "InsertNode",
87
89
  # Block
88
90
  "Section",
@@ -137,6 +137,17 @@ class RawInlineHtml(IRNode):
137
137
  html_str: str
138
138
 
139
139
 
140
+ @dataclass(frozen=True)
141
+ class AnchorNode(IRNode):
142
+ """An in-page anchor target (``<a id="...">`` or ``<a name="...">``) in Markdown.
143
+
144
+ Emitted as a Confluence ``anchor`` macro so that same-page links
145
+ (e.g. ``[text](#target)``) resolve correctly in Confluence.
146
+ """
147
+
148
+ name: str
149
+
150
+
140
151
  @dataclass(frozen=True)
141
152
  class ImageNode(IRNode):
142
153
  """An image reference.
@@ -56,6 +56,7 @@ from typing import Union
56
56
 
57
57
  from mkdocs_to_confluence.ir.nodes import (
58
58
  Admonition,
59
+ AnchorNode,
59
60
  BlockQuote,
60
61
  BoldNode,
61
62
  BulletList,
@@ -981,6 +982,18 @@ def _scan_inline(text: str, fn_map: dict[str, int] | None = None) -> list[IRNode
981
982
  i = close_idx + len(close)
982
983
  continue
983
984
 
985
+ # Anchor target: <a id="..."> / <a name="..."> (all forms)
986
+ anchor_m = re.match(
987
+ r'<a\s+(?:id|name)="([^"]+)"[^>]*/?>(?:\s*</a>)?',
988
+ text[i:],
989
+ re.IGNORECASE,
990
+ )
991
+ if anchor_m:
992
+ flush()
993
+ nodes.append(AnchorNode(name=anchor_m.group(1)))
994
+ i += len(anchor_m.group(0))
995
+ continue
996
+
984
997
  # Generic inline HTML with attributes (e.g. <span class="...">) — pass through verbatim
985
998
  generic_m = re.match(
986
999
  r"<([a-z][a-z0-9]*)\b[^>]*>.*?</\1>", text[i:], re.DOTALL | re.IGNORECASE
@@ -1343,3 +1343,41 @@ class TestGridCards:
1343
1343
  nodes = parse(md)
1344
1344
  gc = next(n for n in nodes if isinstance(n, GridCards))
1345
1345
  assert len(gc.items) == 3
1346
+
1347
+
1348
+ class TestAnchorNode:
1349
+ def test_anchor_id_parsed(self) -> None:
1350
+ from mkdocs_to_confluence.ir import AnchorNode
1351
+ para = first(parse('Some text <a id="my-anchor"></a> more\n'), Paragraph)
1352
+ assert any(isinstance(n, AnchorNode) and n.name == "my-anchor" for n in para.children)
1353
+
1354
+ def test_anchor_name_parsed(self) -> None:
1355
+ from mkdocs_to_confluence.ir import AnchorNode
1356
+ para = first(parse('Some text <a name="my-anchor"></a> more\n'), Paragraph)
1357
+ assert any(isinstance(n, AnchorNode) and n.name == "my-anchor" for n in para.children)
1358
+
1359
+ def test_anchor_self_closing_parsed(self) -> None:
1360
+ from mkdocs_to_confluence.ir import AnchorNode
1361
+ para = first(parse('Before <a id="target"/> after\n'), Paragraph)
1362
+ assert any(isinstance(n, AnchorNode) and n.name == "target" for n in para.children)
1363
+
1364
+ def test_anchor_open_only_parsed(self) -> None:
1365
+ from mkdocs_to_confluence.ir import AnchorNode
1366
+ para = first(parse('Before <a id="target"> after\n'), Paragraph)
1367
+ assert any(isinstance(n, AnchorNode) and n.name == "target" for n in para.children)
1368
+
1369
+ def test_anchor_emitted_as_confluence_macro(self) -> None:
1370
+ from mkdocs_to_confluence.emitter.xhtml import emit
1371
+ nodes = parse('Go to <a id="section-1"></a> here\n')
1372
+ xhtml = emit(nodes)
1373
+ assert 'ac:structured-macro ac:name="anchor"' in xhtml
1374
+ assert "section-1" in xhtml
1375
+
1376
+ def test_anchor_link_and_target_roundtrip(self) -> None:
1377
+ """[Text](#target) link + <a id="target"> definition renders correctly."""
1378
+ from mkdocs_to_confluence.emitter.xhtml import emit
1379
+ nodes = parse('[Jump](#section-1)\n\n<a id="section-1"></a>\n')
1380
+ xhtml = emit(nodes)
1381
+ assert 'href="#section-1"' in xhtml
1382
+ assert 'ac:structured-macro ac:name="anchor"' in xhtml
1383
+ assert "section-1" in xhtml