Social Media MCP Server

# 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.