Bluesky MCP Server

by brianellin
Verified
import { AtpAgent } from "@atproto/api"; export interface TextContent { type: "text"; text: string; [key: string]: unknown; } export interface ImageContent { type: "image"; data: string; mimeType: string; [key: string]: unknown; } export interface ResourceContent { type: "resource"; resource: { text: string; uri: string; mimeType?: string; [key: string]: unknown; } | { uri: string; blob: string; mimeType?: string; [key: string]: unknown; }; [key: string]: unknown; } export type McpResponseContent = Array<TextContent | ImageContent | ResourceContent>; export interface McpSuccessResponse { content: McpResponseContent; [key: string]: unknown; } export interface McpErrorResponse { isError: true; content: McpResponseContent; [key: string]: unknown; } /** * Helper function to get a human-readable name for built-in feeds */ export function getFeedNameFromId(id: string): string { const knownFeeds: Record<string, string> = { 'home': 'Home Timeline', 'following': 'Following', 'what-hot': 'What\'s Hot', 'discover': 'Discover', 'for-you': 'For You' }; return knownFeeds[id] || id; } /** * Format a post for display in the response */ export function formatPost(item: any, index: number): string { // Handle both scenarios - when passed just a post or a full item with post and reason let post: any; let reason: any = null; // Check if this is an item with post property (feed item) or a direct post if (item && item.post) { post = item.post; reason = item.reason; } else { post = item; // For backwards compatibility, also check if the post itself has a reason if (post && post.reason) { reason = post.reason; } } // Check if this is a repost and extract repost information let isRepost = false; let reposter: any = null; if (reason && reason.$type === 'app.bsky.feed.defs#reasonRepost' && reason.by) { isRepost = true; reposter = reason.by; } // For reposts, the author is in post.author const author = post.author; // Safeguard against missing author if (!author) { return `Post #${index + 1}: Error - Could not determine author of post`; } // Extract and process thread/reply context let threadInfo: string[] = []; let isReply = false; // Check if post is a reply if (post.record?.reply || post.reply) { isReply = true; const replyInfo = post.record?.reply || post.reply; if (replyInfo) { threadInfo.push('🧵 Reply in thread:'); // Add root post info if available if (replyInfo.root) { if (replyInfo.root.uri) { threadInfo.push(` Root: ${replyInfo.root.uri}`); } if (replyInfo.root.author) { const rootAuthor = replyInfo.root.author; threadInfo.push(` Root author: ${rootAuthor.displayName || rootAuthor.handle} (@${rootAuthor.handle})`); } } // Add parent post info if different from root if (replyInfo.parent && (!replyInfo.root || replyInfo.parent.uri !== replyInfo.root.uri)) { if (replyInfo.parent.uri) { threadInfo.push(` Replying to: ${replyInfo.parent.uri}`); } if (replyInfo.parent.author) { const parentAuthor = replyInfo.parent.author; threadInfo.push(` Replying to: ${parentAuthor.displayName || parentAuthor.handle} (@${parentAuthor.handle})`); } } } } // Handle cases where the reply info is at the top level of the post object if (post.reply && !isReply) { isReply = true; threadInfo.push('🧵 Reply in thread:'); // Add root info if available if (post.reply.root) { if (post.reply.root.uri) { threadInfo.push(` Thread root: ${post.reply.root.uri}`); } const rootAuthor = post.reply.root.author; if (rootAuthor) { threadInfo.push(` Thread started by: ${rootAuthor.displayName || rootAuthor.handle} (@${rootAuthor.handle})`); // Add root post content preview if (post.reply.root.record?.text) { const rootText = post.reply.root.record.text; threadInfo.push(` Original post: ${rootText.length > 80 ? rootText.substring(0, 80) + '...' : rootText}`); } } } // Add parent info if available and different from root if (post.reply.parent && (!post.reply.root || post.reply.parent.uri !== post.reply.root.uri)) { if (post.reply.parent.uri) { threadInfo.push(` Replying to: ${post.reply.parent.uri}`); } const parentAuthor = post.reply.parent.author; if (parentAuthor) { threadInfo.push(` Replying to: ${parentAuthor.displayName || parentAuthor.handle} (@${parentAuthor.handle})`); // Add parent post content preview if (post.reply.parent.record?.text) { const parentText = post.reply.parent.record.text; threadInfo.push(` Parent post: ${parentText.length > 80 ? parentText.substring(0, 80) + '...' : parentText}`); } } } } // Extract rich text elements from facets (links, mentions, hashtags) let links: string[] = []; let mentions: string[] = []; let hashtags: string[] = []; let nestedLinks: {[uri: string]: string[]} = {}; // Try to find facets in different possible locations const possibleFacets = [ post.record?.facets, post.facets, post.embed?.record?.facets, post.embed?.recordWithMedia?.record?.facets ].filter(Boolean); // For nested content facets, we'll track them separately const nestedFacets = [ post.embed?.record?.value?.facets ].filter(Boolean); // Process all possible facet locations for the main post possibleFacets.forEach((facets: any[]) => { if (Array.isArray(facets)) { facets.forEach((facet) => { if (facet.features && Array.isArray(facet.features)) { facet.features.forEach((feature: any) => { // Extract links if (feature.$type === 'app.bsky.richtext.facet#link' && feature.uri) { links.push(feature.uri); } // Extract mentions else if (feature.$type === 'app.bsky.richtext.facet#mention' && feature.did) { mentions.push(feature.did); } // Extract hashtags else if (feature.$type === 'app.bsky.richtext.facet#tag' && feature.tag) { hashtags.push(feature.tag); } }); } }); } }); // Process nested facets separately to track which quoted post they belong to nestedFacets.forEach((facets: any[]) => { if (Array.isArray(facets)) { // Identify which post these facets belong to const parentUri = post.embed?.record?.uri || 'unknown'; if (!nestedLinks[parentUri]) { nestedLinks[parentUri] = []; } facets.forEach((facet) => { if (facet.features && Array.isArray(facet.features)) { facet.features.forEach((feature: any) => { // Extract links from nested content if (feature.$type === 'app.bsky.richtext.facet#link' && feature.uri) { nestedLinks[parentUri].push(feature.uri); } }); } }); } }); // Extract embed information from the post with improved nested structure handling let embedInfo: string[] = []; // Process embeds recursively to handle nested content function processEmbed(embed: any, depth: number = 0): void { if (!embed) return; const indent = ' '.repeat(depth); // Handle image embeds if (embed.$type === 'app.bsky.embed.images' && embed.images) { const imageCount = Array.isArray(embed.images) ? embed.images.length : 0; embedInfo.push(`${indent}🖼️ ${imageCount} image${imageCount !== 1 ? 's' : ''} attached`); // Add image details if available if (imageCount > 0 && Array.isArray(embed.images)) { embed.images.forEach((img: any, idx: number) => { const details: string[] = []; if (img.alt && img.alt.trim()) { details.push(`alt: "${img.alt}"`); } if (img.aspectRatio) { details.push(`aspect: ${img.aspectRatio.width}:${img.aspectRatio.height}`); } if (img.image?.mimeType) { details.push(`type: ${img.image.mimeType}`); } if (details.length > 0) { embedInfo.push(`${indent} Image ${idx + 1}: ${details.join(', ')}`); } }); } } // External link embeds (website cards) else if (embed.$type === 'app.bsky.embed.external' && embed.external) { const external = embed.external; embedInfo.push(`${indent}🔗 Website card:`); if (external.title) { embedInfo.push(`${indent} Title: ${external.title}`); } if (external.description) { embedInfo.push(`${indent} Description: ${external.description.substring(0, 100)}${external.description.length > 100 ? '...' : ''}`); } if (external.uri) { embedInfo.push(`${indent} URL: ${external.uri}`); links.push(external.uri); } if (external.thumb) { embedInfo.push(`${indent} Thumbnail: ${external.thumb.mimeType || 'image'}`); } } // Special handling for app.bsky.embed.record#view which is different from app.bsky.embed.record else if (embed.$type === 'app.bsky.embed.record#view' && embed.record) { embedInfo.push(`${indent}💬 Quoted post:`); // Get the record details const quotedRecord = embed.record; if (quotedRecord.$type) { embedInfo.push(`${indent} Type: ${quotedRecord.$type}`); } if (quotedRecord.uri) { embedInfo.push(`${indent} URI: ${quotedRecord.uri}`); // Add nested links if they exist for this URI if (nestedLinks[quotedRecord.uri] && nestedLinks[quotedRecord.uri].length > 0) { const uniqueNestedLinks = [...new Set(nestedLinks[quotedRecord.uri])]; embedInfo.push(`${indent} Links: ${uniqueNestedLinks.join(', ')}`); } } // Show author details if (quotedRecord.author) { const authorInfo = quotedRecord.author; embedInfo.push(`${indent} By: ${authorInfo.displayName || authorInfo.handle} (@${authorInfo.handle})`); } // Show the content - prioritize value.text over record.text const quotedText = quotedRecord.value?.text || quotedRecord.text; if (quotedText) { embedInfo.push(`${indent} Content: ${quotedText}`); } // Show statistics if (quotedRecord.likeCount !== undefined || quotedRecord.repostCount !== undefined) { const stats = [ quotedRecord.likeCount !== undefined ? `${quotedRecord.likeCount} likes` : null, quotedRecord.repostCount !== undefined ? `${quotedRecord.repostCount} reposts` : null, quotedRecord.replyCount !== undefined ? `${quotedRecord.replyCount} replies` : null ].filter(Boolean).join(', '); if (stats) { embedInfo.push(`${indent} Stats: ${stats}`); } } // Process nested embeds if (quotedRecord.embeds?.length > 0) { embedInfo.push(`${indent} Nested embeds:`); quotedRecord.embeds.forEach((nestedEmbed: any, idx: number) => { embedInfo.push(`${indent} Nested embed #${idx + 1}:`); processEmbed(nestedEmbed, depth + 2); }); } // Try to process nested embed in value if it exists if (quotedRecord.value?.embed) { embedInfo.push(`${indent} Nested content in quoted post:`); processEmbed(quotedRecord.value.embed, depth + 2); } } // Record embeds (quote posts) else if (embed.$type === 'app.bsky.embed.record' && embed.record) { embedInfo.push(`${indent}💬 Quoted post:`); // Add URI of quoted post if (embed.record.uri) { embedInfo.push(`${indent} URI: ${embed.record.uri}`); } // If the record is resolved and has data, show details if (embed.record.value || embed.record.author) { // Show the quoted author and text if available const quotedAuthor = embed.record.author || (embed.record.value?.author); const quotedText = embed.record.value?.text; const quotedRecord = embed.record.value || embed.record; if (quotedAuthor) { embedInfo.push(`${indent} By: ${quotedAuthor.displayName || quotedAuthor.handle} (@${quotedAuthor.handle})`); } if (quotedText) { embedInfo.push(`${indent} Content: ${quotedText}`); } // Handle stats if available if (embed.record.likeCount !== undefined || embed.record.repostCount !== undefined) { const stats = [ embed.record.likeCount !== undefined ? `${embed.record.likeCount} likes` : null, embed.record.repostCount !== undefined ? `${embed.record.repostCount} reposts` : null, embed.record.replyCount !== undefined ? `${embed.record.replyCount} replies` : null ].filter(Boolean).join(', '); if (stats) { embedInfo.push(`${indent} Stats: ${stats}`); } } // Handle nested embeds recursively if (quotedRecord.embed || quotedRecord.embeds?.length > 0) { embedInfo.push(`${indent} Nested content:`); // Process main embed if (quotedRecord.embed) { processEmbed(quotedRecord.embed, depth + 2); } // Process multiple embeds if (Array.isArray(quotedRecord.embeds)) { quotedRecord.embeds.forEach((nestedEmbed: any, idx: number) => { embedInfo.push(`${indent} Nested embed #${idx + 1}:`); processEmbed(nestedEmbed, depth + 2); }); } } } } // Record with media embeds (quote posts with images) else if (embed.$type === 'app.bsky.embed.recordWithMedia') { embedInfo.push(`${indent}💬 Quoted post with media:`); // Handle the record part if (embed.record?.record) { if (embed.record.record.uri) { embedInfo.push(`${indent} URI: ${embed.record.record.uri}`); } // If the record is resolved and has data, show details if (embed.record.record.value || embed.record.record.author) { const quotedAuthor = embed.record.record.author || (embed.record.record.value?.author); const quotedText = embed.record.record.value?.text; const quotedRecord = embed.record.record.value || embed.record.record; if (quotedAuthor) { embedInfo.push(`${indent} By: ${quotedAuthor.displayName || quotedAuthor.handle} (@${quotedAuthor.handle})`); } if (quotedText) { embedInfo.push(`${indent} Content: ${quotedText}`); } // Process nested embeds in the record if (quotedRecord.embed) { embedInfo.push(`${indent} Nested content in quote:`); processEmbed(quotedRecord.embed, depth + 2); } } } // Handle the media part if (embed.media) { embedInfo.push(`${indent} Attached media:`); processEmbed(embed.media, depth + 2); } } // If embed has its own embeds array (nested embeds) if (embed.embeds && Array.isArray(embed.embeds)) { embedInfo.push(`${indent}Multiple embedded content items:`); embed.embeds.forEach((subEmbed: any, idx: number) => { embedInfo.push(`${indent}Item #${idx + 1}:`); processEmbed(subEmbed, depth + 1); }); } } // Start embed processing from the top level const embed = post.embed || post.record?.embed; if (embed) { embedInfo.push('Embeds:'); processEmbed(embed); } // Format the post content with improved layout let formattedPost = `Post #${index + 1}:`; // Add repost information if applicable if (isRepost && reposter) { formattedPost += `\n🔄 Reposted by: ${reposter.displayName || reposter.handle} (@${reposter.handle})`; if (reason.indexedAt) { formattedPost += ` at ${new Date(reason.indexedAt).toLocaleString()}`; } } // Add author information with richer details formattedPost += `\nAuthor: ${author.displayName || author.handle} (@${author.handle})`; // Add thread context if available if (isReply && threadInfo.length > 0) { formattedPost += `\nThread: ${isReply ? 'Reply' : 'Thread starter'}`; formattedPost += `\n${threadInfo.join('\n')}`; } // Add post content formattedPost += `\nContent: ${post.record?.text || post.text || ''}`; // Add hashtags if present if (hashtags.length > 0) { formattedPost += `\nHashtags: ${hashtags.map(tag => `#${tag}`).join(' ')}`; } // Add mentions if present if (mentions.length > 0) { formattedPost += `\nMentions: ${mentions.join(', ')}`; } // Add engagement metrics const engagementMetrics = [ post.likeCount !== undefined ? `${post.likeCount} likes` : null, post.repostCount !== undefined ? `${post.repostCount} reposts` : null, post.replyCount !== undefined ? `${post.replyCount} replies` : null, post.quoteCount !== undefined ? `${post.quoteCount} quotes` : null ].filter(Boolean); if (engagementMetrics.length > 0) { formattedPost += `\nEngagement: ${engagementMetrics.join(', ')}`; } // Add embed information if present if (embedInfo.length > 0) { formattedPost += `\n${embedInfo.join('\n')}`; } // Add links if they exist and aren't already shown in embeds if (links.length > 0) { // Remove duplicates from links array const uniqueLinks = [...new Set(links)]; formattedPost += `\nLinks: ${uniqueLinks.join(', ')}`; } // Add post timestamp and URI formattedPost += `\nPosted: ${new Date(post.indexedAt).toLocaleString()}`; formattedPost += `\nURI: ${post.uri}`; formattedPost += `\n---`; return formattedPost; } /** * Format the summary text for the response */ export function formatSummaryText(postsCount: number, entityType: string = 'feed'): string { return `Retrieved ${postsCount} posts from the ${entityType}.`; } /** * Create a standardized error response */ export function createErrorResponse(message: string): McpErrorResponse { return { isError: true, content: [{ type: "text", text: message }] }; } /** * Create a standardized success response */ export function createSuccessResponse(text: string): McpSuccessResponse { return { content: [{ type: "text", text }] }; } /** * Fetch posts from a feed with pagination support * @param agent The ATP agent instance * @param feed The feed URI to fetch posts from * @param options Pagination and filtering options * @returns Array of fetched posts */ export async function fetchFeedPosts( agent: AtpAgent, feed: string, options: { maxPosts: number, initialCursor?: string } ): Promise<{posts: any[], cursor: string | undefined}> { const { maxPosts, initialCursor } = options; const MAX_FETCH_LOOPS = 10; // Safety limit for number of API calls // Initial fetch const initialFetch = async () => { const response = await agent.app.bsky.feed.getFeed({ feed, limit: Math.min(100, maxPosts), cursor: initialCursor }); if (!response.success) throw new Error("Failed to fetch feed"); return response; }; try { // First fetch const response = await initialFetch(); const allPosts: any[] = [...response.data.feed]; let nextCursor = response.data.cursor; let fetchCount = 1; // Paginate if needed and cursor is available while (nextCursor && allPosts.length < maxPosts && fetchCount < MAX_FETCH_LOOPS) { fetchCount++; try { const nextPage = await agent.app.bsky.feed.getFeed({ feed, limit: Math.min(100, maxPosts - allPosts.length), cursor: nextCursor }); if (!nextPage.success) break; // Add posts to our collection for (const post of nextPage.data.feed) { // Add the post allPosts.push(post); // Stop if we've reached the max if (allPosts.length >= maxPosts) break; } // Update cursor for next pagination nextCursor = nextPage.data.cursor; } catch (err) { // On error, just return what we have console.error("Error during pagination:", err); break; } } return { posts: allPosts, cursor: nextCursor }; } catch (err) { console.error("Error fetching feed:", err); return { posts: [], cursor: undefined }; } } /** * Validate a feed or list URI by fetching its information * @param agent The ATP agent instance * @param uri The feed or list URI to validate * @param type The type of URI ('feed' or 'list') * @returns The feed/list information or null if invalid */ export async function validateUri( agent: AtpAgent, uri: string, type: 'feed' | 'list' ): Promise<any | null> { try { let response; if (type === 'list' || uri.includes('app.bsky.graph.list')) { response = await agent.app.bsky.graph.getList({ list: uri }); } else { response = await agent.app.bsky.feed.getFeedGenerator({ feed: uri }); } if (!response.success) { return null; } return response.data; } catch (error) { return null; } } /** * Fetch posts from members of a list * @param agent The ATP agent instance * @param members Array of DIDs representing list members * @param options Pagination and filtering options * @returns Array of posts from the list members */ export async function fetchPostsFromListMembers( agent: AtpAgent, members: string[], options: { maxPosts: number } ): Promise<any[]> { const { maxPosts } = options; let allPosts: any[] = []; // Get posts from each member // (limit the number of members to avoid excessive API calls) const memberLimit = Math.min(members.length, 50); for (let i = 0; i < memberLimit && allPosts.length < maxPosts; i++) { const member = members[i]; try { // Try to fetch user posts const { posts: memberPosts } = await fetchUserPosts(agent, member, { maxPosts: Math.min(20, maxPosts - allPosts.length) // Only fetch a small number per member }); // Add to total posts allPosts.push(...memberPosts); } catch (err) { // Skip this member on error console.error(`Error fetching posts for member ${member}:`, err); continue; } } // Sort posts by indexedAt date (most recent first) allPosts.sort((a, b) => { const aTime = new Date(a.post.indexedAt).getTime(); const bTime = new Date(b.post.indexedAt).getTime(); return bTime - aTime; }); // Limit to the requested number of posts return allPosts.slice(0, maxPosts); } /** * Debugs the structure of a post to see where facets are stored * This is a temporary function to help with development */ export function debugPostStructure(post: any): void { console.error('DEBUG POST STRUCTURE:'); console.error('Post has record:', !!post.record); if (post.record) { console.error('Record properties:', Object.keys(post.record)); console.error('Has facets:', !!post.record.facets); if (post.record.facets) { console.error('First facet:', JSON.stringify(post.record.facets[0], null, 2)); } } // Check if facets might be at another location console.error('Post properties:', Object.keys(post)); console.error('Has facets at root:', !!post.facets); if (post.facets) { console.error('First facet at root:', JSON.stringify(post.facets[0], null, 2)); } } /** * Fetch posts from a specific user by handle or DID * @param agent The ATP agent instance * @param user The user handle or DID to fetch posts from * @param options Pagination and filtering options * @returns Array of fetched posts */ export async function fetchUserPosts( agent: AtpAgent, user: string, options: { maxPosts: number, initialCursor?: string } ): Promise<{posts: any[], cursor: string | undefined}> { const { maxPosts, initialCursor } = options; const MAX_FETCH_LOOPS = 10; // Safety limit for number of API calls // Initial fetch const initialFetch = async () => { const response = await agent.app.bsky.feed.getAuthorFeed({ actor: user, limit: Math.min(100, maxPosts), cursor: initialCursor }); if (!response.success) throw new Error("Failed to fetch user posts"); return response; }; try { // First fetch const response = await initialFetch(); const allPosts: any[] = [...response.data.feed]; let nextCursor = response.data.cursor; let fetchCount = 1; // Paginate if needed and cursor is available while (nextCursor && allPosts.length < maxPosts && fetchCount < MAX_FETCH_LOOPS) { fetchCount++; try { const nextPage = await agent.app.bsky.feed.getAuthorFeed({ actor: user, limit: Math.min(100, maxPosts - allPosts.length), cursor: nextCursor }); if (!nextPage.success) break; // Add posts to our collection for (const post of nextPage.data.feed) { // Add the post allPosts.push(post); // Stop if we've reached the max if (allPosts.length >= maxPosts) break; } // Update cursor for next pagination nextCursor = nextPage.data.cursor; } catch (err) { // On error, just return what we have console.error("Error during pagination:", err); break; } } return { posts: allPosts, cursor: nextCursor }; } catch (err) { console.error("Error fetching user posts:", err); return { posts: [], cursor: undefined }; } }