Private crossposting dashboard for your own social accounts.
Compose once, attach media, publish now, or schedule posts from a local/self-hosted server.
Supported Channels · Run Locally · Provider Setup · Security
Crossposter is a small local/self-hosted publishing dashboard. It is built for personal use: no database, no queue service, and no multi-user product layer. Posts can be sent immediately with Publish now or saved to the local scheduler for a later time.
This is intentionally not a Postiz-style stack. Full scheduling products usually need services such as Postgres, Redis, queues, workers, and separate background jobs. Crossposter keeps the surface area narrow so it can run on your Mac, a small VPS, Render, or a simple Node host. Scheduled posts publish only while that server process is running.
Crossposter includes a mix of official APIs and local unofficial integrations. X / Twitter, Instagram, YouTube, Pinterest, Peerlist, and Hacker News may use cookies, local sessions, private APIs, third-party tools, or normal web submit flows.
Use Crossposter only with accounts, pages, boards, channels, and profiles you own or are authorized to manage. You are responsible for each platform's API terms, automation rules, rate limits, account policies, and content policies. Unofficial integrations can break when platforms change and may trigger login challenges, rate limits, failed posts, or account restrictions.
| Channel | Current support |
|---|---|
| X / Twitter | Unofficial local posting through bird, with text, images, GIFs, and video |
| Personal profile posts and approved Page posts, with optional images or MP4 video | |
| Bluesky | Text posts and local image media |
| Mastodon | Text posts and local media |
Unofficial local media publishing through instagrapi session files |
|
| YouTube | Unofficial local video uploads through YouTube.js/InnerTube cookies |
| Dev.to | Markdown articles |
Unofficial local Pin uploads through py3-pinterest session folders |
|
| Peerlist | Unofficial local Scroll posting through Peerlist cookies and API requests |
| Hacker News | Personal link/text submission through HN's normal form flow |
| Nostr | Kind-1 text notes published to configured relays |
| Dribbble | Official OAuth shot uploads through the Dribbble API |
- Dashboard composer for title, body text, channel selection, and media upload
- Multi-profile configuration per provider, with active profile selection
- Inline Schedule draft control for local timed posting
- Scheduler calendar for scanning queued posts by month and reviewing posts by day
- Scheduler page controls for editing timing and discarding queued or failed posts
- Per-platform profile configuration from the UI
- Local config saved to
poster.config.local.json - Local publish history
- Local media upload storage in
.poster-uploads - Image conversion/compression with quality and target size controls
- Dribbble crop tool for 400x300 or 800x600 shot images
- Video conversion/compression to MP4 for supported channels
- Platform preflight warnings before publishing
- Per-platform title, post, and media limits shown only when a selected draft exceeds the limit
- Light/dark/system theme controls
- macOS auto-start service for
http://localhost:2004
Use Schedule draft next to Publish now on the Dashboard. It opens a
local date/time popup next to the button you clicked, then saves the post to
poster.config.local.json.
Manage the queue from:
http://localhost:2004/scheduled
The Scheduler page lets you:
- view the current month calendar by default
- click a calendar date to review posts scheduled for that day
- edit the scheduled timing
- discard queued or failed posts
- review target channels, media, and last publish errors
When a scheduled post publishes successfully, it is added to local publish history and removed from the Scheduler queue.
The scheduler is local/self-hosted. The Crossposter server must be running at the scheduled time:
npx @apoorvdarshan/crossposter@latestpings the scheduler every 30 seconds- the macOS auto-start service keeps
http://localhost:2004alive after login - on Render or a VPS, keep the Node service running with persistent disk
If the server is offline when a post is due, it will publish the next time the server starts and the scheduler tick runs.
Use the npx path when you want Crossposter running on localhost without cloning the repo or installing a global binary:
npx @apoorvdarshan/crossposter@latestThe launcher opens or prints the local URL:
http://localhost:2004
Run the command from the folder where you want Crossposter data to live. The local config, uploads, schedule, and history stay in that folder.
mkdir -p ~/Crossposter
cd ~/Crossposter
npx @apoorvdarshan/crossposter@latestIf you already have poster.config.local.json or .poster-uploads, run npx
from that same folder so the app uses your existing data.
For a persistent command:
npm install -g @apoorvdarshan/crossposter
crossposterFor development from the Git repo:
npm install
npm run dev:localCrossposter checks npm on startup by default. You can also update manually from Settings > Version & Updates, or run:
npx @apoorvdarshan/crossposter@latestTurn off startup update checks from Settings by setting Auto-update on launch to off.
Set POSTER_LOCAL_PORT in Settings or poster.config.local.json, then restart
the local service.
POSTER_LOCAL_PORT=2080 npx @apoorvdarshan/crossposter@latestInstall the launchd service:
crossposter install-serviceYou can also control it from:
Settings > Local Settings > Auto-start
Turn on Always restart localhost and macOS will keep
http://localhost:2004 available after login/restart.
Use a custom port once:
crossposter install-service --port 2080Remove the service:
crossposter uninstall-serviceThe app reads configuration from:
poster.config.local.json- environment variables
- defaults in the code
poster.config.local.json is gitignored. It is the preferred place for local
tokens and profile settings because it is managed by the Settings UI.
For public/self-hosted deployments, set:
POSTER_REQUIRE_ADMIN_PASSWORD=true
POSTER_ADMIN_PASSWORD=strong-password-here
X publishing is unofficial local posting through
@steipete/bird. bird uses browser
cookies from an account you are already signed into.
Required field:
X_BIRD_COMMAND
Optional fields:
X_BIRD_COOKIE_SOURCE
X_BIRD_CHROME_PROFILE
X_BIRD_FIREFOX_PROFILE
X_BIRD_TIMEOUT_MS
X_PREMIUM_LONG_POSTS
Set X_PREMIUM_LONG_POSTS=true only for Premium accounts. Crossposter uses
Bird's 280 character tweet limit for X text. The Premium toggle is only used
for larger X video uploads.
Media limits:
- photos: 5 MB
- GIFs: 15 MB
- video: 512 MB, or 16 GB when
X_PREMIUM_LONG_POSTS=true
Use this only for accounts you control. X can still challenge, limit, or lock accounts for suspicious or high-volume automation.
LinkedIn can be connected from the local Settings page after creating a LinkedIn developer app.
Add this callback URL in the LinkedIn app Auth tab:
http://localhost:2004/api/auth/linkedin/callback
For personal profile posting:
- Enable Share on LinkedIn.
- Enable Sign In with LinkedIn using OpenID Connect.
- Use scopes:
openid profile w_member_social
- Add the LinkedIn client ID and secret in Crossposter.
- Click Connect LinkedIn from Settings.
The local callback saves LINKEDIN_ACCESS_TOKEN and a personal
LINKEDIN_AUTHOR_URN automatically.
For LinkedIn Page posting:
- Create or choose the LinkedIn Page that owns the developer app.
- Make sure the signed-in member is an admin or content admin for that Page.
- Make sure the LinkedIn app has access to
w_organization_social. - Use scopes:
openid profile w_member_social w_organization_social
- Click Connect LinkedIn.
- Replace
LINKEDIN_AUTHOR_URNwith the Page author:
urn:li:organization:YOUR_PAGE_ORG_ID
Valid author examples:
urn:li:person:YOUR_PERSON_ID
urn:li:organization:YOUR_PAGE_ORG_ID
LINKEDIN_VERSION defaults to 202605.
LinkedIn local media upload supports JPG, PNG, and GIF images, plus MP4 videos between 75 KB and 500 MB. Unsupported local media is rejected before publishing.
Create a Bluesky app password. Do not use your main account password.
Required fields:
BLUESKY_IDENTIFIER
BLUESKY_APP_PASSWORD
BLUESKY_IDENTIFIER should be your handle without @, for example:
name.bsky.social
Create an application/access token from your Mastodon instance settings.
Required fields:
MASTODON_INSTANCE
MASTODON_ACCESS_TOKEN
Example instance:
https://mastodon.social
Mastodon post text is limited to 500 characters.
Instagram publishing is unofficial local posting through instagrapi.
Crossposter stores one session JSON per Instagram profile.
Required fields:
INSTAGRAM_USERNAME
INSTAGRAM_PASSWORD
INSTAGRAM_SESSION_FILE
Optional fields:
INSTAGRAM_2FA_CODE
INSTAGRAM_PYTHON_COMMAND
INSTAGRAM_TIMEOUT_MS
Install Python dependencies:
crossposter install-instagram-deps
# or, from a Git clone:
./scripts/install-instagram-deps.shOn first login or after a challenge, Instagram may require a current 2FA code or web/app verification. After the session file is saved, later publishes reuse that session until Instagram invalidates or challenges it.
Supported media:
- image: JPG, PNG, or WebP up to 8 MB
- video: MP4 or MOV up to 300 MB
YouTube publishing is unofficial local upload through YouTube.js and InnerTube. Crossposter can read cookies from a signed-in Chrome profile at publish time.
Title becomes the YouTube video title. Post text becomes the description.
Required field:
YOUTUBE_COOKIE_SOURCE
Optional fields:
YOUTUBE_CHROME_PROFILE
YOUTUBE_COOKIE
YOUTUBE_PRIVACY
YOUTUBE_TIMEOUT_MS
YOUTUBE_PRIVACY defaults to PUBLIC. Common video formats are accepted up to
256 GB or 12 hours.
Create an API key from Dev.to account settings.
Required field:
DEVTO_API_KEY
Dev.to publishing expects a title and Markdown body text.
Pinterest publishing is unofficial local posting through py3-pinterest.
Crossposter stores one session folder per Pinterest profile.
Required fields:
PINTEREST_EMAIL
PINTEREST_PASSWORD
PINTEREST_USERNAME
PINTEREST_BOARD_ID
PINTEREST_CRED_ROOT
Optional fields:
PINTEREST_SECTION_ID
PINTEREST_ALT_TEXT
PINTEREST_PYTHON_COMMAND
PINTEREST_TIMEOUT_MS
PINTEREST_HEADLESS
Install Python dependencies:
crossposter install-pinterest-deps
# or, from a Git clone:
./scripts/install-pinterest-deps.shPinterest requires a board ID because every Pin belongs to a board. Title is limited to 100 characters and description/post text is limited to 800 characters.
Supported media:
- image: JPG, PNG, GIF, or WebP up to 20 MB
- video: MP4 or MOV up to 100 MB
Peerlist publishing is unofficial local Scroll posting through Peerlist cookies and API requests. Crossposter reads cookies from your signed-in Chrome profile.
Required field:
PEERLIST_CHROME_PROFILE
Optional fields:
PEERLIST_CONTEXT
PEERLIST_USERNAME
PEERLIST_TIMEOUT_MS
Peerlist can publish post text, media-only posts, or post text with optional title and image. Local media supports JPG, PNG, WebP, or GIF up to 15 MB.
Hacker News has no official write/submit API. Crossposter uses unofficial personal automation through Hacker News' normal login and submit form flow.
Required fields:
HACKERNEWS_USERNAME
HACKERNEWS_PASSWORD
Optional field:
HACKERNEWS_COOKIE
How publishing works:
- Title is required.
- Link is optional. If set, it is submitted as Hacker News'
urlfield. - Post text is optional for Hacker News. If set, it is submitted as Hacker News'
textfield. - Leave Link empty to submit a discussion/text post.
- For non-Hacker News channels, Crossposter still requires Post text.
- Local media is ignored.
- A saved browser cookie can be used before password login.
Use this only for your own Hacker News account and normal personal submissions. Do not use it for spam, vote/comment solicitation, or bulk promotional posting. If Hacker News requires browser validation or CAPTCHA for the login, Crossposter will fail and you must submit manually.
Nostr publishes signed kind-1 text notes directly to relay WebSocket URLs.
Required fields:
NOSTR_PRIVATE_KEY
NOSTR_RELAYS
NOSTR_PRIVATE_KEY can be an nsec... key or a 64-character hex private key.
Use a dedicated Nostr key if you do not want Crossposter to sign as your main
identity.
NOSTR_RELAYS is a comma or newline separated list of relays:
wss://relay.example.com,wss://another-relay.example
Local media is ignored for Nostr. Paste public image/video links into the post body if you want Nostr clients to render media previews.
Dribbble publishing uses the official Dribbble API. Create a Dribbble API app and connect the profile from Settings.
Callback URL:
http://localhost:2004/settings/socials/dribbble/callback
Required after OAuth:
DRIBBBLE_ACCESS_TOKEN
Setup fields:
DRIBBBLE_CLIENT_ID
DRIBBBLE_CLIENT_SECRET
DRIBBBLE_OAUTH_SCOPES
Optional fields:
DRIBBBLE_TAGS
DRIBBBLE_TEAM_ID
DRIBBBLE_LOW_PROFILE
Dribbble requires a title and a local JPG, PNG, or GIF shot image that is exactly 400x300 or 800x600 and no larger than 8 MB. Crossposter can crop non-GIF images before publishing.
The composer can convert and compress media before publishing:
- images are converted to JPG output with quality, target size, and estimated size
- videos are transcoded to MP4 with quality and target size controls
- platform warnings offer conversion only when conversion can fix a selected channel's media problem
- Dribbble image warnings can open a cropper that outputs a valid 800x600 JPG
The web/ folder contains a standalone static docs website with overview,
quickstart, provider setup, limits, privacy, and terms sections. It is intended
for a domain such as:
crossposter.apoorvdarshan.com
Live site:
https://crossposter.apoorvdarshan.com
Files:
web/index.html
web/assets/logo-crossposter.png
Privacy and terms are published inside web/index.html so the static website is
the canonical policy page.
vercelSet environment variables in Vercel Project Settings. At minimum:
POSTER_ADMIN_PASSWORD
POSTER_REQUIRE_ADMIN_PASSWORD=true
Then add provider credentials only for the channels you want to use.
Render or a self-hosted Node server can run the same app. Persistent disk is recommended if you want uploaded media and local publish history to survive restarts.
Crossposter is private by convention, not a full multi-user auth system.
For local-only use:
POSTER_REQUIRE_ADMIN_PASSWORD=false
Before exposing it publicly:
- set
POSTER_REQUIRE_ADMIN_PASSWORD=true - use a strong
POSTER_ADMIN_PASSWORD - keep
poster.config.local.jsonprivate - keep
.instagram-sessions,.pinterest-sessions, and.poster-uploadsprivate - never commit API keys, access tokens, refresh tokens, app secrets, browser cookies, session files, or platform passwords
- only connect accounts, pages, and profiles you own or are authorized to manage
- Email: apoorvdarshan@gmail.com
- Email: ad13dtu@gmail.com
- X: https://x.com/apoorvdarshan
- Product Hunt: https://www.producthunt.com/products/crossposter-2
- Report an issue: https://github.com/apoorvdarshan/crossposter/issues/new
- Request a feature: https://github.com/apoorvdarshan/crossposter/issues/new
- View open issues: https://github.com/apoorvdarshan/crossposter/issues
Do not post API keys, tokens, app secrets, or private account details in public issues.
MIT
