A command-line tool for posting tweets to Twitter/X from the command line.
This repo uses the uv package manager to manage dependencies. If you don't already have uv installed, you can install it with the following curl command:
curl -LsSf https://astral.sh/uv/install.sh | shTo verify install, use:
uv --versionConsult the uv installation docs for more detailed instructions and troubleshooting.
You can install the CLI tool globally using uv:
uv tool install -U birdappAfter installation, you can use:
birdapp auth config
birdapp tweet --text "Hello world!"There are two login flows available through birdapp: OAuth1 and OAuth2. Both require creating an X "app" in the X Developer Dashboard.
Important: The X API no longer has a free tier. All API usage is billed on a usage basis. You will need to enable billing in your X Developer Dashboard and add funds to your account before you can make API calls.
When creating your app, X requires a Website URL and a Callback URI:
- Website URL: You can enter any valid URL here (e.g.,
https://example.com). - Callback URI: Set this to
http://127.0.0.1:8080/callback. Birdapp starts a local server at this address to capture the OAuth2 callback automatically.
Once you've created your app, click the "Keys and Tokens" button for the app to generate authentication credentials.
With OAuth1, you generate a "Consumer Key" and "Secret Key" along with an "Access Token" and "Access Token Secret" that provide API access to a single X account. Store these in your birdapp config, and they work forever.
With OAuth2, you generate a "Client ID" and "Client Secret" for your X app, store these app secrets in your birdapp config, and login to your X accounts via a two-step flow that involves authorizing the app per-account in your browser.
In general, OAuth1 is simpler for tweeting from a single account, while OAuth2 is better for tweeting from multiple accounts. (With OAuth1 you would need a separate developer account for each account you want to use.)
OAuth2 is also generally more secure, because it doesn't involve storing secrets that could give an attacker permanent and sweeping access to your account.
Run the configuration command to set up your credentials: birdapp auth config
- Choose OAuth1 or OAuth2 based on your app registration and security posture.
- OAuth1: run
birdapp auth config --oauth1per profile. - OAuth2: run
birdapp auth config --oauth2once, thenbirdapp auth loginper account.- Optional:
birdapp auth whoamito verify the token after login.
- Optional:
This will prompt you for your Twitter API credentials and store them securely in
~/.config/birdapp/config.json.
To view your current configuration status (without showing secrets):
birdapp auth config --showBirdapp stores credentials by username profile. Each profile is keyed by the X username
(without @).
How profiles are created:
- OAuth1: created when you run
birdapp auth config --oauth1and enter a username. - OAuth2: created when you run
birdapp auth login(the username comes from the login).
How profiles are selected:
- Set the active profile with
birdapp profile use <username>. - Use
--profile <username>to override the active profile for a single command. - To list available profiles, run
birdapp profile list.
To post a tweet:
birdapp tweet --text "Your tweet content here"To post a tweet with media:
birdapp tweet --text "Check out this image!" --media /path/to/image.jpgTo post a media-only tweet (no text):
birdapp tweet --media /path/to/image.jpgTo reply to a tweet using its ID:
birdapp tweet --text "Great point!" --reply-to 1234567890To reply to a tweet using its URL:
birdapp tweet --text "I agree!" --reply-to "https://x.com/user/status/1234567890"You can also include media in replies:
birdapp tweet --text "Here's my response" --media /path/to/image.jpg --reply-to 1234567890To retrieve tweets by ID (up to 100 at a time):
birdapp get 1234567890
birdapp get 1234567890 9876543210 --format detailed
birdapp get 1234567890 --jsonTo look up users by username or ID (up to 100 at a time):
birdapp user elonmusk
birdapp user @nasa @spacex
birdapp user 44196397 --by-id
birdapp user elonmusk --format detailed --fields public_metrics created_atbirdapp supports bulk import of your tweets from Twitter/X’s “Download your data” export ZIP:
birdapp import-archive --path /path/to/twitter-archive.zipTo get the ZIP from Twitter/X:
- Request the export: go to Settings → Your account → Download an archive of your data (wording may vary) and request the archive.
- Wait for processing: it can take ~24 hours.
- Download: you’ll get an email with a download link once it’s ready.
If you have shared your tweets with the public via the Twitter Community Archive, you can download them from the archive and import them into a SQLite database:
birdapp import-archive --username yourusernameIf you've already downloaded archive.json, you can import it from a local file. The importer auto-detects the file type and safely parses the JSON payloads:
birdapp import-archive --path /path/to/archive.jsonYou can import multiple archives into the same database.
birdapp import-archive --username alice
birdapp import-archive --username bobKeyword search uses SQLite FTS5 for full-text search across imported tweets:
birdapp search "machine learning" --limit 20
birdapp search "climate policy" --author @alice
birdapp search "startup" --since 2024-01-01 --until 2024-06-30
birdapp search "distributed systems" --jsonSemantic search is opt-in and uses embeddings. It requires OPENAI_API_KEY and the sqlite-vec extension. You can optionally override the model with BIRDAPP_EMBEDDING_MODEL.
To configure these durably:
birdapp embed config --api-key sk-... --model text-embedding-3-small
birdapp embed config --showEnvironment variables still take precedence if set.
Generate embeddings for stored tweets:
birdapp embed
birdapp embed --db sqlite:////path/to/birdapp.db --model text-embedding-3-smallThen run semantic search:
birdapp search "productivity systems" --semantic --limit 10
birdapp search "startup hiring" --semantic --author @aliceYou can also embed immediately after import:
birdapp import-archive --username yourusername --embedTo see all available commands:
birdapp --helpTo see help for a specific command:
birdapp tweet --help
birdapp auth --help
birdapp get --help
birdapp user --helpPull requests are welcome! Please open an issue first to discuss any changes you want to make.
To run the tests, first capture test fixtures:
uv run tests/capture_oauth2_fixtures.pyThen run the tests:
uv run pytestTo lint and type check:
uv run ruff check --fix
uv run ty checkCommit messages must follow the Conventional Commits specification.