mkdocs2confluence 0.9.7__tar.gz → 0.9.9__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.9}/PKG-INFO +2 -1
  2. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/README.md +1 -0
  3. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/pyproject.toml +1 -1
  4. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/PKG-INFO +2 -1
  5. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/emitter/xhtml.py +8 -0
  6. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/ir/__init__.py +3 -1
  7. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/ir/nodes.py +11 -0
  8. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/parser/markdown.py +13 -0
  9. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_parser.py +38 -0
  10. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_publish_client.py +169 -0
  11. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/LICENSE +0 -0
  12. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/setup.cfg +0 -0
  13. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/SOURCES.txt +0 -0
  14. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/dependency_links.txt +0 -0
  15. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/entry_points.txt +0 -0
  16. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/requires.txt +0 -0
  17. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs2confluence.egg-info/top_level.txt +0 -0
  18. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/__init__.py +0 -0
  19. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/cli.py +0 -0
  20. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/emitter/__init__.py +0 -0
  21. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/ir/document.py +0 -0
  22. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/ir/treeutil.py +0 -0
  23. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/loader/__init__.py +0 -0
  24. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/loader/config.py +0 -0
  25. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/loader/extra_css.py +0 -0
  26. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/loader/nav.py +0 -0
  27. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/loader/page.py +0 -0
  28. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/parser/__init__.py +0 -0
  29. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/pdf/__init__.py +0 -0
  30. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/pdf/generator.py +0 -0
  31. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/pdf/render.py +0 -0
  32. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/__init__.py +0 -0
  33. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/abbrevs.py +0 -0
  34. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/fence.py +0 -0
  35. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/frontmatter.py +0 -0
  36. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/icons.py +0 -0
  37. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/includes.py +0 -0
  38. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preprocess/linkdefs.py +0 -0
  39. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preview/__init__.py +0 -0
  40. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preview/render.py +0 -0
  41. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/preview/server.py +0 -0
  42. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/publisher/__init__.py +0 -0
  43. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/publisher/client.py +0 -0
  44. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/publisher/pipeline.py +0 -0
  45. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/__init__.py +0 -0
  46. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/anchoring.py +0 -0
  47. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/command.py +0 -0
  48. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/comments.py +0 -0
  49. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/github.py +0 -0
  50. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/platform.py +0 -0
  51. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/sync/state.py +0 -0
  52. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/__init__.py +0 -0
  53. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/abbrevs.py +0 -0
  54. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/assets.py +0 -0
  55. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/editlink.py +0 -0
  56. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/footer.py +0 -0
  57. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/images.py +0 -0
  58. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/internallinks.py +0 -0
  59. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/src/mkdocs_to_confluence/transforms/mermaid.py +0 -0
  60. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_abbrevs.py +0 -0
  61. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_children_macro.py +0 -0
  62. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_cli.py +0 -0
  63. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_editlink.py +0 -0
  64. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_emitter.py +0 -0
  65. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_extra_css.py +0 -0
  66. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_footer.py +0 -0
  67. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_frontmatter.py +0 -0
  68. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_icons.py +0 -0
  69. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_images.py +0 -0
  70. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_internallinks.py +0 -0
  71. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_ir.py +0 -0
  72. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_linkdefs.py +0 -0
  73. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_loader.py +0 -0
  74. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_mermaid.py +0 -0
  75. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_page_loader.py +0 -0
  76. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_pdf.py +0 -0
  77. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_preprocess.py +0 -0
  78. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_preview.py +0 -0
  79. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_publish_config.py +0 -0
  80. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_publish_pipeline.py +0 -0
  81. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_server.py +0 -0
  82. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_sync_anchoring.py +0 -0
  83. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_sync_command.py +0 -0
  84. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_sync_comments.py +0 -0
  85. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_sync_github.py +0 -0
  86. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/tests/test_sync_state.py +0 -0
  87. {mkdocs2confluence-0.9.7 → mkdocs2confluence-0.9.9}/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.9
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.9"
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.9
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
@@ -728,6 +728,37 @@ def test_find_folder_under_returns_none_when_not_found() -> None:
728
728
  assert result is None
729
729
 
730
730
 
731
+ # ── resolve_inline_comment ─────────────────────────────────────────────────────
732
+
733
+
734
+ def test_resolve_inline_comment_sends_get_then_put() -> None:
735
+ """resolve_inline_comment GETs the current comment, then PUTs with resolved=True and a version bump."""
736
+ existing = {
737
+ "id": "42",
738
+ "version": {"number": 3},
739
+ "body": {"storage": {"value": "<p>Original body</p>"}},
740
+ "resolved": False,
741
+ }
742
+ transport = _MockTransport(
743
+ _json_response(existing), # GET → current state
744
+ _json_response({}), # PUT → resolved
745
+ )
746
+ config = _make_config()
747
+ with ConfluenceClient(config) as client:
748
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
749
+ client.resolve_inline_comment("42")
750
+ assert len(transport.requests) == 2
751
+ get_req = transport.requests[0]
752
+ assert get_req.method == "GET"
753
+ assert "/inline-comments/42" in str(get_req.url)
754
+ put_req = transport.requests[1]
755
+ assert put_req.method == "PUT"
756
+ body = json.loads(put_req.content)
757
+ assert body["version"]["number"] == 4
758
+ assert body["resolved"] is True
759
+ assert body["body"]["value"] == "<p>Original body</p>"
760
+
761
+
731
762
  # ── find_folder_in_space ──────────────────────────────────────────────────────
732
763
 
733
764
 
@@ -786,3 +817,141 @@ def test_create_folder_returns_existing_on_400_duplicate() -> None:
786
817
  client._client = httpx.Client(transport=transport) # type: ignore[assignment]
787
818
  result = client.create_folder("42", "MySection")
