Skip to main content
Glama
documents.ts4.53 kB
import { z } from "zod"; import type { ShortcutClientWrapper } from "@/client/shortcut"; import type { CustomMcpServer } from "@/mcp/CustomMcpServer"; import { BaseTools } from "./base"; export class DocumentTools extends BaseTools { static create(client: ShortcutClientWrapper, server: CustomMcpServer) { const tools = new DocumentTools(client); server.addToolWithWriteAccess( "documents-create", "Create a new document in Shortcut with a title and content. Returns the document's id, title, and app_url. Note: Use HTML markup for the content (e.g., <p>, <h1>, <ul>, <strong>) rather than Markdown.", { title: z.string().max(256).describe("The title for the new document (max 256 characters)"), content: z .string() .describe( "The content for the new document in HTML format (e.g., <p>Hello</p>, <h1>Title</h1>, <ul><li>Item</li></ul>)", ), }, async ({ title, content }) => await tools.createDocument(title, content), ); server.addToolWithReadAccess( "documents-list", "List all documents in Shortcut.", async () => await tools.listDocuments(), ); server.addToolWithReadAccess( "documents-search", "Find documents.", { nextPageToken: z .string() .optional() .describe( "If a next_page_token was returned from the search result, pass it in to get the next page of results. Should be combined with the original search parameters.", ), title: z.string().describe("Find documents matching the specified name"), archived: z .boolean() .optional() .describe("Find only documents matching the specified archived status"), createdByCurrentUser: z .boolean() .optional() .describe("Find only documents created by current user"), followedByCurrentUser: z .boolean() .optional() .describe("Find only documents followed by current user"), }, async ({ nextPageToken, title, archived, createdByCurrentUser, followedByCurrentUser }) => await tools.searchDocuments( { title, archived, createdByCurrentUser, followedByCurrentUser }, nextPageToken, ), ); server.addToolWithReadAccess( "documents-get-by-id", "Get a document as markdown by its ID", { docId: z.string().describe("The ID of the document to retrieve"), }, async ({ docId }: { docId: string }) => await tools.getDocumentById(docId), ); return tools; } private async createDocument(title: string, content: string) { try { const doc = await this.client.createDoc({ title, content, }); return this.toResult("Document created successfully", { id: doc.id, title: doc.title, app_url: doc.app_url, }); } catch (error) { const errorMessage = error instanceof Error ? error.message : "Unknown error"; return this.toResult(`Failed to create document: ${errorMessage}`); } } private async listDocuments() { try { const docs = await this.client.listDocs(); if (!docs?.length) return this.toResult("No documents were found."); return this.toResult(`Found ${docs.length} documents.`, docs); } catch (error) { const errorMessage = error instanceof Error ? error.message : "Unknown error"; return this.toResult(`Failed to list documents: ${errorMessage}`); } } private async searchDocuments( params: { title: string; archived?: boolean; createdByCurrentUser?: boolean; followedByCurrentUser?: boolean; }, nextPageToken?: string, ) { try { const { documents, total, next_page_token } = await this.client.searchDocuments({ ...params, nextPageToken, }); if (!documents) throw new Error(`Failed to search for document matching your query.`); if (!documents.length) return this.toResult(`Result: No documents found.`); return this.toResult( `Result (${documents.length} shown of ${total} total documents found):`, documents, next_page_token, ); } catch (error) { const errorMessage = error instanceof Error ? error.message : "Unknown error"; return this.toResult(`Failed to search documents: ${errorMessage}`); } } private async getDocumentById(docId: string) { try { const doc = await this.client.getDocById(docId); if (!doc) return this.toResult(`Document with ID ${docId} not found.`); return this.toResult(`Document with ID ${docId}`, doc); } catch (error) { const errorMessage = error instanceof Error ? error.message : "Unknown error"; return this.toResult(`Failed to get document: ${errorMessage}`); } } }

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/useshortcut/mcp-server-shortcut'

If you have feedback or need assistance with the MCP directory API, please join our Discord server