Skip to main content

Convert Markdown files into Anki cards via AnkiConnect

Project description

Markdown to Anki

markdown-to-anki is a lightweight Python CLI that automatically converts raw Markdown files into Anki cards using AnkiConnect. It's designed for learners and note-takers who prefer writing in Markdown and want to seamlessly sync their content into Anki.

Installation

pip install markdown-to-anki

Requires Python 3.10+ and AnkiConnect running in your Anki app.

Getting Started

Open Anki, then run once to register note models and do the first import:

m2a --folder ~/notes/anki init

From then on, use sync for day-to-day updates — it re-imports changed files and pushes everything to AnkiWeb:

m2a --folder ~/notes/anki sync

Commands

Command Description
m2a check Verify the AnkiConnect server is reachable.
m2a init Register note models, import media and notes. Use for first-time setup.
m2a sync Import media and notes, then trigger an AnkiWeb sync.
m2a sync_web Trigger an AnkiWeb sync only (no re-import). Useful for cron.

Only files modified within the last TIME_RANGE seconds (default 2 hours) are processed.

Force a full sync — to re-sync all notes regardless of modification time, touch every Markdown file first:

find ~/notes/anki -name "*.md" -exec touch {} +
m2a --folder ~/notes/anki sync

Configuration

Settings are resolved in this order — highest priority wins:

CLI flag  >  environment variable / .env  >  config file  >  default

CLI flags

m2a --folder ~/notes/anki --url http://localhost:8765 sync
m2a --resources ~/my-templates init
Flag Description Default
--folder Path to your Markdown notes dir see Config file
--url AnkiConnect URL http://localhost:8765
--resources Path to custom card templates/styles directory built-in resources

Config file

Create ~/.config/markdown-to-anki/config.yaml (respects $XDG_CONFIG_HOME):

md_folder: ~/notes/anki
time_range: 7200
anki_url: http://localhost:8765
resources_dir: ~/notes/anki-templates

Environment variables / .env

Place a .env file in the directory where you run m2a, or set the variables in your shell:

MD_FOLDER=~/notes/anki
TIME_RANGE=7200
ANKI_URL=http://localhost:8765
M2A_RESOURCES_DIR=~/notes/anki-templates

Markdown Style

Each Markdown file may contain one or more cards. Use YAML frontmatter for per-file metadata and the comment separators below to split cards and fields:

---
deck: M2A::Example::Math
tags: math, theorem
model: m2a-basic
# skip: 1   # uncomment to skip this file
---

<!--CARD-->

### Front of card 1

<!--FIELD-->

Back of card 1

<!--CARD-->
<!--TAGS: hard, exam-->

### Front of card 2

<!--FIELD-->

Back of card 2

Supported metadata keys: deck, tags, model, skip.

Per-card tags via <!--TAGS: a, b--> (anywhere inside a card section) are merged with the file-level tags from frontmatter.

Built-in note types

Model Fields Description
m2a-basic Front, Back Simple front → back card
m2a-basic-reverse Front, Back Front → back and back → front
m2a-cloze Text, Extra Cloze deletion card
m2a-english Word, Audio, Meaning Vocabulary card with audio support

Cloze cards

Use Anki's native {{c1::...}} syntax in the Text field. Multiple deletions (c1, c2, …) are supported in a single card.

---
deck: M2A::Example::Cloze
model: m2a-cloze
---

<!--CARD-->

The capital of {{c1::France}} is {{c2::Paris}}.

<!--FIELD-->

European geography.

<!--CARD-->

{{c1::Water}} boils at {{c2::100}}°C at standard pressure.

<!--FIELD-->

Basic chemistry fact.

The Text field holds the cloze content; Extra is shown on the back and can hold supplementary notes.

Custom Resources

Point --resources (or resources_dir / M2A_RESOURCES_DIR) at your own directory to override built-in templates/styles and define new note types. Only the files you provide are overridden — everything else falls back to the built-in defaults.

