Post tweets • Upload images • Search tweets • Reply to conversations • Look up user profiles • Fetch conversation threads • Monitor mentions • Publish smart threads • Quote tweets • Extract media • Like, retweet & bookmark
🚀 Quick Start • ✨ Features • 📖 Documentation • 🤝 Contributing
A Model Context Protocol (MCP) server that enables seamless interaction with Twitter/X through AI assistants such as Claude Desktop.
- 🐦 Post Tweets - Share your thoughts with the world
- 🖼️ Image Support - Post tweets with images (JPG, PNG, GIF, WEBP)
- 🔍 Search Tweets - Find and analyze tweets by query
- 👤 User Profile Context - Fetch comprehensive user profiles with bio, metrics, pinned tweet, and recent activity
- 💬 Thread History - Retrieve full conversation threads by tweet ID
- 🔔 Mention Monitoring - Search recent mentions of the authenticated user or custom keywords
- 🧵 Smart Threads - Auto-split long content into threaded tweet chains
- 💬 Quote Tweets - Quote an existing tweet with AI commentary
- 🎬 Media Extraction - Extract direct media URLs from tweets (images, video, GIF)
- ❤️ Engagement - Like, retweet, or bookmark tweets directly
- 🔎 Optional Xquik Search - Use Hermes Tweet/Xquik for read-only search
- 💬 Reply to Tweets - Engage in conversations
- 🔐 Secure Authentication - OAuth 1.0a authentication
- ⚡ Rate Limiting - Built-in protection against API limits
Twitter MCP makes it easy for AI assistants to interact with X (Twitter) through natural language.
- 🚀 Zero installation using NPX
- 🤖 Works with Claude Desktop and MCP-compatible clients
- 🐦 Post tweets directly from AI conversations
- 🖼️ Upload images with tweets
- 🔍 Search Twitter using multiple backends
- 🔐 Secure OAuth authentication
- ⚡ Built-in rate limiting and validation
- 🛠️ Open-source and TypeScript powered
- Content creators
- Social media managers
- AI agent developers
- Marketing teams
- Researchers and analysts
- MCP enthusiasts
- Installation
- Configuration
- Usage
- API Reference
- Development
- Troubleshooting
- Environment Variables
- License
- Node.js 18 or higher
- npm or npx
- Twitter Developer Account with API credentials
- Claude Desktop App
The easiest way to use this MCP server is through the interactive installer:
npx @muhammadsiddiq/twitter-mcpStart the setup flow with:
npx @muhammadsiddiq/twitter-mcpThe installer guides you through every question and writes the correct MCP config for the client you choose.
The first prompt asks which AI assistant environment you want to integrate with. Available options are:
- Claude Desktop App
- Cursor IDE
- OpenCode IDE
- Hermes
- Kilo
- Custom Agent
If you choose a built-in client, the installer already knows the right config location and format. If you choose Custom Agent, it asks for the config file path and the config structure to use.
Next, the installer asks for the Twitter/X API credentials it needs to work:
- Consumer Key
- Consumer Secret Key
- Access Token
- Access Token Secret
These values are required for posting, replying, and Twitter API search.
- Visit Twitter Developer Portal
- Create a new App or use an existing one
- Navigate to "Keys and Tokens"
- Generate/Copy the following:
- Consumer Key
- Consumer Secret Key
- Access Token
- Access Token Secret
Provide these values when the installer prompts for them.
The installer then asks whether you want to add optional read-only search backends for search_tweets.
If you say yes, it can collect:
- XQuik / Hermes Tweet API key
- GetXAPI API key
These are optional. If you skip them, the server still works with the Twitter OAuth credentials.
- Sign in at dashboard.xquik.com.
- Open Account > API Keys.
- Create an API key for this MCP server and copy it once.
- Store that value as
XQUIK_API_KEYin your MCP config or shell environment. - If your deployment uses the Hermes Tweet naming, set the same value as
HERMES_TWEET_API_KEYinstead. - Leave
XQUIK_BASE_URLunset unless your team runs a compatible non-default deployment. - Restart your client and call
search_tweetsto verify read-only search.
Keep the key out of Git, chat prompts, screenshots, and shared config files. The key only changes search_tweets; posting and replying still use the Twitter OAuth variables.
- Sign in at getxapi.com.
- Create an API key for this MCP server and copy it once.
- Store that value as
GETXAPI_API_KEYin your MCP config or shell environment. - Leave
GETXAPI_BASE_URLunset unless your team runs a compatible non-default deployment. - Restart your client and call
search_tweetsto verify read-only search.
After the questions are answered, the installer writes the config automatically and tells you to restart the selected client.
Can't find Connectors in Settings?
- Make sure you're using the latest version of Claude Desktop
- Try restarting the application
Path not working?
- Use the full absolute path (complete path from root)
- Avoid spaces in folder names, or use quotes around the path
- Check that the directory actually exists on your computer
Changes not taking effect?
- Make sure you completely closed Claude Desktop (check system tray/menu bar)
- Wait a few seconds before reopening
- Restart your computer if issues persist
Once configured, you can interact with Twitter through natural language commands to Claude.
Simple Tweet:
Post a tweet: "Hello World! 🌍"
Important: Make sure you have configured the filesystem MCP server as shown in Step 4.
Tweet with Image:
Post this image with caption: "Check out this amazing view!"
take image from desktop
Working with Images
Images must be accessible to the MCP server process that your client launches. Below are the common flows and what to expect:
-
CLI flow (running the server directly):
- You can pass an absolute or relative path to the image when using the CLI/server directly.
- Example:
npx @muhammadsiddiq/twitter-mcp --stdiothen provide/home/me/Pictures/photo.jpgor./images/photo.jpgwhen prompted. - If the CLI is started from a different working directory, provide an absolute path or change to the correct folder first.
-
Agent flow (Claude Desktop, Cherry Studio, etc.):
- Most desktop agents use a filesystem MCP connector to grant the agent access to files. Claude Desktop includes this; some clients (e.g., Cherry Studio) require you to enable/configure the filesystem connector.
- Configure the client's filesystem MCP (or connector) to include the folder(s) containing your images. After granting access, restart the client and then ask the agent to attach or use the image by filename or path.
-
Permission notes:
- The MCP server process (or the agent) must have read permission for the image file.
- When using desktop agents, the client will typically prompt you to grant filesystem access; follow the client's permission flow rather than changing file permissions globally.
Supported path examples:
- Windows:
C:\Users\YourName\Pictures\image.jpg - macOS:
/Users/YourName/Pictures/image.jpg - Linux:
/home/yourname/pictures/image.jpg - Relative:
./images/photo.jpg(relative to the server's working directory)
Supported formats and limits:
- JPEG/JPG, PNG, GIF, WEBP
- Max size: ~5MB for static images, ~15MB for animated GIFs (subject to client limits)
Best practices:
- Use absolute paths if you are unsure of the current working directory.
- Keep images in a dedicated folder and add that folder to your client's filesystem permissions.
- Prefer small, optimized images for faster uploads and fewer failures.
- If an agent cannot see a file, check the client's filesystem connector settings and restart the client after granting access.
If you're using an MCP client not listed here, check the client's documentation for how to grant or configure filesystem access (search for "filesystem connector", "desktop connectors", or "MCP filesystem").
Basic Search:
Search for tweets about "artificial intelligence"
Advanced Search:
Search for 50 tweets about "climate change" from the past week
Get profile context:
Look up the profile of "elonmusk"
The tool returns bio, profile metadata, follower/following metrics, pinned tweet, and the 5 most recent original tweets.
Get conversation thread:
Show me the full conversation thread for tweet 1234567890
The tool finds the conversation the tweet belongs to and returns all tweets ordered chronologically, with author IDs, text, timestamps, and reply relationships.
Check mentions and search:
What are the latest mentions of my account?
Search recent tweets about "product launch"
Without a query, returns tweets mentioning the authenticated user. With a query, searches recent tweets matching it. Each result includes author, text, timestamp, and engagement metrics.
Publish long content as a thread:
Publish this as a thread: "Part 1: Introduction to AI\n\nPart 2: Key Concepts\n\nPart 3: Applications"
Use double newlines to indicate tweet breaks. The tool splits by paragraph, then by sentence if needed (max 280 chars per tweet), and posts them as a connected reply chain.
Quote an existing tweet:
Quote tweet 1234567890 with commentary: "This is an interesting perspective!"
The tool posts your commentary as a new tweet with the target tweet embedded as a quote beneath it.
Extract media from a tweet:
Get the media URLs from tweet 1234567890
Returns direct URLs for all attached images, video variants (with bit rates), and animated GIFs.
Interact with a tweet:
Like tweet 1234567890
Retweet the post by user
Bookmark that tweet for later
Supports three actions: like, retweet, and bookmark. Returns a success confirmation.
The server provides the following tools:
Post a text-only tweet.
Parameters:
text(string) — Tweet contentreply_to_tweet_id(string, optional) — ID of a tweet to reply to
Post a tweet with an attached image.
Parameters:
text(string) — Tweet contentimage_path(string) — Local path to image (absolute or relative)reply_to_tweet_id(string, optional) — ID of a tweet to reply to
Supported Image Formats:
- JPEG/JPG
- PNG
- GIF (animated, max 15MB)
- WEBP
Fetch a comprehensive Twitter/X user profile.
Parameters:
username(string) — Twitter handle (without@)
Returns:
id,name,username,description,profile_image_urlverified,protected,location,url,created_atpublic_metrics(followers/following/tweet/listed counts)pinned_tweet(if set)recent_tweets(last 5 original tweets with engagement metrics)
Example:
// Request:
{
"username": "elonmusk"
}
// Response:
{
"status": "success",
"message": "User profile fetched successfully",
"data": {
"id": "44196397",
"name": "Elon Musk",
"username": "elonmusk",
"description": "...",
"public_metrics": {
"followers_count": 190000000,
"following_count": 743,
"tweet_count": 30000,
"listed_count": 150000
},
"pinned_tweet": {
"id": "123456789",
"text": "...",
"created_at": "2026-06-01T12:00:00.000Z"
},
"recent_tweets": [
{
"id": "987654321",
"text": "...",
"created_at": "2026-06-12T08:00:00.000Z",
"like_count": 50000,
"retweet_count": 10000,
"reply_count": 2000
}
]
}
}Retrieve the full conversation thread for a tweet.
Parameters:
tweet_id(string) — ID of the tweet in the thread
Returns:
conversation_id— the thread's conversation IDthread— array of tweets ordered oldest-first withid,text,author_id,created_at, engagement metrics, andin_reply_to_tweet_id
Example:
// Request:
{
"tweet_id": "1234567890"
}
// Response:
{
"status": "success",
"message": "Thread history fetched successfully",
"data": {
"conversation_id": "1234567890",
"thread": [
{
"id": "1234567880",
"text": "Original post...",
"author_id": "user1",
"created_at": "2026-06-12T10:00:00.000Z",
"like_count": 120,
"retweet_count": 30,
"reply_count": 5,
"in_reply_to_tweet_id": null
},
{
"id": "1234567890",
"text": "Reply to the thread...",
"author_id": "user2",
"created_at": "2026-06-12T10:05:00.000Z",
"like_count": 10,
"retweet_count": 1,
"reply_count": 0,
"in_reply_to_tweet_id": "1234567880"
}
]
}
}Monitor mentions of the authenticated user or search recent tweets by keyword.
Parameters:
query(string, optional) — Custom search query. Omitting returns mentions of your account.max_results(number, optional, default 10) — Results to return (5–100).
Returns:
tweets— array withid,text,author_id,author_username,created_at,like_count,retweet_count,reply_countmeta—result_countand optionalnext_tokenfor pagination
Example:
// Request (mentions):
{
"max_results": 10
}
// Request (custom search):
{
"query": "product launch",
"max_results": 20
}
// Response:
{
"status": "success",
"data": {
"tweets": [
{
"id": "1234567890",
"text": "@user Great post!",
"author_id": "98765",
"author_username": "follower1",
"created_at": "2026-06-12T12:00:00.000Z",
"like_count": 5,
"retweet_count": 1,
"reply_count": 0
}
],
"meta": {
"result_count": 10
}
}
}Split long content into a threaded tweet chain.
Parameters:
content(string, 1–10000 chars) — Full text to publish. Use double newlines to indicate tweet breaks.
Returns:
thread— array of posted tweets withposition,id,text,created_attotal_tweets— count of tweets in the threadfirst_tweet_url— URL to the first tweet on X/Twitter
Example:
// Request:
{
"content": "Excited to announce our new product!\n\nIt has three key features:\n\nFeature 1: Lightning fast.\n\nFeature 2: Easy to use.\n\nFeature 3: Open source.\n\nCheck it out at example.com!"
}
// Response:
{
"status": "success",
"message": "Smart thread published successfully",
"data": {
"thread": [
{ "position": 1, "id": "111", "text": "Excited to announce our new product!", "created_at": "..." },
{ "position": 2, "id": "112", "text": "It has three key features:", "created_at": "..." },
{ "position": 3, "id": "113", "text": "Feature 1: Lightning fast.", "created_at": "..." }
],
"total_tweets": 3,
"first_tweet_url": "https://x.com/i/status/111"
}
}Quote an existing tweet with AI commentary.
Parameters:
target_tweet_id(string) — ID of the tweet to quotecommentary(string, max 280) — Text to display above the quoted tweet
Returns:
id,text,author_id,created_at— quote tweet detailsquoted_tweet_id— the ID of the quoted tweettweet_url— URL to the quote tweet on X/Twitter
Example:
// Request:
{
"target_tweet_id": "1234567890",
"commentary": "This is a great take on the topic!"
}
// Response:
{
"status": "success",
"message": "Quote tweet drafted successfully",
"data": {
"id": "9876543210",
"text": "This is a great take on the topic!",
"author_id": "self",
"created_at": "2026-06-12T12:00:00.000Z",
"quoted_tweet_id": "1234567890",
"tweet_url": "https://x.com/i/status/9876543210"
}
}Extract direct media URLs from a tweet.
Parameters:
tweet_id(string) — ID of the tweet to extract media from
Returns:
tweet_id— the source tweet IDmedia— array of media items, each withtype,url,preview_image_url,width,height,duration_ms, andvariants(for videos/GIFs)media_count— total count
Example:
// Request:
{
"tweet_id": "1234567890"
}
// Response:
{
"status": "success",
"message": "Media extracted successfully",
"data": {
"tweet_id": "1234567890",
"media": [
{
"type": "photo",
"url": "https://pbs.twimg.com/media/ABC123.jpg",
"width": 1200,
"height": 800
},
{
"type": "video",
"preview_image_url": "https://pbs.twimg.com/media/DEF456.jpg",
"duration_ms": 30000,
"width": 1920,
"height": 1080,
"variants": [
{
"url": "https://video.twimg.com/video.mp4",
"content_type": "video/mp4",
"bit_rate": 2000000
}
]
}
],
"media_count": 2
}
}Like, retweet, or bookmark a tweet.
Parameters:
tweet_id(string) — ID of the tweet to interact withaction(enum:"like"|"retweet"|"bookmark") — Action to perform
Returns:
tweet_id,action,success(boolean)
Example:
// Request:
{
"tweet_id": "1234567890",
"action": "like"
}
// Response:
{
"status": "success",
"message": "Engagement action performed successfully",
"data": {
"tweet_id": "1234567890",
"action": "like",
"success": true
}
}Search for tweets matching a query.
Set XQUIK_API_KEY or HERMES_TWEET_API_KEY to route search through
Hermes Tweet/Xquik. Set GETXAPI_API_KEY to route search through GetXAPI.
Without those variables, search uses the configured Twitter API credentials.
Types:
interface SearchTweetsRequest {
query: string; // Search query string
count: number; // Number of results (10-100)
}
interface SearchResponse {
tweets: Tweet[];
meta: {
result_count: number;
next_token?: string;
};
}Example:
// Request:
{
"query": "machine learning",
"count": 25
}
// Response:
{
"status": "success",
"message": "Search completed successfully",
"data": {
"tweets": [
{
"id": "1234567891",
"text": "Exploring machine learning concepts...",
"author_id": "user123",
"created_at": "2025-11-06T12:00:00.000Z"
}
// ... more tweets
],
"meta": {
"result_count": 25,
"next_token": "abc123xyz"
}
}
}- Clone the repository:
git clone https://github.com/genaiwithms/twitter-mcp.git
cd twitter-mcp- Install dependencies:
npm install- Build the project:
npm run build- Set up environment:
Create a .env file in the project root:
API_KEY=your_api_key
API_SECRET_KEY=your_api_secret
ACCESS_TOKEN=your_access_token
ACCESS_TOKEN_SECRET=your_access_token_secret- Run locally:
Update your Claude config to use local build:
{
"mcpServers": {
"twitter": {
"command": "node",
"args": ["${absolute_path_to_project}/build/index.js"],
"envFile": ".env"
}
}
}- Development commands:
# Start the server
npm start
# Run tests
npm test
# Build for production
npm run build
# Publish to npm (maintainers only)
npm publish --access publictwitter-mcp/
├── src/
│ ├── index.ts # Main server entry point
│ ├── twitter-api.ts # Twitter API client
│ ├── types.ts # TypeScript type definitions
│ ├── formatter.ts # Response formatting
│ ├── types/ # Type declarations
│ │ └── modelcontextprotocol.d.ts
│ └── evals/
│ └── evals.ts # Test utilities
├── .github/ # GitHub Actions workflows
│ └── workflows/
│ └── ci.yml # CI pipeline
├── build/ # Compiled JavaScript (generated)
├── package.json # Project metadata and dependencies
├── tsconfig.json # TypeScript configuration
├── .gitignore # Git ignore rules
├── .env.example # Example environment variables
├── CHANGELOG.md # Version history
├── CONTRIBUTING.md # Contribution guidelines
└── README.md # Project documentation
npm run build- Compile TypeScript to JavaScriptnpm start- Run the compiled servernpm run prepublishOnly- Build before publishing
Problem: "401 Unauthorized" or authentication failed
Solutions:
- Verify Twitter API credentials in Developer Portal
- Ensure all four tokens are correct and complete
- Check app permissions (needs Read + Write)
- Try regenerating access tokens
- Verify
.envfile format if using local development
Problem: "Rate limit exceeded" or requests failing
Solutions:
- Built-in rate limiting protects against overuse
- Wait 15 minutes for limits to reset
- Check your Twitter API tier limits
- Use exponential backoff for retries
- Monitor usage in Twitter Developer Portal
Problem: Image upload fails or missing media
Solutions:
- Verify file exists and is readable
- Check size limits: 5MB (images), 15MB (GIFs)
- Ensure format is supported (JPG, PNG, GIF, WEBP)
- Use absolute file paths
- Check file permissions
- Verify image is not corrupted
Problem: Image file cannot be uploaded.
Solutions:
- Verify the file path is correct and absolute
- Check file exists and is readable
- Ensure file size is under limits (5MB for images, 15MB for GIFs)
- Verify file format is supported (JPG, PNG, GIF, WEBP)
- Check file permissions
Problem: Dependencies not installed or build not completed.
Solution:
# Remove old dependencies
rm -rf node_modules package-lock.json
# Reinstall
npm install
# Rebuild
npm run buildProblem: MCP server not connecting to Claude.
Solutions:
- Restart Claude Desktop completely
- Check config file syntax is valid JSON
- Verify file path in config matches actual location
- Check Node.js is installed:
node --version - Look for errors in Claude's logs
To see detailed logs, check:
Windows:
%APPDATA%\Claude\logs\
macOS:
~/Library/Logs/Claude/
Linux:
~/.config/Claude/logs/
Posting tools require Twitter OAuth credentials. search_tweets can use
Twitter OAuth credentials, the optional Hermes Tweet/Xquik read-only backend,
or the optional GetXAPI read-only backend.
| Variable | Description | Required |
|---|---|---|
API_KEY |
Twitter API Key | For posting and Twitter API search |
API_SECRET_KEY |
Twitter API Secret Key | For posting and Twitter API search |
ACCESS_TOKEN |
Twitter Access Token | For posting and Twitter API search |
ACCESS_TOKEN_SECRET |
Twitter Access Token Secret | For posting and Twitter API search |
XQUIK_API_KEY |
Optional Hermes Tweet/Xquik key for search_tweets |
No |
HERMES_TWEET_API_KEY |
Optional alias for XQUIK_API_KEY |
No |
XQUIK_BASE_URL |
Optional Xquik base URL, defaults to https://xquik.com |
No |
GETXAPI_API_KEY |
Optional GetXAPI key for search_tweets |
No |
GETXAPI_BASE_URL |
Optional GetXAPI base URL, defaults to https://api.getxapi.com |
No |
- Never commit credentials to version control
- Use environment variables for sensitive data
- Rotate credentials periodically
- Monitor API usage in Twitter Developer Portal
- Set up alerts for unusual activity
- Use separate credentials for development and production
- Maximum tweet length: 280 characters
- Image file size limits: 5MB (images), 15MB (GIFs)
- Rate limits apply based on your Twitter API tier
- Media must be uploaded before tweeting (handled automatically)
This project uses Jest for testing. Run tests with:
# Run all tests
npm test
# Run tests in watch mode
npm test -- --watch
# Run tests with coverage
npm test -- --coverageTest files are located in src/evals/. Example test:
Contributions are welcome! Please follow these steps:
-
Fork & Clone:
git clone https://github.com/genaiwithms/twitter-mcp.git cd twitter-mcp -
Create Branch:
git checkout -b feature/your-feature # or git checkout -b fix/your-bugfix -
Make Changes:
- Follow TypeScript practices
- Add/update tests
- Update documentation
-
Test & Build:
npm install npm run build npm test -
Commit & Push:
git add . git commit -m "feat: add amazing feature" git push origin feature/your-feature
-
Open Pull Request:
- Use clear title and description
- Reference issues if applicable
- Include test results
- Update documentation
Follow Conventional Commits:
feat:New featurefix:Bug fixdocs:Documentationtest:Testsrefactor:Code refactoringchore:Maintenance
Thanks to everyone who contributes to Twitter MCP Server ❤️
- Issues: GitHub Issues
- Documentation: MCP Documentation
- Twitter API: Twitter Developer Docs
- Model Context Protocol ecosystem
- Claude Desktop community
- twitter-api-v2 maintainers
- Xquik team
- GetXAPI team
- Open-source contributors
Special thanks to all community contributors who help improve this project.
This project is licensed under the MIT License - see the LICENSE file for details.