Skip to main content

File-based social media publisher. Write markdown, post to LinkedIn, Bluesky, and Mastodon.

Project description

mdsend — File-based social media publisher

mdsend

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
LinkedIn 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 & Mastodon)

Posts longer than the platform limit are automatically split into threaded replies. Each chunk is prefixed with 🧵 (n/N) so readers know it's part of a multi-part post. Media is attached to the first chunk only. Links are moved to the first chunk so the preview card appears at the top of the thread.

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


Download files

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

Source Distribution

mdsend-0.3.2.tar.gz (14.6 kB view details)

Uploaded Source

Built Distribution

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

mdsend-0.3.2-py3-none-any.whl (15.3 kB view details)

Uploaded Python 3

File details

Details for the file mdsend-0.3.2.tar.gz.

File metadata

  • Download URL: mdsend-0.3.2.tar.gz
  • Upload date:
  • Size: 14.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.4

File hashes

Hashes for mdsend-0.3.2.tar.gz
Algorithm Hash digest
SHA256 cfa49a140c4adedf9427bd53611f386a3d512735b7ad7fc0d0ea525d1bfa1c84
MD5 fa50f91dfb0f5b3cabb10b2db2eae9f8
BLAKE2b-256 dda5d6b67936d3b7e4be82a6ca01436f015836cc3677e3e5d64ae0d17f8abfdc

See more details on using hashes here.

File details

Details for the file mdsend-0.3.2-py3-none-any.whl.

File metadata

  • Download URL: mdsend-0.3.2-py3-none-any.whl
  • Upload date:
  • Size: 15.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.4

File hashes

Hashes for mdsend-0.3.2-py3-none-any.whl
Algorithm Hash digest
SHA256 5e9260fef7731790a9fe700a04dc1dc8660b24a3d2bcdc4604dd96beb4ddf4ad
MD5 aa74c3f7876f05f064da98c8619aed1a
BLAKE2b-256 189a7bd8c8fe9bed3bac69d96e5aa34c4302ba3686c45a4d7ffde80b8c06a684

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