File-based social media publisher. Write markdown, post to LinkedIn, Bluesky, and Mastodon.
Project description
mdsend — File-based social media publisher
Write markdown posts, run mdsend, and it cross-posts to your social media platforms.
The main purpose is simple: set up a git repo with a simple directory structure and publish from it to various destinations. Your posts are in a single unified format that works across different social media platforms. You get a version-controlled archive that can be queried and managed by AI should the need arise.
Want to know what you've been focusing on, what your next posts should be, how to improve your writing, or which tags to use? Just ask your AI tool — it can read the repo and answer and provide you with a perspective.
This approach keeps you disconnected from the destination network so you don't get pulled into everyone else's feeds.
Supported platforms
| Platform | Max chars | Max media | API auth |
|---|---|---|---|
| 3,000 | 1 image | OAuth 2.0 (Bearer) | |
| Bluesky | 300 | 4 images | AT Protocol (App Password) |
| Mastodon | 500 | 4 images | OAuth (Access Token) |
How it works
Each post is a directory named YYYY-MM-DD_HH-MM_slug/ containing:
post.md— the post text (markdown or plain text)- Optional media files (jpg, png, gif, mp4, mov)
posts/
├── 2026-05-03_15-30_my-thoughts/
│ ├── post.md
│ └── image.jpg
└── 2026-05-04_18-00_hello-world/
└── post.md
Per-post platform targeting
By default every post goes to all platforms. To control this per post,
add an optional front matter block at the top of post.md:
---
platforms: [bluesky]
---
This post will only appear on Bluesky.
---
platforms: []
---
This post is a draft — it will never be published.
Installation
pip install mdsend
Or from source:
git clone https://github.com/algodesigner/mdsend
cd mdsend
pip install -e .
Setup
Create a .env file with your API credentials:
cp .env.example .env
Fill in the credentials for the platforms you want to use. See .env.example
for the full list of required fields.
Usage
# Preview what would be posted (default — no API calls made)
cd /path/to/your/posts
mdsend
# Actually post live
mdsend --publish
# Post to specific platforms
mdsend --platforms bluesky
mdsend --platforms bluesky --publish
Creating a new post
# Create a post directory and open the editor
mdsend --new "my-post-slug"
This creates a directory like 2026-05-04_21-30_my-post-slug/ with
a post.md file containing a front matter template, then opens it in
your $EDITOR (or vi by default).
Dry run
mdsend runs in dry-run mode by default — it prints what would be posted
without calling any API. Use --publish to post live.
Multi-post threading (Bluesky)
Posts longer than 300 characters are automatically split into a threaded
thread on Bluesky. Each chunk is prefixed with 🧵 (n/N) so readers
know it's part of a multi-part post.
Idempotent publishing
Each platform gets its own sentinel file under posts/.published/.
Re-running is safe — already-published platforms are skipped.
License
BSD 3-Clause. See LICENSE.
Happy microblogging! ☕
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 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 mdsend-0.2.0.tar.gz.
File metadata
- Download URL: mdsend-0.2.0.tar.gz
- Upload date:
- Size: 13.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
11ff1f5a9d927f5be3227e7c7e6cf9962c801024ee693b0ef7521919d335ff0f
|
|
| MD5 |
9261769bc01b61693ed909aa555b9104
|
|
| BLAKE2b-256 |
b5dded999a621b9c5b942456ff782f0b998b703d5951afe97a06807e1dd73169
|
File details
Details for the file mdsend-0.2.0-py3-none-any.whl.
File metadata
- Download URL: mdsend-0.2.0-py3-none-any.whl
- Upload date:
- Size: 13.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9347c23b55b954af54b2836f04d0f10170d1ca9abb84a24475a72753ac77b43e
|
|
| MD5 |
d3c9c9c6e2db08e5815b1243ac60516a
|
|
| BLAKE2b-256 |
8e971563ae7f0219f9df0c9c2790a23bd387f6b61c11b47e9a1a628e5484e42d
|