788
819
  assert result["id"] == "77"
820
+
821
+
822
+ def test_create_folder_with_parent_id() -> None:
823
+ """create_folder sends parentId in the payload when provided."""
824
+ payload = {"id": "99", "title": "Sub", "spaceId": "42"}
825
+ transport = _MockTransport(_json_response(payload))
826
+ config = _make_config()
827
+ with ConfluenceClient(config) as client:
828
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
829
+ result = client.create_folder("42", "Sub", parent_id="77")
830
+ assert result["id"] == "99"
831
+ body = json.loads(transport.requests[0].content)
832
+ assert body["parentId"] == "77"
833
+
834
+
835
+ # ── get_page_inline_comments ───────────────────────────────────────────────────
836
+
837
+
838
+ def test_get_page_inline_comments_returns_comments() -> None:
839
+ payload = {
840
+ "results": [
841
+ {"id": "c1", "body": {"storage": {"value": "<p>Nice</p>"}}},
842
+ {"id": "c2", "body": {"storage": {"value": "<p>Fix this</p>"}}},
843
+ ],
844
+ "_links": {},
845
+ }
846
+ transport = _MockTransport(_json_response(payload))
847
+ config = _make_config()
848
+ with ConfluenceClient(config) as client:
849
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
850
+ comments = client.get_page_inline_comments("42")
851
+ assert len(comments) == 2
852
+ assert comments[0]["id"] == "c1"
853
+ assert comments[1]["id"] == "c2"
854
+ req = transport.requests[0]
855
+ assert "/pages/42/inline-comments" in str(req.url)
856
+ assert "resolution-status=open" in str(req.url)
857
+
858
+
859
+ def test_get_page_inline_comments_paginates() -> None:
860
+ page1 = {
861
+ "results": [{"id": "c1"}],
862
+ "_links": {"next": "/wiki/api/v2/pages/42/inline-comments?cursor=abc"},
863
+ }
864
+ page2 = {
865
+ "results": [{"id": "c2"}],
866
+ "_links": {},
867
+ }
868
+ transport = _MockTransport(_json_response(page1), _json_response(page2))
869
+ config = _make_config()
870
+ with ConfluenceClient(config) as client:
871
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
872
+ comments = client.get_page_inline_comments("42")
873
+ assert len(comments) == 2
874
+ assert len(transport.requests) == 2
875
+
876
+
877
+ # ── get_page_footer_comments ───────────────────────────────────────────────────
878
+
879
+
880
+ def test_get_page_footer_comments_returns_comments() -> None:
881
+ payload = {
882
+ "results": [{"id": "f1", "body": {"storage": {"value": "<p>Footer note</p>"}}}],
883
+ "_links": {},
884
+ }
885
+ transport = _MockTransport(_json_response(payload))
886
+ config = _make_config()
887
+ with ConfluenceClient(config) as client:
888
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
889
+ comments = client.get_page_footer_comments("42")
890
+ assert len(comments) == 1
891
+ assert comments[0]["id"] == "f1"
892
+ req = transport.requests[0]
893
+ assert "/pages/42/footer-comments" in str(req.url)
894
+
895
+
896
+ def test_get_page_footer_comments_paginates() -> None:
897
+ page1 = {
898
+ "results": [{"id": "f1"}],
899
+ "_links": {"next": "/wiki/api/v2/pages/42/footer-comments?cursor=xyz"},
900
+ }
901
+ page2 = {
902
+ "results": [{"id": "f2"}],
903
+ "_links": {},
904
+ }
905
+ transport = _MockTransport(_json_response(page1), _json_response(page2))
906
+ config = _make_config()
907
+ with ConfluenceClient(config) as client:
908
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
909
+ comments = client.get_page_footer_comments("42")
910
+ assert len(comments) == 2
911
+ assert len(transport.requests) == 2
912
+
913
+
914
+ # ── add_comment_reply ──────────────────────────────────────────────────────────
915
+
916
+
917
+ def test_add_comment_reply_posts_reply() -> None:
918
+ transport = _MockTransport(_json_response({"id": "reply-1"}))
919
+ config = _make_config()
920
+ with ConfluenceClient(config) as client:
921
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
922
+ client.add_comment_reply("42", "Thanks, fixed!")
923
+ req = transport.requests[0]
924
+ assert req.method == "POST"
925
+ assert "/content/42/child/comment" in str(req.url)
926
+ body = json.loads(req.content)
927
+ assert body["type"] == "comment"
928
+ assert "<p>Thanks, fixed!</p>" in body["body"]["storage"]["value"]
929
+
930
+
931
+ # ── resolve_footer_comment ─────────────────────────────────────────────────────
932
+
933
+
934
+ def test_resolve_footer_comment_sends_get_then_put() -> None:
935
+ existing = {
936
+ "id": "99",
937
+ "version": {"number": 2},
938
+ "body": {"storage": {"value": "<p>Old footer</p>"}},
939
+ "resolved": False,
940
+ }
941
+ transport = _MockTransport(
942
+ _json_response(existing), # GET
943
+ _json_response({}), # PUT
944
+ )
945
+ config = _make_config()
946
+ with ConfluenceClient(config) as client:
947
+ client._client = httpx.Client(transport=transport) # type: ignore[assignment]
948
+ client.resolve_footer_comment("99")
949
+ assert len(transport.requests) == 2
950
+ get_req = transport.requests[0]
951
+ assert "/footer-comments/99" in str(get_req.url)
952
+ put_req = transport.requests[1]
953
+ assert put_req.method == "PUT"
954
+ body = json.loads(put_req.content)
955
+ assert body["version"]["number"] == 3
956
+ assert body["resolved"] is True
957
+ assert body["body"]["value"] == "<p>Old footer</p>"