Override built-in templates or styles

Mirror the built-in layout under your resources directory:

my-resources/
└── anki/
    ├── styles/
    │   └── basic.css          # overrides the built-in basic.css
    └── templates/
        └── basic/
            └── front.html     # overrides the built-in front template

Define custom note types

Place a YAML file per model inside a models/ subdirectory:

my-resources/
├── models/
│   └── my-vocab.yaml
├── styles/
│   └── my-vocab.css
└── templates/
    └── my-vocab/
        ├── front.html
        └── back.html

models/my-vocab.yaml:

name: my-vocab
fields:
  - Word
  - Definition
  - Example
is_cloze: false
css_file: styles/my-vocab.css      # relative to my-resources/
templates:
  - name: word-to-definition
    front_file: templates/my-vocab/front.html
    back_file: templates/my-vocab/back.html

Inline CSS and HTML are also supported instead of file references:

name: my-simple
fields:
  - Question
  - Hint
  - Answer
css: ".card { font-family: sans-serif; }"
templates:
  - name: card
    front: "<div>{{Question}}</div><div class='hint'>{{Hint}}</div>"
    back: "<div>{{Answer}}</div>"

Fields map to <!--FIELD--> sections in order. A card with three fields uses two separators:

---
model: my-vocab
deck: Languages::English
---

<!--CARD-->

hello

<!--FIELD-->

A common English greeting.

<!--FIELD-->

"Hello, how are you?" — used when meeting someone.

Run m2a --resources my-resources/ init to register your models in Anki before syncing notes.

Limitation

  • DO NOT use dark theme in Anki — the rendered Markdown style does not support it.
  • Attachments (images, audio) must be placed in the same directory as the Markdown file, or a subdirectory of it.

Cronjob

Automatically sync your notes to AnkiWeb hourly:

crontab -e

# sync hourly (replace /usr/local/bin/m2a with the output of: which m2a)
5 * * * *  /usr/local/bin/m2a --folder ~/notes/anki sync_web &>/dev/null

Examples

See the example notes for working Markdown files covering basic cards, cloze deletions, math, code, and audio.

License

This project is open-sourced and licensed under the MIT license.

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

markdown_to_anki-0.1.4.tar.gz (75.9 kB view details)

Uploaded Source

Built Distribution

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

markdown_to_anki-0.1.4-py3-none-any.whl (20.7 kB view details)

Uploaded Python 3

File details

Details for the file markdown_to_anki-0.1.4.tar.gz.

File metadata

  • Download URL: markdown_to_anki-0.1.4.tar.gz
  • Upload date:
  • Size: 75.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.21 {"installer":{"name":"uv","version":"0.11.21","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for markdown_to_anki-0.1.4.tar.gz
Algorithm Hash digest
SHA256 e9f3cf7f723a8dbecde3704c78bbc6e78d8d21cba2ca9fa2d496aa44485a6c91
MD5 a043e2003c5b4b1f7938045d38963bc5
BLAKE2b-256 97697d10e6362c74de7088d8090bb8697a1fa63514cda570bd5bdb4fed446719

See more details on using hashes here.

File details

Details for the file markdown_to_anki-0.1.4-py3-none-any.whl.

File metadata

  • Download URL: markdown_to_anki-0.1.4-py3-none-any.whl
  • Upload date:
  • Size: 20.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.21 {"installer":{"name":"uv","version":"0.11.21","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for markdown_to_anki-0.1.4-py3-none-any.whl
Algorithm Hash digest
SHA256 89a79f1293ec829c2704221fbae5953ee45d632296cdf0333d701abcc3923cbf
MD5 b31e9840ebe17788cca9b44bd1efceeb
BLAKE2b-256 4bca4b823f8cd700ece9da54e192ad19cf6d188bb14a8cde9630fac178d88137

See more details on using hashes here.

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