diff --git a/package.json b/package.json index f4d2b25a6..4339d982c 100644 --- a/package.json +++ b/package.json @@ -85,6 +85,8 @@ "@sparticuz/chromium": "132.0.0", "@svelte-put/shortcut": "^3.2.0", "@sveltejs/adapter-vercel": "^5.7.2", + "@tmcp/adapter-zod-v3": "^0.2.1", + "@tmcp/transport-http": "^0.8.0", "@types/core-js": "^2.5.8", "@upstash/redis": "^1.35.1", "chroma-js": "^2.6.0", @@ -115,6 +117,7 @@ "svelte-local-storage-store": "^0.6.4", "svelte-turnstile": "^0.8.0", "sveltekit-search-params": "^2.1.2", + "tmcp": "^1.16.1", "ts-node": "^10.9.2", "unified": "^11.0.5", "waait": "^1.0.5", @@ -124,5 +127,8 @@ }, "prisma": { "seed": "node --loader ts-node/esm prisma/seed.ts" + }, + "volta": { + "node": "22.20.0" } } diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 28c8e74ac..a8dcd9aa9 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -44,6 +44,12 @@ importers: '@sveltejs/adapter-vercel': specifier: ^5.7.2 version: 5.7.2(@sveltejs/kit@2.5.27(@sveltejs/vite-plugin-svelte@4.0.4(svelte@5.35.3)(vite@5.4.19(@types/node@22.16.0)))(svelte@5.35.3)(vite@5.4.19(@types/node@22.16.0)))(rollup@4.44.2) + '@tmcp/adapter-zod-v3': + specifier: ^0.2.1 + version: 0.2.1(tmcp@1.16.1(typescript@5.8.3))(zod@3.25.75) + '@tmcp/transport-http': + specifier: ^0.8.0 + version: 0.8.0(tmcp@1.16.1(typescript@5.8.3)) '@types/core-js': specifier: ^2.5.8 version: 2.5.8 @@ -134,6 +140,9 @@ importers: sveltekit-search-params: specifier: ^2.1.2 version: 2.1.2(@sveltejs/kit@2.5.27(@sveltejs/vite-plugin-svelte@4.0.4(svelte@5.35.3)(vite@5.4.19(@types/node@22.16.0)))(svelte@5.35.3)(vite@5.4.19(@types/node@22.16.0)))(svelte@5.35.3)(vite@5.4.19(@types/node@22.16.0)) + tmcp: + specifier: ^1.16.1 + version: 1.16.1(typescript@5.8.3) ts-node: specifier: ^10.9.2 version: 10.9.2(@types/node@22.16.0)(typescript@5.8.3) @@ -1636,6 +1645,9 @@ packages: resolution: {integrity: sha512-7SAhVa4nHAP7bxEhnMFuwV7KNDqAF4RVGZNwRhvyss8lJNT7Syi9nPwIPYvsHtjJ1S7Nx8OL8uqe9V3a/ED/Mg==} engines: {node: '>= 16'} + '@standard-schema/spec@1.0.0': + resolution: {integrity: sha512-m2bOd0f2RT9k8QJx1JN85cZYyH1RqFBdlwtkSlf4tBDYLCiiZnv1fIIwacK6cqwXavOydf0NPToMQgpKq+dVlA==} + '@svelte-put/shortcut@3.2.0': resolution: {integrity: sha512-3VFU6TX4DwXT7vrtKGqwWa5WqEsFWQVZMSffE0owCwKUIBAhjGtZGy69G4+2quGhl/r+qi4jLrXkE00xw38M7g==} peerDependencies: @@ -1690,6 +1702,26 @@ packages: svelte: ^5.0.0-next.96 || ^5.0.0 vite: ^5.0.0 + '@tmcp/adapter-zod-v3@0.2.1': + resolution: {integrity: sha512-0CwZqyy1UFYlGvzBPdkxaMZJx7FKVUNqnU41VgZ4X5LCxHgf2/31ZGWcpFjCZNwoorrjcbFeABQ81ABJThRJ0w==} + peerDependencies: + tmcp: ^1.10.2 + zod: ^3.0.0 + + '@tmcp/session-manager@0.2.0': + resolution: {integrity: sha512-/MjEFFvnvrV3DbnwEAXMBKhIFmm7z81EgazBtPhKP9i5pe37/iDkYHOa+SD6aaqT3V7gACQ1nmqoJH3MdrC7gQ==} + peerDependencies: + tmcp: ^1.16.0 + + '@tmcp/transport-http@0.8.0': + resolution: {integrity: sha512-XkIFDIDr8zzQhJiwI+6He0pIGmEg0SAwHdsv914UtL0zqKuUfFvFLLARkryB95Rhp1ib4nVWAgLcV1FtdWGzIw==} + peerDependencies: + '@tmcp/auth': ^0.3.3 + tmcp: ^1.16.0 + peerDependenciesMeta: + '@tmcp/auth': + optional: true + '@tootallnate/quickjs-emscripten@0.23.0': resolution: {integrity: sha512-C5Mc6rdnsaJDjO3UpGW/CQTHtCKaYlScZTly4JIu97Jxo/odCiH0ITnDXSJPTOrEKk/ycSZ0AOgTmkDtkOsvIA==} @@ -3184,6 +3216,9 @@ packages: resolution: {integrity: sha512-fi0NG4bPjCHunUJffmLd0gxssIgkNmArMvis4iNah6Owg1MCJjWhEcDLmsK6iGkJq3tHwbDkTlce70/tmXN4cQ==} engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + json-rpc-2.0@1.7.1: + resolution: {integrity: sha512-JqZjhjAanbpkXIzFE7u8mE/iFblawwlXtONaCvRqI+pyABVz7B4M1EUNpyVW+dZjqgQ2L5HFmZCmOCgUKm00hg==} + json-schema-traverse@0.4.1: resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} @@ -4251,6 +4286,9 @@ packages: sprintf-js@1.1.3: resolution: {integrity: sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==} + sqids@0.3.0: + resolution: {integrity: sha512-lOQK1ucVg+W6n3FhRwwSeUijxe93b51Bfz5PMRMihVf1iVkl82ePQG7V5vwrhzB11v0NtsR25PSZRGiSomJaJw==} + sqlstring@2.3.3: resolution: {integrity: sha512-qC9iz2FlN7DQl3+wjwn3802RTyjCx7sDvfQEXchwa6CWOx07/WVfh91gBmQ9fahw8snwGEWU3xGzOt4tFyHLxg==} engines: {node: '>= 0.6'} @@ -4471,6 +4509,9 @@ packages: resolution: {integrity: sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q==} engines: {node: '>=14.0.0'} + tmcp@1.16.1: + resolution: {integrity: sha512-pmqdFMfYi8e57p8bzpIkEtTt6GfAXGpub8rwySgo1yEEQogTpGdXFjGbISeS0SKrYN/MnIsHN6nj8gPjmAc+EQ==} + to-regex-range@5.0.1: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} @@ -4594,6 +4635,9 @@ packages: uri-js@4.4.1: resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} + uri-template-matcher@1.1.1: + resolution: {integrity: sha512-ooMjzxaqlquLZZU0Y+Ol+RAp5xav3oJ3qc3gO585QFmB9iX88ip8pflaCKKtkwPd40ro2noB8yQG2OhFddmddg==} + utf-8-validate@5.0.10: resolution: {integrity: sha512-Z6czzLq4u8fPOyx7TU6X3dvUZVvoJmxSQ+IcrlmagKhilxlhZgxPK6C5Jqbkw1IDUmFTM+cz9QDnnLTwDz/2gQ==} engines: {node: '>=6.14.2'} @@ -4607,6 +4651,14 @@ packages: v8-compile-cache-lib@3.0.1: resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} + valibot@1.1.0: + resolution: {integrity: sha512-Nk8lX30Qhu+9txPYTwM0cFlWLdPFsFr6LblzqIySfbZph9+BFsAHsNvHOymEviUepeIW6KFHzpX8TKhbptBXXw==} + peerDependencies: + typescript: '>=5' + peerDependenciesMeta: + typescript: + optional: true + vfile-location@5.0.2: resolution: {integrity: sha512-NXPYyxyBSH7zB5U6+3uDdd6Nybz6o6/od9rk8bp9H8GR3L+cm/fC0uUTbqBmUTnMCUDslAGBOIKNfvvb+gGlDg==} @@ -4829,6 +4881,11 @@ packages: zimmerframe@1.1.2: resolution: {integrity: sha512-rAbqEGa8ovJy4pyBxZM70hg4pE6gDgaQ0Sl9M3enG3I0d6H4XSAM3GeNGLKnsBpuijUow064sf7ww1nutC5/3w==} + zod-to-json-schema@3.24.6: + resolution: {integrity: sha512-h/z3PKvcTcTetyjl1fkj79MHNEjm+HpD6NXheWjzOekY7kV+lwDYnHw+ivHkijnCSMz1yJaWBD9vu/Fcmk+vEg==} + peerDependencies: + zod: ^3.24.1 + zod@3.24.1: resolution: {integrity: sha512-muH7gBL9sI1nciMZV67X5fTKKBLtwpZ5VBp1vsOQzj1MhrBZ4wlVCm3gedKZWLp0Oyel8sIGfeiz54Su+OVT+A==} @@ -6251,6 +6308,8 @@ snapshots: - bare-buffer - debug + '@standard-schema/spec@1.0.0': {} + '@svelte-put/shortcut@3.2.0(svelte@5.35.3)': dependencies: svelte: 5.35.3 @@ -6332,6 +6391,24 @@ snapshots: transitivePeerDependencies: - supports-color + '@tmcp/adapter-zod-v3@0.2.1(tmcp@1.16.1(typescript@5.8.3))(zod@3.25.75)': + dependencies: + '@standard-schema/spec': 1.0.0 + '@types/json-schema': 7.0.15 + tmcp: 1.16.1(typescript@5.8.3) + zod: 3.25.75 + zod-to-json-schema: 3.24.6(zod@3.25.75) + + '@tmcp/session-manager@0.2.0(tmcp@1.16.1(typescript@5.8.3))': + dependencies: + tmcp: 1.16.1(typescript@5.8.3) + + '@tmcp/transport-http@0.8.0(tmcp@1.16.1(typescript@5.8.3))': + dependencies: + '@tmcp/session-manager': 0.2.0(tmcp@1.16.1(typescript@5.8.3)) + esm-env: 1.2.2 + tmcp: 1.16.1(typescript@5.8.3) + '@tootallnate/quickjs-emscripten@0.23.0': {} '@tsconfig/node10@1.0.9': {} @@ -7937,6 +8014,8 @@ snapshots: json-parse-even-better-errors@3.0.2: {} + json-rpc-2.0@1.7.1: {} + json-schema-traverse@0.4.1: {} json-schema-traverse@1.0.0: {} @@ -9251,6 +9330,8 @@ snapshots: sprintf-js@1.1.3: {} + sqids@0.3.0: {} + sqlstring@2.3.3: {} stackback@0.0.2: {} @@ -9506,6 +9587,16 @@ snapshots: tinyspy@3.0.2: {} + tmcp@1.16.1(typescript@5.8.3): + dependencies: + '@standard-schema/spec': 1.0.0 + json-rpc-2.0: 1.7.1 + sqids: 0.3.0 + uri-template-matcher: 1.1.1 + valibot: 1.1.0(typescript@5.8.3) + transitivePeerDependencies: + - typescript + to-regex-range@5.0.1: dependencies: is-number: 7.0.0 @@ -9639,6 +9730,8 @@ snapshots: dependencies: punycode: 2.3.1 + uri-template-matcher@1.1.1: {} + utf-8-validate@5.0.10: dependencies: node-gyp-build: 4.8.4 @@ -9656,6 +9749,10 @@ snapshots: v8-compile-cache-lib@3.0.1: {} + valibot@1.1.0(typescript@5.8.3): + optionalDependencies: + typescript: 5.8.3 + vfile-location@5.0.2: dependencies: '@types/unist': 3.0.2 @@ -9857,6 +9954,10 @@ snapshots: zimmerframe@1.1.2: {} + zod-to-json-schema@3.24.6(zod@3.25.75): + dependencies: + zod: 3.25.75 + zod@3.24.1: {} zod@3.25.75: {} diff --git a/src/hooks.server.ts b/src/hooks.server.ts index cc75f7f25..f7e479acb 100644 --- a/src/hooks.server.ts +++ b/src/hooks.server.ts @@ -10,6 +10,7 @@ import { dev } from '$app/environment'; import { UPSPLASH_TOKEN, UPSPLASH_URL } from '$env/static/private'; import { Redis } from '@upstash/redis'; import { nodeProfilingIntegration } from '@sentry/profiling-node'; +import { transport } from './lib/mcp'; export const cache_status = UPSPLASH_URL && UPSPLASH_TOKEN ? 'ONLINE' : 'OFFLINE'; @@ -106,6 +107,25 @@ export const safe_form_data: Handle = async function ({ event, resolve }) { return resolve(event); }; +export const mcp: Handle = async function ({ event, resolve }) { + const mcp_response = await transport.respond(event.request); + // we are deploying on vercel the SSE connection will timeout after 5 minutes...for + // the moment we are not sending back any notifications (logs, or list changed notifications) + // so it's a waste of resources to keep a connection open that will error + // after 5 minutes making the logs dirty. For this reason if we have a response from + // the MCP server and it's a GET request we just return an empty response (it has to be + // 200 or the MCP client will complain) + if (mcp_response && event.request.method === 'GET') { + try { + await mcp_response.body?.cancel(); + } catch { + // ignore + } + return new Response('', { status: 200 }); + } + return mcp_response ?? resolve(event); +}; + // * END HOOKS // Wraps requests in this sequence of hooks @@ -115,7 +135,8 @@ export const handle: Handle = sequence( auth, admin, safe_form_data, - document_policy + document_policy, + mcp ); export const handleError = Sentry.handleErrorWithSentry(); diff --git a/src/lib/mcp/icons/index.ts b/src/lib/mcp/icons/index.ts new file mode 100644 index 000000000..1acbc1c7a --- /dev/null +++ b/src/lib/mcp/icons/index.ts @@ -0,0 +1,13 @@ +import type { Icons } from 'tmcp'; + +export const icons: Icons['icons'] = [ + { + src: 'https://syntax.fm/android-chrome-192x192.png', + mimeType: 'image/png' + }, + { + // fallback to data:image format of the same image + src: '', + mimeType: 'image/png' + } +]; diff --git a/src/lib/mcp/index.ts b/src/lib/mcp/index.ts new file mode 100644 index 000000000..3bee972c5 --- /dev/null +++ b/src/lib/mcp/index.ts @@ -0,0 +1,34 @@ +import { McpServer } from 'tmcp'; +import { ZodV3JsonSchemaAdapter } from '@tmcp/adapter-zod-v3'; +import { icons } from './icons/index.js'; +import { HttpTransport } from '@tmcp/transport-http'; +import { setup_tools } from './tools'; +import { setup_resources } from './resources'; + +export type SyntaxMCP = typeof server; + +const server = new McpServer( + { + description: 'MCP server to access Syntax episodes and transcripts', + name: 'Syntax MCP Server', + websiteUrl: 'https://syntax.fm', + version: '0.1.0', + icons + }, + { + adapter: new ZodV3JsonSchemaAdapter(), + capabilities: { + tools: {}, + resources: {}, + completions: {} + } + } +); + +setup_tools(server); +setup_resources(server); + +export const transport = new HttpTransport(server, { + cors: true, + path: '/mcp' +}); diff --git a/src/lib/mcp/resources/index.ts b/src/lib/mcp/resources/index.ts new file mode 100644 index 000000000..ff8bb3136 --- /dev/null +++ b/src/lib/mcp/resources/index.ts @@ -0,0 +1,96 @@ +import { prisma_client } from '$/server/prisma-client.js'; +import type { SyntaxMCP } from '../index.js'; +import { transcript_to_string } from '../utils.js'; +import { icons } from '../icons/index.js'; + +export function setup_resources(server: SyntaxMCP) { + server.template( + { + uri: 'syntaxfm://show/{slug}.json', + description: 'Get all info about a specific show given its slug', + name: 'show_info', + title: 'Show Info', + icons, + async list() { + const slugs = await prisma_client.show.findMany({ + select: { + slug: true, + title: true + }, + orderBy: { + date: 'desc' + } + }); + return slugs.map((s) => ({ + name: `${s.title}.json`, + value: s.slug, + uri: `syntaxfm://show/${s.slug}.json`, + title: `${s.title}.json` + })); + }, + complete: { + async slug(query) { + const slugs = await prisma_client.show.findMany({ + select: { + slug: true + }, + where: { + slug: { + contains: query.toString() + } + }, + take: 50 + }); + + return { + completion: { + values: slugs.map((s) => s.slug) + } + }; + } + } + }, + async (uri, { slug }) => { + const show = await prisma_client.show.findFirst({ + select: { + number: true, + show_notes: true, + title: true, + youtube_url: true, + date: true, + guests: true, + show_type: true, + url: true, + videos: true, + transcript: { + select: { + utterances: true + } + } + }, + where: { + slug: { + equals: slug.toString() + } + } + }); + + if (!show) throw new Error(`No show found with slug ${slug}`); + + return { + contents: [ + { + uri, + mimeType: 'application/json', + text: JSON.stringify({ + show: { + ...show, + transcript: show.transcript ? transcript_to_string(show.transcript) : '' + } + }) + } + ] + }; + } + ); +} diff --git a/src/lib/mcp/tools/index.ts b/src/lib/mcp/tools/index.ts new file mode 100644 index 000000000..05e9eb70f --- /dev/null +++ b/src/lib/mcp/tools/index.ts @@ -0,0 +1,198 @@ +import z from 'zod'; +import type { SyntaxMCP } from '../index.js'; +import { prisma_client } from '$/server/prisma-client.js'; +import { transcript_to_string } from '../utils.js'; +import { icons } from '../icons/index.js'; + +export function setup_tools(server: SyntaxMCP) { + server.tool( + { + name: 'list_episodes', + description: 'Get a list of all the episodes with relative show notes', + icons, + schema: z.object({ + query: z + .string() + .describe( + 'A comma separated list of keywords to search for in the title or the notes for the shows' + ) + .optional(), + year_cutoff: z + .number() + .optional() + .describe( + 'Only shows published after this year (YYYY) will be returned, will default to current year but better to ask the user for a specific range if possible' + ) + }), + outputSchema: z.object({ + shows: z.array( + z.object({ + show_notes: z.string(), + title: z.string(), + number: z.number() + }) + ) + }) + }, + async ({ query = '', year_cutoff }) => { + const year = new Date(); + if (year_cutoff) { + year.setFullYear(year_cutoff); + } + year.setMonth(0, 1); + + const shows = await prisma_client.show.findMany({ + select: { + number: true, + show_notes: true, + title: true + }, + where: { + AND: [ + { date: { gte: year } }, + { + OR: query.split(',').flatMap((query_part) => [ + { + title: { + contains: query_part.trim() + } + }, + { + show_notes: { + contains: query_part.trim() + } + } + ]) + } + ] + }, + orderBy: { + number: 'desc' + } + }); + return { + content: [ + { + type: 'text', + text: JSON.stringify({ shows }) + } + ], + structuredContent: { + shows + } + }; + } + ); + + server.tool( + { + name: 'get_episode', + description: 'Get information about a specific episode by its number', + icons, + schema: z.object({ + show_number: z + .number() + .describe('The number of the episode to get information about (e.g. 500 for episode 500)') + }) + }, + async ({ show_number }) => { + const show = await prisma_client.show.findFirst({ + select: { + number: true, + show_notes: true, + title: true, + youtube_url: true, + date: true, + guests: true, + show_type: true, + url: true, + videos: true, + transcript: { + select: { + utterances: true + } + } + }, + where: { + number: { + equals: show_number + } + } + }); + + if (!show) { + return { + isError: true, + content: [ + { + type: 'text', + text: `No show found with number ${show_number}` + } + ] + }; + } + + return { + content: [ + { + type: 'text', + text: JSON.stringify({ + show: { + ...show, + transcript: show.transcript ? transcript_to_string(show.transcript) : '' + } + }) + } + ] + }; + } + ); + + server.tool( + { + name: 'get_transcript', + description: 'Get the transcript for a specific show number', + icons, + schema: z.object({ + show_number: z + .number() + .describe('The number of the episode to get transcript of (e.g. 500 for episode 500)') + }) + }, + async ({ show_number }) => { + const transcript = await prisma_client.transcript.findFirst({ + select: { + utterances: true + }, + where: { + show_number: { + equals: show_number + } + } + }); + + if (!transcript) { + return { + isError: true, + content: [ + { + type: 'text', + text: `No transcript found with number ${show_number}` + } + ] + }; + } + + const text = transcript_to_string(transcript); + + return { + content: [ + { + type: 'text', + text + } + ] + }; + } + ); +} diff --git a/src/lib/mcp/utils.ts b/src/lib/mcp/utils.ts new file mode 100644 index 000000000..d2cff21a2 --- /dev/null +++ b/src/lib/mcp/utils.ts @@ -0,0 +1,13 @@ +import type { Prisma } from '@prisma/client'; + +type TranscriptWithUtterances = Prisma.TranscriptGetPayload<{ + select: { utterances: true }; +}>; + +export function transcript_to_string(transcript: TranscriptWithUtterances): string { + return transcript.utterances + .map((utterance) => { + return `${utterance.start}-${utterance.end} ${utterance.speaker}: ${utterance.transcript_value}`; + }) + .join('\n'); +}