UNPKG

@aashari/mcp-server-atlassian-confluence

Version:

Node.js/TypeScript MCP server for Atlassian Confluence. Provides tools enabling AI systems (LLMs) to list/get spaces & pages (content formatted as Markdown) and search via CQL. Connects AI seamlessly to Confluence knowledge bases using the standard MCP in

198 lines (197 loc) 8.83 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.formatSpacesList = formatSpacesList; exports.formatSpaceDetails = formatSpaceDetails; const formatter_util_js_1 = require("../utils/formatter.util.js"); /** * Format a list of spaces for display * @param spacesData - Raw spaces data from the API * @returns Formatted string with spaces information in markdown format */ function formatSpacesList(spacesData) { if (!spacesData.results || spacesData.results.length === 0) { return ('No Confluence spaces found matching your criteria.' + '\n\n' + (0, formatter_util_js_1.formatSeparator)() + '\n' + `*Information retrieved at: ${(0, formatter_util_js_1.formatDate)(new Date())}*`); } const lines = [(0, formatter_util_js_1.formatHeading)('Confluence Spaces', 1), '']; // Use the numbered list formatter for consistent formatting const formattedList = (0, formatter_util_js_1.formatNumberedList)(spacesData.results, (space, _index) => { // Ensure space has the correct inferred type, or use any if complex const typedSpace = space; const itemLines = []; itemLines.push((0, formatter_util_js_1.formatHeading)(typedSpace.name, 2)); // Basic properties const properties = { ID: typedSpace.id, Key: typedSpace.key, Type: typedSpace.type, Status: typedSpace.status, Created: typedSpace.createdAt ? (0, formatter_util_js_1.formatDate)(typedSpace.createdAt) : 'N/A', 'Homepage ID': typedSpace.homepageId || 'Not set', Description: typedSpace.description?.view?.value || 'Not available', URL: (0, formatter_util_js_1.formatUrl)(spacesData._links?.base ? `${spacesData._links.base}/spaces/${typedSpace.key}` : `/spaces/${typedSpace.key}`, typedSpace.key), }; if (typedSpace.currentActiveAlias) { properties['Alias'] = typedSpace.currentActiveAlias; } // Format as a bullet list with proper formatting for each value type itemLines.push((0, formatter_util_js_1.formatBulletList)(properties, (key) => key)); return itemLines.join('\n'); }); lines.push(formattedList); // Add standard footer with timestamp lines.push('\n\n' + (0, formatter_util_js_1.formatSeparator)()); lines.push(`*Information retrieved at: ${(0, formatter_util_js_1.formatDate)(new Date())}*`); return lines.join('\n'); } /** * Format detailed space information for display * @param spaceData - Raw space details from the API * @param homepageContent - Optional homepage content to include * @param topLevelPagesData - Optional top-level pages data to include * @returns Formatted string with space details in markdown format */ function formatSpaceDetails(spaceData, homepageContent, topLevelPagesData) { // Create URL const baseUrl = spaceData._links.base || ''; const spaceUrl = spaceData._links.webui || ''; const fullUrl = spaceUrl.startsWith('http') ? spaceUrl : `${baseUrl}${spaceUrl}`; const lines = [ (0, formatter_util_js_1.formatHeading)(`Confluence Space: ${spaceData.key}`, 1), '', `> A ${spaceData.status} ${spaceData.type} space with key \`${spaceData.key}\` created on ${(0, formatter_util_js_1.formatDate)(spaceData.createdAt)}.`, '', (0, formatter_util_js_1.formatHeading)('Basic Information', 2), ]; // Format basic information as a bullet list const basicProperties = { ID: spaceData.id, Key: spaceData.key, Name: spaceData.name, Type: spaceData.type, Status: spaceData.status, 'Created At': (0, formatter_util_js_1.formatDate)(spaceData.createdAt), 'Author ID': spaceData.authorId, 'Homepage ID': spaceData.homepageId, 'Current Alias': spaceData.currentActiveAlias, }; lines.push((0, formatter_util_js_1.formatBulletList)(basicProperties, (key) => key)); // Description section if (spaceData.description?.view?.value || spaceData.description?.plain?.value) { lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Description', 2)); const viewValue = spaceData.description?.view?.value; const plainValue = spaceData.description?.plain?.value; if (viewValue && viewValue.trim()) { lines.push(viewValue.trim()); } else if (plainValue && plainValue.trim()) { lines.push(plainValue.trim()); } else { lines.push('*No description provided*'); } } // Homepage content section (placed after description) if (spaceData.homepageId) { lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Homepage Content', 2)); if (homepageContent) { lines.push(homepageContent); } else { lines.push('*No homepage content available*'); } } // Top-level pages section if (topLevelPagesData && topLevelPagesData.results && topLevelPagesData.results.length > 0) { lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Recent Pages', 2)); const pagesFormatted = (0, formatter_util_js_1.formatNumberedList)(topLevelPagesData.results, (page) => { // Get the page URL const pageUrl = page._links?.webui || ''; const fullPageUrl = pageUrl.startsWith('http') ? pageUrl : `${baseUrl}${pageUrl}`; // Format last modified date const lastModified = page.version?.createdAt ? (0, formatter_util_js_1.formatDate)(page.version.createdAt) : 'N/A'; // Create a list of page properties const pageProps = { ID: page.id, Title: page.title, 'Last Modified': lastModified, Link: (0, formatter_util_js_1.formatUrl)(fullPageUrl, 'Open Page'), }; // Return the formatted page info return (0, formatter_util_js_1.formatBulletList)(pageProps, (key) => key); }); lines.push(pagesFormatted); lines.push(''); // Add link to view all pages in this space const spaceViewAllPagesUrl = `${baseUrl}/spaces/${spaceData.key}/pages`; lines.push(`*${(0, formatter_util_js_1.formatUrl)(spaceViewAllPagesUrl, 'View all pages in this space')}*`); } else if (topLevelPagesData) { lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Recent Pages', 2)); lines.push('*No pages found in this space*'); } // Labels section if (spaceData.labels?.results) { lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Labels', 2)); if (spaceData.labels.results.length === 0) { lines.push('*No labels assigned to this space*'); } else { const labelLines = []; spaceData.labels.results.forEach((label) => { const prefix = label.prefix ? `${label.prefix}:` : ''; labelLines.push(`- **${prefix}${label.name}** (ID: ${label.id})`); }); lines.push(labelLines.join('\n')); // The meta property might not have hasMore in the Zod schema if (spaceData.labels.meta?.count && spaceData.labels.results.length < (spaceData.labels.meta.count || 0)) { lines.push(''); lines.push('*More labels are available but not shown*'); } } } // Links section lines.push(''); lines.push((0, formatter_util_js_1.formatHeading)('Links', 2)); const links = []; links.push(`- **Web UI**: ${fullUrl}`); links.push(`- ${(0, formatter_util_js_1.formatUrl)(fullUrl, 'Open in Confluence')}`); if (spaceData.homepageId) { // Construct homepage URL carefully using base and ID const homepagePath = `/wiki/spaces/${spaceData.key}/pages/${spaceData.homepageId}`; const fullHomepageUrl = `${baseUrl}${homepagePath}`; links.push(`- ${(0, formatter_util_js_1.formatUrl)(fullHomepageUrl, 'View Homepage')}`); } lines.push(links.join('\n')); // Add standard footer with timestamp lines.push('\n\n' + (0, formatter_util_js_1.formatSeparator)()); lines.push(`*Information retrieved at: ${(0, formatter_util_js_1.formatDate)(new Date())}*`); // Optionally keep the direct link if (fullUrl) { lines.push(`*View this space in Confluence: ${fullUrl}*`); } return lines.join('\n'); }