Python client for ETAPI of Trilium Note. With some extra features powered by Python :)
Project description
๐ trilium-py
English | ็ฎไฝไธญๆ
Python client for Trilium Note's ETAPI and Web API, with additional advanced features :)
๐ฆฎ Table of Contents
- ๐ trilium-py
- ๐ฆฎ Table of Contents
- ๐ง Installation
- ๐ Initialization
- ๐ (Basic) ETAPI Usage
- ๐ Application Information
- ๐ Search note
- ๐ญ Create Note
- ๐ Get note
- ๐ Update note
- ๐๏ธ Delete note
- ๐ Create note clone/branch
- ๐ Get note clone info
- โ๏ธ Modify note clone info
- โ Delete note clone
- ๐ Day note
- ๐ค Export note
- ๐ฅ import note
- ๐พ Save revision
- ๐พ Create data backup
- ๐ท Create attribute
- Get attachment list
- Get attachment info
- Update attachment info
- Get attachment content
- Update attachment content
- Create attachment
- (Advanced Usage) โ TODO List
- (Advanced Usage) ๐ Upload Markdown files
- (Advanced Usage) ๐จ Beautify notes
- (Advanced Usage) ๐งน Sort note content
- (Advanced Usage) ๐งน Delete empty new note
- (Advanced Usage) ๐๏ธ Optimize image size
- (Advanced Usage) ๐ Automatically Add Internal Links
- (Advanced Usage) ๐๐ณ Traverse Note Tree
- (Advanced Usage) ๐๏ธ๐ Periodic TODOs
- (Basic) Web API Usage
- ๐ ๏ธ Develop
- ๐ Original OpenAPI Documentation
- ๐ฐ Donation
- ๐ Thanks
๐ง Installation
python3 -m pip install trilium-py --user
๐ Initialization
In the following code example, ea stands for ETAPI, and wa stands for Web API. Please perform the corresponding
initialization process based on whether you need to use the ETAPI or the Web API.
ETAPI initialization
If you have a ETAPI token, change the server_url and token to yours.
from trilium_py.client import ETAPI
server_url = 'http://localhost:8080'
token = 'YOUR_TOKEN'
ea = ETAPI(server_url, token)
If you haven't created ETAPI token, you can create one with your password. Please note, you can only see this token once, please save it if you want to reuse the token.
from trilium_py.client import ETAPI
server_url = 'http://localhost:8080'
password = '1234'
ea = ETAPI(server_url)
token = ea.login(password)
print(token)
After initialization, you can use Trilium ETAPI with python now.
Web API initialization
You need to login every time you use the web API due to the CSRF limit.
from trilium_py.src.trilium_py.web_client import WEBAPI
server_url = 'http://localhost:8080'
password = '1234'
wa = WEBAPI(server_url)
wa.login(password)
After initialization, you can use Trilium web API with python now.
๐ (Basic) ETAPI Usage
These are basic function that Trilium's ETAPI provides. Down below are some simple example code to use this package.
๐ Application Information
To start with, you can get the application information like this.
print(ea.app_info())
It should give you the version of your server application and some extra information.
๐ Search note
Search note with keyword.
res = ea.search_note(
search="python",
)
for x in res['results']:
print(x['noteId'], x['title'])
Search with regular expression. For example, search and get all child notes under certain note:
res = ea.search_note(
# regular expression search for note title
search="note.title %= '.*'",
ancestorNoteId="Parent Note ID",
fastSearch=False,
orderBy=["title"],
limit=100,
)
Note: limit will not work without orderBy.
๐ญ Create Note
You can create a simple note like this.
res = ea.create_note(
parentNoteId="root",
title="Simple note 1",
type="text",
content="Simple note example",
noteId="note1",
dateCreated="2026-01-01 12:34:56.000+0900",
)
The noteId is not mandatory, if not provided, Trilium will generate a random one. You can retrieve it in the return.
dateCreated is also not mandatory. It can be used to assist importing notes from other apps, preserving creation date.
noteId = res['note']['noteId']
๐ผ๏ธ Create Image note
Image note is a special kind of note. You can create an image note with minimal information like this. The image_file
refers to the path of image.
res = ea.create_image_note(
parentNoteId="root",
title="Image note 1",
image_file="shield.png",
)
๐ Get note
To retrieve the note's content.
ea.get_note_content("noteid")
You can get a note metadata by its id.
ea.get_note(note_id)
๐ Update note
Update note content
ea.update_note_content("noteid", "updated by python")
Modify note title
ea.patch_note(
noteId="noteid",
title="Python client moded",
)
๐๏ธ Delete note
Simply delete a note by id.
ea.delete_note("noteid")
๐ Create note clone/branch
In the ETAPI, this is called a branch, but in the Trilium UI it represents a note clone.
Here is an example to create a clone of note2 under note1.
res = ea.create_branch(
noteId="note2",
parentNoteId="note1",
)
It will return the note clone info like this:
{
'branchId': 'note1_note2',
'noteId': 'note2',
'parentNoteId': 'note1',
'prefix': None,
'notePosition': 10,
'isExpanded': False,
'utcDateModified': '2025-10-21T03:25:16.397Z'
}
๐ Get note clone info
Retrieve details about a specific note clone (branch) by its branchId
res = ea.get_branch(branchId="note1_note2")
โ๏ธ Modify note clone info
Update an existing note cloneโs display attributes such as prefix, position, or expansion state
res = ea.patch_branch(branchId="note1_note2", notePosition=0, prefix="patched", isExpanded=False)
โ Delete note clone
Remove a note clone (branch) from the tree.
res = ea.delete_branch(branchId="note1_note2")
๐ Day note
You can get the content of a certain date with get_day_note. The date string should be in format of "%Y-%m-%d", e.g. "
2022-02-25".
ea.get_day_note("2022-02-25")
Then set/update a day note with set_day_note. The content should be a (html) string.
ea.set_day_note(date, new_content)
๐ค Export note
Export note comes in two formats html or markdown/md. Setting noteId to root will export all notes.
res = ea.export_note(
noteId='sK5fn4T6yZRI',
format='md',
save_path='/home/nate/data/1/test.zip',
)
๐ฅ import note
This is the built-in feature in trilium. The input file should be a zip file.
res = ea.import_note(
noteId='sK5fn4T6yZRI',
file_path='/home/nate/data/1/test.zip',
)
๐พ Save revision
Save note revision manually.
res = ea.save_revision(
noteId='MJzyFRXAVaC9',
)
๐พ Create data backup
This example will create a database backup file like this trilium-data/backup/backup-test.db.
res = ea.backup("test")
You can use the cron utility in Linux to schedule regular automatic backups. For example, to set up a daily backup at 3: 00 AM, you would use the following cron expression:
0 3 * * * python /path/to/backup-script.py
๐ท Create attribute
You can create a tag for a note
res = ea.create_attribute(
noteId='noteid',
type='label',
name='name_of_the_tag',
value='value_of_the_tag',
isInheritable=True
)
The noteId is not mandatory, if not provided, Trilium will generate a random one. You can retrieve it in the return.
noteId = res['note']['noteId']
Get attachment list
Get all attachments of a single note.
res = ea.get_attachments('uMJt0Ajr1CuC')
Get attachment info
Get image title and etc.
res = ea.get_attachment('Y5V6pYq6nwXo')
Update attachment info
Change image title and etc.
res = ea.update_attachment(
attachmentId='2b7pPzqocS1s', title='hello etapi', role='image', mime='image/png'
)
Get attachment content
Get the real image file
res = ea.get_attachment_content('icpDE4orQxlI')
with open('1.png', 'wb') as f:
f.write(res)
Update attachment content
Replace the image with new one
res = ea.update_attachment_content('icWqV6zFtE0V', '/home/nate/data/1.png')
Create attachment
Upload a image file as attachment of a note.
res = ea.create_attachment(
ownerId='8m8luXym5LxT',
file_path='/home/nate/data/ksnip_20230630-103509.png',
)
(Advanced Usage) โ TODO List
With the power of Python, I have expanded the basic usage of ETAPI. You can do something with todo list now.
Add TODO item
You can use add_todo to add a TODO item, param is the TODO description
ea.add_todo("ไนฐๆๅฎๅฎ")
Check/Uncheck a TODO item
param is the index of the TODO item
ea.todo_check(0)
ea.todo_uncheck(1)
Update a TODO item
Use update_todo to update a TODO item description at certain index.
ea.update_todo(0, "ๅป็ ๅคดๆด็น่ฏๆก")
Delete a TODO item
Remove a TODO item by its index.
ea.delete_todo(1)
Move yesterday's unfinished todo to today
As the title suggests, you can move yesterday's unfinished things to today. Unfinished todo's will be deleted from yesterday's note.
ea.move_yesterday_unfinished_todo_to_today()
(Advanced Usage) ๐ Upload Markdown files
Upload single Markdown file with images
You can import Markdown file with images into Trilium now! Trilium-py will help you to upload the images and fix the links for you!
res = ea.upload_md_file(
parentNoteId="root",
file="./md-demo/manjaro ไฟฎๆนcaps lock.md",
hasFrontMatter=False,
cleanText=False
)
hasFrontMatter is optional. If set to true, it is expected that the markdown file has front matter block such as:
---
title: NoteTitle
updated: 2026-01-01 12:45:56Z
created: 2026-01-01 12:34:56Z
latitude: 10.1234
longitude: 100.1234
altitude: 0.0000
---
cleanText is optional. If set to true, it will tidy up the note import, including adding paragraph breaks.
Disable math formula parsing
When uploading Markdown files that contain dollar signs ($) which are not meant to be interpreted as math formulas, you can disable the math formula parsing:
res = ea.upload_md_file(
parentNoteId="root",
file="./md-demo/document_with_dollar_signs.md",
parse_math=False, # Disable math formula parsing
)
Bulk upload Markdown files in a folder
You can upload a folder with lots of Markdown files to Trilium and preserve the folder structure!
Import from VNote
Say, upload all the notes from VNote, simply do this:
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="~/data/vnotebook/",
ignoreFolder=['vx_notebook', 'vx_recycle_bin', 'vx_images', '_v_images'],
)
Import from Joplin
To import from Joplin, first extract your Joplin notes to Markdown + Front Matter:
File > Export All > MD - Markdown + Front Matter
Then run the following snippet, replacing mdFolder to your actual exported notes location.
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="/home/nate/data/joplin_data/",
ignoreFolder=['_resources', ],
hasFrontMatter=True,
cleanText=True
)
Importing with Front Matter will ensure consistent note creation time.
Import from Logseq
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="/home/nate/data/logseq_data/",
ignoreFolder=['assets', 'logseq'],
)
Import from Obsidian
Obsidian has a very unique linking system for files. You should use obsidian-export to convert a Obsidian vault to regular Markdown files. Then you should be able to import the note into Trilium with trilium-py.
Convert it first.
obsidian-export /path/to/your/vault /out
Then import just like a normal markdown, trilium-py will handle the images for you.
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="E:/data/out",
)
Import from Youdao Note/ๆ้ไบ็ฌ่ฎฐ
Youdao does not provide an export feature anymore. Luckily, you can use https://github.com/DeppWang/youdaonote-pull to download your notes and convert them into markdown files. After that, trilium-py should be able to help you import them.
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="/home/nate/gitRepo/youdaonote-pull/out/",
)
Import from Turtl
You need to convert Turtl from json to markdown first. See turtl-to-markdown for details.
Then you can import with trilium-py like this:
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="/home/nate/gitRepo/turtl-to-markdown/out/",
ignoreFolder=['_resources'],
)
Import from other markdown software
In general, markdown files have variety of standards. You can always try import them with
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="/home/nate/data/your_markdown_files/",
)
If there is any problem, please feel free to create an issue.
Disable math formula parsing for folder import
Similarly, you can disable math formula parsing when importing a folder of Markdown files:
res = ea.upload_md_folder(
parentNoteId="root",
mdFolder="~/data/financial_documents/",
parse_math=False, # Disable math formula parsing for all files in the folder
)
(Advanced Usage) ๐จ Beautify notes
Because of the constraints imposed by the library utilized by Trilium, imported notes may experience minor formatting problems. These issues include an additional line appearing at the end of code blocks, images becoming integrated with the note content, and the absence of line breaks between headings, resulting in a cramped appearance of the note content.
When collecting information, some notes copied from the web, saved via clipping plugins, or imported from * other applications* may contain redundant line breaks, inconsistent heading levels, or messy formatting, making them look cluttered.
Beautify notes will automatically cleans up unnecessary content, normalizes headings and paragraph layouts, and makes your notes cleaner, clearer, and easier to read.
Here is what you can do to beautify your note.
Beautify a note
Specify a note id to beautify note content.
ea.beautify_note('krm8B9JthNfi')
Beautify a note and its child notes
ea.beautify_sub_notes('tlPuzU2szLJh')
(Advanced Usage) ๐งน Sort note content
Sort a note by the heading names. This feature could prove invaluable for notes containing extensive lists, such as book titles sorted into various genres. It's equally useful for managing browser bookmarks or collecting links.
Additionally, you have the option to specify a language code for sorting based on your local language. This enhances the sorting process and tailors it to your linguistic preferences.
res = ea.sort_note_content('lPxtkknjR2bJ')
res = ea.sort_note_content('y6hROhWjNmHQ', 'zh_CN.UTF-8')
(Advanced Usage) ๐งน Delete empty new note
Sometimes I inadvertently create numerous "new notes" which remain undeleted within my note tree. These "new notes" clutter my workspace, scattered across various locations. I made this bulk deletion of these empty "new notes." Additionally, it generates warning messages for "new notes" that contain content, maybe we should change the title for those notes.
ea.delete_empty_note()
(Advanced Usage) ๐๏ธ Optimize image size
Try to reduce image size by using PIL's optimize feature. If the image in your note is not compressed, you can try this. I've successfully convert a note of 44MB to 9.9MB after this process. Backup your data before try this.
The default quality is set to 90.
optimize_image_attachments will keep the original image format and try to compress it.
ea.optimize_image_attachments('uMJt0Ajr1CuC')
To save even more space, you can try the following method.
The optimize_image_attachments_to_webp function converts images to the WebP format, significantly reducing file
sizes. Based on my experience, WebP images can be as little as 25% to 50% of the size of PNG images.
ea.optimize_image_attachments_to_webp('H2q3901uFDCH')
This action can save significant space if you have many clipped pages. Whoever invented WebP is a genius.
(Advanced Usage) ๐ Automatically Add Internal Links
This feature allows you to automatically create internal links within your notes. Letโs take a look at how it works.
Example
Here is a sample note:
After running a single line of code:
auto_create_internal_link('put_note_id_here')
The note transforms into this:
As you can see, some text has been replaced with internal links. The feature follows these rules:
- Title Match: Content that matches any other note's title is replaced with an internal link.
- Duplicate Titles Ignored: If multiple notes share the same title, no link is created for that title.
- Longer Matches First: Longer titles take precedence. For example, in the above example, "Nate River" is linked, not just "River."
- Existing Links Remain: Pre-existing links in the text are left untouched.
However, some words like "make" and "work" in the example are part of my "English Words That I Do Not Know" note. Since they are common and frequently used, I donโt want them to create excessive internal links.
Excluding Notes from Internal Linking
To prevent certain notes from being linked:
- Add the tag
#ignoreAutoInternalLinkto a note. This note (and optionally its sub-notes) will be excluded from link creation. - You can make it inheritableโyou can apply it to a parent note and inheritable, then it will be automatically exclude all its sub-notes.
Hereโs how it looks after applying the exclusion rule:
The result is cleaner and more intentional.
Special Case: Duplicate Titles
When multiple notes share the same title, a specific condition allows for internal links:
- Direct Sub-Notes Have Priority: Direct child notes take precedence over other notes with the same title.
For instance:
In this case, the note TriliumNext links "How to compile" to its own child note, not the one from Trilium.
Final Rule: No Self-Linking
A note will never create an internal link to itself.
Code Samples
Add an internal link to a specific note by its ID:
auto_create_internal_link('gLmmsIM8yPqx')
Add internal links for multiple notes:
auto_create_internal_link(target_notes=['gLmmsIM8yPqx', 'T4Ui3wNByO03'])
(Experimental - Use with Caution) Add internal links to all text notes:
This is an experimental feature. Backup your database before using it, as it may irreversibly modify your notes. If issues occur, please provide a minimal note sample to help debug.
auto_create_internal_link(process_all_notes=True)
(Advanced Usage) ๐๐ณ Traverse Note Tree
Fetch a note's title and content along with its descendants'. Great for reassembling large notes split into smaller child notes.
The method can be bfs or dfs.
res = ea.traverse_note_tree('XdOlGz7MeYWC', depth=3, limit=100, method='bfs')
for x in res:
logger.info(x)
(Advanced Usage) ๐๏ธ๐ Periodic TODOs
You can use add_periodic_todos to automatically add recurring tasks (daily, weekly, monthly, yearly). If a task is due today and not already in todayโs TODO list, it will be added automatically.
periodic_todos = [
# Stretch body every day
{"content": "Stretch body", "type": "daily"},
# Buy milk for Nriver every day :)
{"content": "Buy milk for Nriver :)", "type": "daily"},
# Wash clothes on every Saturday
{"content": "Wash clothes", "type": "weekly", "weekday": 6},
# Write monthly report on the first day of every month
{"content": "Write monthly report", "type": "monthly", "day": 1},
# Pay rent on the last day of every month
{"content": "Pay rent", "type": "monthly", "day": -1},
# Buy anti-allergy medicine on the 10th of every month
{"content": "Buy anti-allergy medicine", "type": "monthly", "day": 10},
# Annual review every year on December 31
{"content": "Annual review", "type": "yearly", "month": 12, "day": 31},
# Happy new year on January 1 :)
{"content": "Happy new year :)", "type": "yearly", "month": 1, "day": 1},
# Pay bills on the 15th of every month until 2077-01-01
{"content": "Pay bills", "type": "monthly", "day": 15, "end_date": "2077-01-01"},
# Team meeting every Monday starting from 2025-01-01
{"content": "Team meeting", "type": "weekly", "weekday": 1, "start_date": "2025-01-01"},
# Special project from 2025-09-26 to 2025-09-30
{"content": "Work on special project", "type": "daily", "start_date": "2025-09-26", "end_date": "2025-09-30"},
]
ea.add_periodic_todos(periodic_todos)
(Basic) Web API Usage
These features are made based on the web API from Trilium's web client. Be sure you have done the initialization before use it.
๐ Get note content
Easy to get note content by note id.
res = wa.get_note_content('RfhYrtyQLU8o')
๐ Protected session / encrypted notes
You can enter protected session by login_protected_session method. Then you can access the encrypted notes after entering protected session.
wa.enter_protected_session(password)
res = wa.get_note_content(protected_note_id)
Leave protected session by logout_protected_session method.
wa.leave_protected_session()
๐ Update note content
Simply pass the note id and its content. If the note is protected, you need to enter the protected session first.
wa.update_note_content(note_id, '<p>protected note content updated by trilium-py :)</p>')
๐ฃ Share note & cancel share note
wa.share_note('your_note_id')
wa.cancel_share_note('RfhYrtyQLU8o')
๐ ๏ธ Develop
Install with pip egg link to make package change without reinstall.
python -m pip install --user -e .
๐ Original OpenAPI Documentation
The original OpenAPI document is here. You can open it with swagger editor.
๐ฐ Donation
If you enjoy using Trilium-py, your support would mean a lot!
Alipay
WeChat Pay
๐ Thanks
Thanks Matt Wilkie for donating 10.0 USD!
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distributions
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 trilium_py-1.3.6-py3-none-any.whl.
File metadata
- Download URL: trilium_py-1.3.6-py3-none-any.whl
- Upload date:
- Size: 49.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
bde4bf8f399bad1083d25f57c6d171c6b51499ac25a26d58e205e266bb0f7e1d
|
|
| MD5 |
9231630038cd5ee09fa196763714dfc6
|
|
| BLAKE2b-256 |
66bfada33933b95f385ec3a1947db9374c0b8808cccf0f4b30dc17981750f7a9
|