mkdocs2confluence 0.9.6__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.6 → mkdocs2confluence-0.9.8}/PKG-INFO +2 -1
  2. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/README.md +1 -0
  3. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/pyproject.toml +1 -1
  4. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/PKG-INFO +2 -1
  5. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/emitter/xhtml.py +8 -0
  6. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/__init__.py +3 -1
  7. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/nodes.py +11 -0
  8. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/parser/markdown.py +13 -0
  9. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/mermaid.py +5 -3
  10. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_mermaid.py +39 -2
  11. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_parser.py +38 -0
  12. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/LICENSE +0 -0
  13. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/setup.cfg +0 -0
  14. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/SOURCES.txt +0 -0
  15. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/dependency_links.txt +0 -0
  16. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/entry_points.txt +0 -0
  17. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/requires.txt +0 -0
  18. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs2confluence.egg-info/top_level.txt +0 -0
  19. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/__init__.py +0 -0
  20. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/cli.py +0 -0
  21. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/emitter/__init__.py +0 -0
  22. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/document.py +0 -0
  23. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/ir/treeutil.py +0 -0
  24. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/__init__.py +0 -0
  25. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/config.py +0 -0
  26. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/extra_css.py +0 -0
  27. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/nav.py +0 -0
  28. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/loader/page.py +0 -0
  29. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/parser/__init__.py +0 -0
  30. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/__init__.py +0 -0
  31. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/generator.py +0 -0
  32. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/pdf/render.py +0 -0
  33. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/__init__.py +0 -0
  34. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/abbrevs.py +0 -0
  35. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/fence.py +0 -0
  36. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/frontmatter.py +0 -0
  37. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/icons.py +0 -0
  38. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/includes.py +0 -0
  39. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preprocess/linkdefs.py +0 -0
  40. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/__init__.py +0 -0
  41. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/render.py +0 -0
  42. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/preview/server.py +0 -0
  43. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/__init__.py +0 -0
  44. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/client.py +0 -0
  45. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/publisher/pipeline.py +0 -0
  46. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/__init__.py +0 -0
  47. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/anchoring.py +0 -0
  48. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/command.py +0 -0
  49. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/comments.py +0 -0
  50. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/github.py +0 -0
  51. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/platform.py +0 -0
  52. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/sync/state.py +0 -0
  53. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/__init__.py +0 -0
  54. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/abbrevs.py +0 -0
  55. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/assets.py +0 -0
  56. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/editlink.py +0 -0
  57. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/footer.py +0 -0
  58. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/images.py +0 -0
  59. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/src/mkdocs_to_confluence/transforms/internallinks.py +0 -0
  60. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_abbrevs.py +0 -0
  61. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_children_macro.py +0 -0
  62. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_cli.py +0 -0
  63. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_editlink.py +0 -0
  64. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_emitter.py +0 -0
  65. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_extra_css.py +0 -0
  66. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_footer.py +0 -0
  67. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_frontmatter.py +0 -0
  68. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_icons.py +0 -0
  69. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_images.py +0 -0
  70. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_internallinks.py +0 -0
  71. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_ir.py +0 -0
  72. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_linkdefs.py +0 -0
  73. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_loader.py +0 -0
  74. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_page_loader.py +0 -0
  75. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_pdf.py +0 -0
  76. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_preprocess.py +0 -0
  77. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_preview.py +0 -0
  78. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_publish_client.py +0 -0
  79. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_publish_config.py +0 -0
  80. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_publish_pipeline.py +0 -0
  81. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_server.py +0 -0
  82. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_sync_anchoring.py +0 -0
  83. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_sync_command.py +0 -0
  84. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_sync_comments.py +0 -0
  85. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_sync_github.py +0 -0
  86. {mkdocs2confluence-0.9.6 → mkdocs2confluence-0.9.8}/tests/test_sync_state.py +0 -0
  87. {mkdocs2confluence-0.9.6 → 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.6
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.6"
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.6
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
@@ -135,11 +135,13 @@ def _render_one(source: str, kroki_url: str, *, quiet: bool = False) -> Path | N
135
135
  continue # retry
136
136
  _warn(f"mermaid diagram: Kroki returned HTTP {exc.code} {exc.reason} — falling back to code block")
137
137
  return None
138
- except urllib.error.URLError as exc:
138
+ except (urllib.error.URLError, OSError) as exc:
139
+ # URLError covers connection errors; OSError covers read-phase timeouts
140
+ # (socket.timeout is OSError but not URLError when raised by resp.read()).
139
141
  last_exc = exc
140
- kroki_unavailable = True # timeout or connection refused — kroki unreachable
142
+ kroki_unavailable = True
141
143
  continue # retry — network blip
142
- except (OSError, ValueError) as exc:
144
+ except ValueError as exc:
143
145
  _warn(f"mermaid diagram: {exc} — falling back to code block")
144
146
  return None
145
147
 
@@ -398,7 +398,7 @@ def test_render_one_504_on_public_kroki_falls_back_to_mermaid_ink(tmp_path, caps
398
398
 
399
399
 
400
400
  def test_render_one_timeout_on_public_kroki_falls_back_to_mermaid_ink(tmp_path):
401
- """Public kroki.io timeout → automatic fallback to mermaid.ink."""
401
+ """Public kroki.io URLError timeout → automatic fallback to mermaid.ink."""
402
402
  import urllib.error
403
403
 
404
404
  from mkdocs_to_confluence.transforms.mermaid import _render_one
@@ -416,6 +416,25 @@ def test_render_one_timeout_on_public_kroki_falls_back_to_mermaid_ink(tmp_path):
416
416
  mock_ink.assert_called_once_with(_SAMPLE_SOURCE)
417
417
 
418
418
 
419
+ def test_render_one_read_phase_timeout_falls_back_to_mermaid_ink(tmp_path):
420
+ """socket.timeout during resp.read() (OSError, not URLError) → fallback to mermaid.ink."""
421
+ import socket
422
+
423
+ from mkdocs_to_confluence.transforms.mermaid import _render_one
424
+
425
+ with patch("mkdocs_to_confluence.transforms.mermaid._CACHE_DIR", tmp_path), \
426
+ patch("mkdocs_to_confluence.transforms.mermaid.time.sleep"), \
427
+ patch("mkdocs_to_confluence.transforms.mermaid._kroki_png",
428
+ side_effect=socket.timeout("The read operation timed out")), \
429
+ patch("mkdocs_to_confluence.transforms.mermaid._mermaid_ink_png",
430
+ return_value=_FAKE_PNG) as mock_ink:
431
+ result = _render_one(_SAMPLE_SOURCE, "https://kroki.io")
432
+
433
+ assert result is not None
434
+ assert result.exists()
435
+ mock_ink.assert_called_once_with(_SAMPLE_SOURCE)
436
+
437
+
419
438
  def test_render_one_504_on_self_hosted_kroki_does_not_fall_back(tmp_path):
420
439
  """Self-hosted Kroki 504 → no mermaid.ink fallback (privacy isolation)."""
421
440
  import urllib.error
@@ -435,7 +454,7 @@ def test_render_one_504_on_self_hosted_kroki_does_not_fall_back(tmp_path):
435
454
 
436
455
 
437
456
  def test_render_one_timeout_on_self_hosted_kroki_does_not_fall_back(tmp_path):
438
- """Self-hosted Kroki timeout → no mermaid.ink fallback (privacy isolation)."""
457
+ """Self-hosted Kroki URLError timeout → no mermaid.ink fallback (privacy isolation)."""
439
458
  import urllib.error
440
459
 
441
460
  from mkdocs_to_confluence.transforms.mermaid import _render_one
@@ -452,6 +471,24 @@ def test_render_one_timeout_on_self_hosted_kroki_does_not_fall_back(tmp_path):
452
471
  mock_ink.assert_not_called()
453
472
 
454
473
 
474
+ def test_render_one_read_phase_timeout_self_hosted_does_not_fall_back(tmp_path):
475
+ """socket.timeout on self-hosted Kroki → no mermaid.ink fallback (privacy isolation)."""
476
+ import socket
477
+
478
+ from mkdocs_to_confluence.transforms.mermaid import _render_one
479
+
480
+ with patch("mkdocs_to_confluence.transforms.mermaid._CACHE_DIR", tmp_path), \
481
+ patch("mkdocs_to_confluence.transforms.mermaid.time.sleep"), \
482
+ patch("mkdocs_to_confluence.transforms.mermaid._kroki_png",
483
+ side_effect=socket.timeout("The read operation timed out")), \
484
+ patch("mkdocs_to_confluence.transforms.mermaid._mermaid_ink_png",
485
+ return_value=_FAKE_PNG) as mock_ink:
486
+ result = _render_one(_SAMPLE_SOURCE, "https://my-internal-kroki.corp")
487
+
488
+ assert result is None
489
+ mock_ink.assert_not_called()
490
+
491
+
455
492
  def test_render_one_mermaid_ink_fallback_also_fails_returns_none(tmp_path, capsys):
456
493
  """When mermaid.ink fallback also fails, returns None (code-block fallback)."""
457
494
  import urllib.error
@@ -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