A parser for Org Mode documents
Project description
Overview
This project implements a Python parser for Emacs Org-Mode files providing nice semantic nodes in a hierarchical document tree. It's based on this Tree-Sitter grammar.
org-parser was created as an alternative to the wonderful, but
admittedly incomplete and unmaintained
orgparse. The main
improvements on top of orgparse are:
- Almost complete feature set, including markup, blocks, tables and more niche constructs such as Babel calls.
- Fully mutable tree for easy edits that preserves original document whitespace & formatting.
- Better document error handling and reporting, useful for validation.
Installation
pip install org-parser
Features
- Semantic nodes with all the expected attributes
>>> from org_parser import loads
>>> doc = loads('''
... #+TITLE: Document
... #+AUTHOR: Idorobots
... :PROPERTIES:
... :name: value
... :END:
... An example document.
... * Heading 1
... ** Heading 2 :tag:
... *** Heading 3
... ''')
>>> doc.title
RichText('Document')
>>> doc.body
[BlankLine(), Paragraph(body=RichText('An example document.\n'))]
>>> doc.body_text
'\nAn example document.\n'
>>> doc.properties["name"]
RichText('value')
>>> len(doc.all_headings)
3
>>> doc.all_headings[1].tags
['tag']
- Structured traversal
>>> from org_parser import loads
>>> doc = loads('''
... * Heading 1
... ** Heading 2
... *** Heading 3
... ''')
>>> doc.children[0].title
RichText('Heading 1')
>>> doc.children[0].children[0].title
RichText('Heading 2')
>>> doc.children[0].children[0].parent.title == "Heading 1"
True
>>> doc.children[0].children[0].siblings
[]
- Original source whitespace and formatting preserved by default
>>> from org_parser import loads
>>> doc = loads('''
... #+TITLE: Original formatting
... * Heading 1
... Indented section
... remains indented
... in-tree.
... ''')
>>> print(str(doc.children[0]))
* Heading 1
Indented section
remains indented
in-tree.
>>> print(doc.render())
#+TITLE: Original formatting
* Heading 1
Indented section
remains indented
in-tree.
- Fully mutable tree allowing dynamic edits
>>> from org_parser import loads
>>> from org_parser.element import Paragraph
>>> doc = loads('''
... #+TITLE: Document title
... * Heading 1
... Body text.
... ''')
>>> doc.title
RichText('Document title')
>>> doc.title.text = "Another title"
>>> doc[0].body = [Paragraph.from_source("New *and* improved!")]
>>> doc[0].title.text = "Improved heading"
>>> print(doc.render())
#+TITLE: Another title
* Improved heading
New *and* improved!
- Org Table support
>>> from org_parser import loads
>>> doc = loads('''
... |Value|Double|
... |1| |
... |2| |
... |3| |
... ''')
>>> for r in doc.body[1].rows[1:]:
... r[1].text = str(2 * int(r[0].text))
...
>>> print(doc.render())
| Value | Double |
| 1 | 2 |
| 2 | 4 |
| 3 | 6 |
- Rich Text support
>>> from org_parser.text import RichText
>>> text = RichText.from_source("Supports *org-mode* /markup/ and inline_{objects}: <2026-03-29>")
>>> text
RichText('Supports *org-mode* /markup/ and inline_{objects}: <2026-03-29>')
>>> text.parts
[PlainText(text='Supports '), Bold(body=[PlainText(text='org-mode')]), PlainText(text=' '), Italic(body=[PlainText(text='markup')]), PlainText(text=' and inline'), Subscript(body=[PlainText(text='objects')], form='{}'), PlainText(text=': '), Timestamp(is_active=True, start_year=2026, start_month=3, start_day=29), PlainText(text='')]
- Error recovery
>>> from org_parser import loads
>>> doc = loads('''
... * Heading
... SCHEDULED: yesterday
... ''')
>>> doc[0].scheduled is None
True
>>> doc.errors
[ParseError(start_point=Point(row=2, column=0), end_point=Point(row=2, column=20), text='SCHEDULED: yesterday')]
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file org_parser-0.23.5.tar.gz.
File metadata
- Download URL: org_parser-0.23.5.tar.gz
- Upload date:
- Size: 69.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ced1fc204c6da52384cb7036e0624b4fba9a5fd271f67c03f72d4a661cd58d99
|
|
| MD5 |
9dae85e40c2970618c582a1cbd10144e
|
|
| BLAKE2b-256 |
d723411118208ee7128ce29ab540b8190d153acc62c92104300ee3f43695feb8
|
File details
Details for the file org_parser-0.23.5-py3-none-any.whl.
File metadata
- Download URL: org_parser-0.23.5-py3-none-any.whl
- Upload date:
- Size: 85.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8335a6141365fe18d029a5997c61f7909c453952ecf179c43aeab82fc16cab18
|
|
| MD5 |
4552f4ff43460f007b356371f5ea8533
|
|
| BLAKE2b-256 |
1a5409fcb74649b8a3bbc27fc9a628025050217185af800757dec7f466beaef1
|