Social Media MCP Server
by tayler-id
- social-media-mcp
- documentation
# Post Deletion Functionality
## Overview
This document outlines the implementation plan for adding post deletion functionality to the Social Media MCP Server. Currently, the system supports posting content to various social media platforms but lacks the ability to delete posts.
## Implementation Plan
### 1. Update Mastodon Client
Add a `deleteStatus` method to the `MastodonClient` class in `src/platforms/mastodon/client.ts`:
```typescript
/**
* Delete a status from Mastodon
* @param statusId The ID of the status to delete
* @returns A result object indicating success or failure
*/
async deleteStatus(statusId: string): Promise<{ success: boolean; error?: string }> {
logger.info('Deleting status', { statusId });
try {
// Wait for authentication to complete before proceeding
const isAuthenticated = await this.waitForAuth();
logger.info('Authentication status before deleting', { isAuthenticated });
// Use rate limit manager to handle API rate limits
const result = await rateLimitManager.executeRequest({
api: 'mastodon',
endpoint: 'deleteStatus',
method: 'DELETE',
priority: 'high',
retryCount: 0,
maxRetries: config.rateLimit.maxRetries,
execute: async () => {
try {
// Check if authenticated
if (!isAuthenticated) {
throw new Error('Not authenticated with Mastodon API');
}
if (this.debug) {
logger.info('Mastodon API Debug: About to delete status', {
statusId
});
}
// Delete status
await this.client.v1.statuses.$select(statusId).delete();
if (this.debug) {
logger.info('Mastodon API Debug: Status deleted successfully');
}
return { success: true };
} catch (apiError) {
logger.error('Error deleting status from Mastodon API', {
error: apiError instanceof Error ? apiError.message : String(apiError)
});
// Fall back to mock implementation for testing
logger.info('Falling back to mock implementation for deleting status');
// Check if this is a mock status ID
const isMock = statusId.startsWith('mock-');
if (isMock) {
if (this.debug) {
logger.info('Mastodon API Debug: Mock status deleted successfully');
}
return { success: true };
} else {
return {
success: false,
error: apiError instanceof Error ? apiError.message : String(apiError)
};
}
}
}
});
logger.info('Status deletion result', { success: result.success });
return result;
} catch (error) {
logger.error('Error deleting status', {
statusId,
error: error instanceof Error ? error.message : String(error)
});
return {
success: false,
error: error instanceof Error ? error.message : String(error)
};
}
}
```
### 2. Update LinkedIn Client
Add a `deleteShare` method to the `LinkedInClient` class in `src/platforms/linkedin/client.ts`:
```typescript
/**
* Delete a share from LinkedIn
* @param shareId The ID of the share to delete
* @returns A result object indicating success or failure
*/
async deleteShare(shareId: string): Promise<{ success: boolean; error?: string }> {
logger.info('Deleting share', { shareId });
try {
// Use rate limit manager to handle API rate limits
const result = await rateLimitManager.executeRequest({
api: 'linkedin',
endpoint: 'deleteShare',
method: 'DELETE',
priority: 'high',
retryCount: 0,
maxRetries: config.rateLimit.maxRetries,
execute: async () => {
try {
// LinkedIn API doesn't support deleting posts through the v2 API
// This would require the Marketing Developer Platform access
throw new Error('LinkedIn API does not support deleting posts through the v2 API');
} catch (apiError) {
logger.error('Error deleting share from LinkedIn API', {
error: apiError instanceof Error ? apiError.message : String(apiError)
});
// Fall back to mock implementation for testing
logger.info('Falling back to mock implementation for deleting share');
// Check if this is a mock share ID
const isMock = shareId.includes('mock-');
if (isMock) {
if (config.linkedin.debug) {
logger.info('LinkedIn API Debug: Mock share deleted successfully');
}
return { success: true };
} else {
return {
success: false,
error: 'LinkedIn API does not support deleting posts through the v2 API'
};
}
}
}
});
logger.info('Share deletion result', { success: result.success });
return result;
} catch (error) {
logger.error('Error deleting share', {
shareId,
error: error instanceof Error ? error.message : String(error)
});
return {
success: false,
error: error instanceof Error ? error.message : String(error)
};
}
}
```
### 3. Create Delete Script
Create a new script `delete-post.js` in the root directory:
```javascript
import { createComponentLogger } from './build/utils/logger.js';
import linkedinClient from './build/platforms/linkedin/client.js';
import mastodonClient from './build/platforms/mastodon/client.js';
import { SocialPlatform } from './build/types/index.js';
import historyManager from './build/history/manager.js';
const logger = createComponentLogger('DeletePost');
/**
* Delete a post from a social media platform
* @param {string} postId - The ID of the post to delete
* @param {string} platform - The platform to delete from (mastodon, linkedin, twitter)
*/
async function deletePost(postId, platform) {
logger.info('Deleting post', { postId, platform });
try {
let result;
switch (platform.toLowerCase()) {
case 'mastodon':
result = await mastodonClient.deleteStatus(postId);
break;
case 'linkedin':
result = await linkedinClient.deleteShare(postId);
break;
case 'twitter':
// Twitter deletion not yet implemented
result = { success: false, error: 'Twitter deletion not yet implemented' };
break;
default:
result = { success: false, error: `Unknown platform: ${platform}` };
}
if (result.success) {
console.log(`Successfully deleted post ${postId} from ${platform}`);
// Update history if needed
try {
const updated = historyManager.updatePostStatus(postId, platform, 'deleted');
if (updated) {
console.log('History updated successfully');
}
} catch (historyError) {
console.error('Error updating history:', historyError.message);
}
} else {
console.error(`Failed to delete post ${postId} from ${platform}: ${result.error}`);
}
return result;
} catch (error) {
logger.error('Error in delete post function', {
error: error instanceof Error ? error.message : String(error)
});
console.error('Error:', error instanceof Error ? error.message : String(error));
return {
success: false,
error: error instanceof Error ? error.message : String(error)
};
}
}
// Check if this script is being run directly
if (process.argv[1].includes('delete-post.js')) {
// Get command line arguments
const args = process.argv.slice(2);
if (args.length < 2) {
console.error('Usage: node delete-post.js <postId> <platform>');
process.exit(1);
}
const postId = args[0];
const platform = args[1];
// Run the delete function
deletePost(postId, platform)
.then(result => {
process.exit(result.success ? 0 : 1);
})
.catch(error => {
console.error('Unhandled error:', error);
process.exit(1);
});
}
// Export for use in other modules
export default deletePost;
```
### 4. Update History Manager
Add a `updatePostStatus` method to the `HistoryManager` class in `src/history/manager.ts`:
```typescript
/**
* Update the status of a post in the history
* @param postId The ID of the post to update
* @param platform The platform the post is on
* @param status The new status of the post (e.g., 'deleted')
* @returns True if the post was found and updated, false otherwise
*/
updatePostStatus(postId: string, platform: string, status: string): boolean {
logger.info('Updating post status in history', { postId, platform, status });
try {
// Load history
const history = this.loadHistory();
// Find the post in history
let found = false;
for (const entry of history) {
// Check if this entry has the post
const platformEnum = this.getPlatformEnum(platform);
if (platformEnum && entry.platforms.includes(platformEnum)) {
const content = entry.content[platformEnum];
if (content && content.postId === postId) {
// Update the status
content.status = status;
found = true;
break;
}
}
}
if (found) {
// Save the updated history
this.saveHistory(history);
logger.info('Post status updated in history', { postId, platform, status });
return true;
} else {
logger.warn('Post not found in history', { postId, platform });
return false;
}
} catch (error) {
logger.error('Error updating post status in history', {
postId,
platform,
status,
error: error instanceof Error ? error.message : String(error)
});
return false;
}
}
/**
* Convert a platform string to the corresponding enum value
* @param platform The platform string
* @returns The platform enum value, or undefined if not found
*/
private getPlatformEnum(platform: string): SocialPlatform | undefined {
switch (platform.toLowerCase()) {
case 'twitter':
return SocialPlatform.TWITTER;
case 'mastodon':
return SocialPlatform.MASTODON;
case 'linkedin':
return SocialPlatform.LINKEDIN;
default:
return undefined;
}
}
```
## Usage
Once implemented, you can delete posts using the following command:
```bash
node delete-post.js <postId> <platform>
```
For example:
```bash
node delete-post.js 114118098200668360 mastodon
```
## Limitations
- LinkedIn API does not support deleting posts through the v2 API. This would require the Marketing Developer Platform access.
- Twitter deletion is not yet implemented.
- The history update functionality assumes that the history manager has been updated to include post IDs in the history entries.
## Future Improvements
1. Add a UI for managing and deleting posts
2. Implement batch deletion for multiple posts
3. Add scheduling for automatic post deletion after a certain time
4. Implement Twitter post deletion when the Twitter API client is fully implemented
5. Add support for deleting posts with media attachments