Skip to main content

Pure-Python Markdown ↔ Atlassian Document Format (ADF) converter with lossless round-tripping

Project description

adflux

CI Release PyPI Python License: MIT

A pure-Python converter between Markdown (CommonMark + GFM) and Atlassian Document Format (ADF) — the JSON document model used by Confluence and Jira. Round-trips losslessly, including ADF-only constructs like panels, status badges, mentions, task lists, expand sections, and macros.

flowchart LR
  MD[Markdown] -->|markdown-it-py| IR((Internal IR + adf-* envelopes))
  ADF[ADF JSON] -->|mapping.yaml| IR
  PF[Panflute JSON] -->|pf.load| IR
  IR -->|options filter| W{writer}
  W -->|CommonMark serializer| MD2[Markdown]
  W -->|mapping.yaml| ADF2[ADF JSON]
  W -->|pf.dump| PF2[Panflute JSON]

Why adflux

ADF is JSON, but no mature pure-Python tool converts it to and from Markdown without losing information. adflux fills that gap:

  • No system dependencies. pip install adflux — no Pandoc binary, no Node, no Ruby.
  • Lossless ADF round-trips. A declarative mapping.yaml plus a small envelope convention round-trips every ADF construct, including ones the library has never seen.
  • Idiomatic Markdown output. Panels become GitHub alert blockquotes, expand sections become <details>, smart cards become autolinks. The output looks correct in GitHub, VS Code, and any standard viewer.
  • Conversion options make lossy / lossless behavior explicit and selectable per call.

Install

pip install adflux

Requires Python ≥ 3.11. Linux, macOS, and Windows are CI-tested.

Quick start

from adflux import convert

# Markdown → ADF JSON
adf = convert(open("README.md").read(), src="md", dst="adf")

# ADF JSON → Markdown, dropping ADF-only constructs to plain content
md = convert(adf, src="adf", dst="md", options={"envelopes": "drop"})
adflux convert --from md  --to adf README.md > readme.adf.json
adflux convert --from adf --to md page.json
adflux convert --from adf --to md --option envelopes=drop page.json
adflux validate page.adf.json
adflux list-options

Full API and CLI reference: docs/usage.md.

Conversion options

Option Choices Default Description
envelopes keep, drop, keep-strict keep How ADF envelope nodes are handled on lossy targets.
jira-strict true, false false Reject non-Jira ADF nodes during serialization.

Worked examples: docs/options.md.

Markdown rendering of ADF constructs

ADF construct Rendered as
panel (info/note/warning/…) GitHub alert blockquote (> [!NOTE], > [!TIP], …)
expand <details><summary>title</summary>…</details>
inlineCard / blockCard / embedCard Autolink (<https://…>)
taskList / taskItem GFM task list (- [ ], - [x])
emoji, mention Plain text (🚀, @alice)
Everything else HTML-comment marker (<!--adf:status …-->)

The reader recognises every form on its way back, so MD → ADF → MD is stable.

Supported formats

Format Aliases Description
Markdown md, markdown CommonMark + GFM via markdown-it-py
ADF adf Atlassian Document Format JSON
Panflute panflute, pf Pandoc JSON AST for integration with panflute tools

How it works

  • IR: an in-memory document tree built on panflute AST classes (used purely as Python data structures — no Pandoc binary involved), with Div / Span envelopes for ADF-only constructs.
  • Envelopes: class prefix adf-<nodeType> plus a base64-JSON blob for complex attributes. A universal adf-raw fallback guarantees zero data loss.
  • Mapping: every ADF node type is described in src/adflux/formats/adf/mapping.yaml. Adding a new node type is a YAML edit.

Deeper dives: docs/design.md · docs/architecture.md · docs/options.md · docs/fidelity-matrix.md · docs/extending.md · docs/e2e-testing.md.

Examples

Runnable scripts in examples/:

python examples/md_to_adf.py README.md
python examples/adf_to_markdown.py
python examples/confluence_roundtrip.py

Development

git clone https://github.com/mikejhill/adflux
cd adflux
uv sync --all-groups
uv run poe check

Common tasks via poethepoet:

poe test         # unit + integration + round-trip + property tests
poe test-e2e     # live Confluence + Jira round-trip suite (requires .env)
poe lint         # ruff format --check + ruff check
poe typecheck    # ty check src
poe cov          # pytest with coverage report
poe check        # lint + typecheck + test (run before opening a PR)
poe build        # sdist + wheel into ./dist

CI runs the full matrix on Linux, macOS, and Windows for Python 3.11–3.13.

Releasing

Releases are automated. Land changes on main using Conventional Commits, then run the Release workflow from the GitHub Actions tab. go-semantic-release computes the next version, builds the sdist + wheel, creates a tagged GitHub Release, and publishes to PyPI via OIDC trusted publishing.

Contributing

Issues and pull requests are welcome. Before opening a PR:

  1. poe check must pass (lint + typecheck + tests).
  2. Use a Conventional Commit message (feat:, fix:, docs:, …).
  3. New ADF node types need a fixture in tests/roundtrip/test_node_coverage.py.

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

adflux-0.3.0.tar.gz (69.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

adflux-0.3.0-py3-none-any.whl (45.5 kB view details)

Uploaded Python 3

File details

Details for the file adflux-0.3.0.tar.gz.

File metadata

  • Download URL: adflux-0.3.0.tar.gz
  • Upload date:
  • Size: 69.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for adflux-0.3.0.tar.gz
Algorithm Hash digest
SHA256 f223efd704c055130c6ed2bebce13213743e7a2546eea2a8524c61310c6ee1ca
MD5 4a06f222044509457e9cf41c3ee315a4
BLAKE2b-256 1829496208536e7a297a0876b6f34dc5ab657a59eafc42046c655e6d9c59d7f1

See more details on using hashes here.

Provenance

The following attestation bundles were made for adflux-0.3.0.tar.gz:

Publisher: release.yml on mikejhill/adflux

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file adflux-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: adflux-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 45.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for adflux-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 d0bf5af6a84b1db1e7659a3373ac097f34b760e8efe4ba87430b860d763beccb
MD5 fd45cbd8f8d2e1279a2f881c42f8e034
BLAKE2b-256 58419aebd0ecbe5a392d68f747543bc061809e8ff11e4045344cdbe4a65d5c1a

See more details on using hashes here.

Provenance

The following attestation bundles were made for adflux-0.3.0-py3-none-any.whl:

Publisher: release.yml on mikejhill/adflux

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page