Social Media MCP Server
by tayler-id
- social-media-mcp
- documentation
# Mastodon Integration Documentation
## Overview
The Mastodon integration allows the Social Media MCP Server to post content to Mastodon accounts. This document outlines how the integration works, the authentication process, and how to troubleshoot common issues.
## Authentication
The Mastodon client uses token-based authentication with the following flow:
1. The client is initialized with an access token from the configuration
2. The client verifies the token by calling the `verifyCredentials` API
3. If verification succeeds, the client is marked as authenticated
4. If verification fails, the client falls back to mock implementation
## Key Components
### MastodonClient Class
Located in `src/platforms/mastodon/client.ts`, this class handles all interactions with the Mastodon API:
- **Authentication**: Manages token verification and authentication state
- **Posting**: Handles posting content to Mastodon
- **Trending Tags**: Retrieves trending hashtags from Mastodon
- **Engagement Metrics**: Gets engagement data for posts
### Authentication Flow
The client uses an asynchronous authentication process:
```typescript
constructor() {
// Initialize the authentication promise
this.authPromise = this.initializeClient();
logger.info('Mastodon client initializing...', {
instance: config.mastodon.credentials.instance || 'https://mastodon.social',
debug: this.debug
});
}
```
The `initializeClient` method handles the actual authentication:
```typescript
private async initializeClient(): Promise<boolean> {
try {
// Initialize with user context
this.client = createRestAPIClient({
url: config.mastodon.credentials.instance || 'https://mastodon.social',
accessToken: config.mastodon.credentials.accessToken,
});
// Verify authentication if access token is provided
if (config.mastodon.credentials.accessToken) {
try {
const verified = await this.verifyCredentials();
this.isAuthenticated = verified;
if (verified) {
logger.info('Mastodon client authenticated successfully');
} else {
logger.warn('Mastodon client authentication failed, will use mock implementation');
}
} catch (error) {
logger.error('Error verifying Mastodon credentials', {
error: error instanceof Error ? error.message : String(error)
});
this.isAuthenticated = false;
}
} else {
logger.warn('No Mastodon access token provided, will use mock implementation');
this.isAuthenticated = false;
}
logger.info('Mastodon client initialized', {
instance: config.mastodon.credentials.instance || 'https://mastodon.social',
debug: this.debug,
authenticated: this.isAuthenticated
});
this.authComplete = true;
return this.isAuthenticated;
} catch (error) {
logger.error('Error initializing Mastodon client', {
error: error instanceof Error ? error.message : String(error)
});
this.isAuthenticated = false;
this.authComplete = true;
return false;
}
}
```
Before any API operation, the client waits for authentication to complete:
```typescript
public async waitForAuth(): Promise<boolean> {
if (this.authComplete) {
return this.isAuthenticated;
}
logger.info('Waiting for Mastodon authentication to complete...');
return this.authPromise;
}
```
## Posting to Mastodon
The `postStatus` method handles posting content to Mastodon:
```typescript
async postStatus(content: Content): Promise<PostResult> {
logger.info('Posting status', { content: content.text.substring(0, 30) + '...' });
try {
// Wait for authentication to complete before proceeding
const isAuthenticated = await this.waitForAuth();
logger.info('Authentication status before posting', { isAuthenticated });
// Use rate limit manager to handle API rate limits
const result = await rateLimitManager.executeRequest({
api: 'mastodon',
endpoint: 'postStatus',
method: 'POST',
priority: 'high',
retryCount: 0,
maxRetries: config.rateLimit.maxRetries,
execute: async () => {
try {
// Check if authenticated
if (!isAuthenticated) {
throw new Error('Not authenticated with Mastodon API');
}
// Create status
const status = await this.client.v1.statuses.create({
status: content.text,
visibility: 'public',
});
return status;
} catch (apiError) {
// Fall back to mock implementation for testing
logger.info('Falling back to mock implementation for posting status');
// Generate a mock status response
const mockStatus = {
id: `mock-${Date.now()}`,
url: `https://mastodon.social/@mock/mock-${Date.now()}`,
content: content.text,
visibility: 'public',
createdAt: new Date().toISOString(),
account: {
id: 'mock-account',
username: 'mock',
displayName: 'Mock Account'
}
};
return mockStatus;
}
}
});
// Return post result
return {
platform: SocialPlatform.MASTODON,
success: true,
postId: result.id,
url: result.url,
timestamp: new Date(),
isMock: typeof result.id === 'string' && result.id.startsWith('mock-')
};
} catch (error) {
return {
platform: SocialPlatform.MASTODON,
success: false,
error: error instanceof Error ? error.message : String(error),
timestamp: new Date(),
};
}
}
```
## Content Limitations
Mastodon has a character limit of 500 characters per post. If your content exceeds this limit, the API will return a validation error. Make sure your content fits within this limit.
## Configuration
The Mastodon client is configured in `src/config/index.ts`:
```typescript
mastodon: {
credentials: {
instance: process.env.MASTODON_INSTANCE || 'https://mastodon.social',
accessToken: process.env.MASTODON_ACCESS_TOKEN || '',
},
debug: true,
},
```
## Troubleshooting
### Authentication Issues
If you're experiencing authentication issues:
1. Check that your access token is valid
2. Verify that the instance URL is correct
3. Look for authentication errors in the logs
### Content Issues
If your posts are failing:
1. Check the character count (must be under 500 characters)
2. Ensure there are no formatting issues
3. Check for any API-specific errors in the logs
### Mock Mode
The client will fall back to mock mode if:
1. No access token is provided
2. Authentication fails
3. The API returns an error
In mock mode, posts will appear successful but will have a URL like `https://mastodon.social/@mock/mock-1234567890` and will be marked with `isMock: true` in the response.
## Current Status
The Mastodon integration is currently working properly with real authentication. Posts are being successfully sent to the Mastodon account.