Skip to content

Commit cf62286

Browse files
committed
fix: use data: uri for local icon & add icons to tools + resources + prompts
1 parent a4dfaab commit cf62286

File tree

10 files changed

+37
-10
lines changed

10 files changed

+37
-10
lines changed

.changeset/hip-cats-marry.md

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@sveltejs/mcp': patch
3+
---
4+
5+
fix: use `data:` uri for local icon & add icons to tools + resources + prompts

.vscode/mcp-snippets.code-snippets

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
"body": [
1313
"import type { SvelteMcp } from '../../index.js';",
1414
"import * as v from 'valibot';",
15+
"import { icons } from '../../icons/index.js';",
1516
"",
1617
"export function ${1:function_name}(server: SvelteMcp) {",
1718
"\t$0",
@@ -35,6 +36,7 @@
3536
"prefix": "!prompt",
3637
"body": [
3738
"import type { SvelteMcp } from '../../index.js';",
39+
"import { icons } from '../../icons/index.js';",
3840
"",
3941
"/**",
4042
" * Function that actually generates the prompt string. You can use this in the MCP server handler to generate the prompt, it can accept arguments",
@@ -69,6 +71,7 @@
6971
"\t\t\ttitle: '${2:title}',",
7072
"\t\t\tdescription:",
7173
"\t\t\t\t'${3:llm_description}',",
74+
"\t\t\ticons,",
7275
"\t\t},",
7376
"\t\tasync () => {",
7477
"\t\t\treturn {",

packages/mcp-server/src/mcp/handlers/prompts/svelte-task.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import type { SvelteMcp } from '../../index.js';
22
import * as v from 'valibot';
33
import { format_sections_list } from '../../utils.js';
4+
import { icons } from '../../icons/index.js';
45

56
/**
67
* Function that actually generates the prompt string. You can use this in the MCP server handler to generate the prompt, it can accept arguments
@@ -64,6 +65,7 @@ export function setup_svelte_task(server: SvelteMcp) {
6465
};
6566
},
6667
},
68+
icons,
6769
},
6870
async ({ task }) => {
6971
const available_docs = await format_sections_list();

packages/mcp-server/src/mcp/handlers/resources/doc-section.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import type { SvelteMcp } from '../../index.js';
22
import { get_sections, fetch_with_timeout } from '../../utils.js';
3+
import { icons } from '../../icons/index.js';
34

45
export async function list_sections(server: SvelteMcp) {
56
const sections = await get_sections();
@@ -42,6 +43,7 @@ export async function list_sections(server: SvelteMcp) {
4243
},
4344
},
4445
uri: 'svelte://{/slug*}.md',
46+
icons,
4547
},
4648
async (uri, { slug }) => {
4749
const section = sections.find((section) => {

packages/mcp-server/src/mcp/handlers/tools/get-documentation.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import type { SvelteMcp } from '../../index.js';
22
import * as v from 'valibot';
33
import { get_sections, fetch_with_timeout, format_sections_list } from '../../utils.js';
44
import { SECTIONS_LIST_INTRO, SECTIONS_LIST_OUTRO } from './prompts.js';
5+
import { icons } from '../../icons/index.js';
56

67
export function get_documentation(server: SvelteMcp) {
78
server.tool(
@@ -17,6 +18,7 @@ export function get_documentation(server: SvelteMcp) {
1718
),
1819
),
1920
}),
21+
icons,
2022
},
2123
async ({ section }) => {
2224
let sections: string[];

packages/mcp-server/src/mcp/handlers/tools/list-sections.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,15 @@
11
import type { SvelteMcp } from '../../index.js';
22
import { format_sections_list } from '../../utils.js';
33
import { SECTIONS_LIST_INTRO, SECTIONS_LIST_OUTRO } from './prompts.js';
4+
import { icons } from '../../icons/index.js';
45

56
export function list_sections(server: SvelteMcp) {
67
server.tool(
78
{
89
name: 'list-sections',
910
description:
1011
'Lists all available Svelte 5 and SvelteKit documentation sections in a structured format. Each section includes a "use_cases" field that describes WHEN this documentation would be useful. You should carefully analyze the use_cases field to determine which sections are relevant for the user\'s query. The use_cases contain comma-separated keywords describing project types (e.g., "e-commerce", "blog"), features (e.g., "authentication", "forms"), components (e.g., "slider", "modal"), development stages (e.g., "deployment", "testing"), or "always" for fundamental concepts. Match these use_cases against the user\'s intent - for example, if building an e-commerce site, fetch sections with use_cases containing "e-commerce", "product listings", "shopping cart", etc. If building a slider, look for "slider", "carousel", "animation", etc. Returns sections as "* title: [section_title], use_cases: [use_cases], path: [file_path]". Always run list-sections FIRST for any Svelte query, then analyze ALL use_cases to identify relevant sections, and finally use get_documentation to fetch ALL relevant sections at once.',
12+
icons,
1113
},
1214
async () => {
1315
const formatted_sections = await format_sections_list();

packages/mcp-server/src/mcp/handlers/tools/playground-link.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import type { SvelteMcp } from '../../index.js';
22
import * as v from 'valibot';
3+
import { icons } from '../../icons/index.js';
34

45
async function compress_and_encode_text(input: string) {
56
const reader = new Blob([input]).stream().pipeThrough(new CompressionStream('gzip')).getReader();
@@ -54,6 +55,7 @@ export function playground_link(server: SvelteMcp) {
5455
outputSchema: v.object({
5556
url: v.string(),
5657
}),
58+
icons,
5759
},
5860
async ({ files, name, tailwind }) => {
5961
const playground_base = new URL('https://svelte.dev/playground');

packages/mcp-server/src/mcp/handlers/tools/svelte-autofixer.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import * as v from 'valibot';
44
import { add_compile_issues } from '../../autofixers/add-compile-issues.js';
55
import { add_eslint_issues } from '../../autofixers/add-eslint-issues.js';
66
import { add_autofixers_issues } from '../../autofixers/add-autofixers-issues.js';
7+
import { icons } from '../../icons/index.js';
78

89
export function svelte_autofixer(server: SvelteMcp) {
910
server.tool(
@@ -38,6 +39,7 @@ export function svelte_autofixer(server: SvelteMcp) {
3839
readOnlyHint: true,
3940
openWorldHint: false,
4041
},
42+
icons,
4143
},
4244
async ({
4345
code,
Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
export const icons = [
2+
{
3+
src: 'https://mcp.svelte.dev/logo.svg',
4+
mimeType: 'image/svg+xml',
5+
},
6+
{
7+
src: 'https://mcp.svelte.dev/logo.png',
8+
mimeType: 'image/png',
9+
},
10+
{
11+
src: '',
12+
mimeType: 'image/png',
13+
sizes: ['96x96'],
14+
},
15+
];

packages/mcp-server/src/mcp/index.ts

Lines changed: 2 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -3,23 +3,15 @@ import { McpServer } from 'tmcp';
33
import { setup_prompts, setup_resources, setup_tools } from './handlers/index.js';
44
import type { LibSQLDatabase } from 'drizzle-orm/libsql';
55
import type { Schema } from '@sveltejs/mcp-schema';
6+
import { icons } from './icons/index.js';
67

78
export const server = new McpServer(
89
{
910
name: 'Svelte MCP',
1011
version: '0.0.1',
1112
description: 'The official Svelte MCP server implementation',
1213
websiteUrl: 'https://mcp.svelte.dev',
13-
icons: [
14-
{
15-
src: 'https://mcp.svelte.dev/logo.svg',
16-
mimeType: 'image/svg+xml',
17-
},
18-
{
19-
src: 'https://mcp.svelte.dev/logo.png',
20-
mimeType: 'image/png',
21-
},
22-
],
14+
icons,
2315
},
2416
{
2517
adapter: new ValibotJsonSchemaAdapter(),

0 commit comments

Comments
 (0)