From 2b368a3bf9bddf06cb4145c4749b1379aa66199a Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 11 Oct 2025 19:42:52 +0530 Subject: [PATCH 01/15] Implement SEP-973 --- mcp/protocol.go | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/mcp/protocol.go b/mcp/protocol.go index 1312dfbd..9514ffbc 100644 --- a/mcp/protocol.go +++ b/mcp/protocol.go @@ -658,6 +658,21 @@ type ProgressNotificationParams struct { func (*ProgressNotificationParams) isParams() {} +// Icon provides visual identifiers for their resources, tools, prompts, and implementations +// See [/specification/draft/basic/index#icons] for notes on icons + +// TODO(iamsurajbobade): update specification url from draft. +type Icon struct { + // Source is A URI pointing to the icon resource (required). This can be: + // - An HTTP/HTTPS URL pointing to an image file + // - A data URI with base64-encoded image data + Source string `json:"src"` + // Optional MIME type if the server's type is missing or generic + MIMEType string `json:"mimeType,omitempty"` + // Optional size specification (e.g., ["48x48"], ["any"] for scalable formats like SVG, or ["48x48", "96x96"] for multiple sizes) + Sizes []string `json:"sizes,omitempty"` +} + // A prompt or prompt template that the server offers. type Prompt struct { // See [specification/2025-06-18/basic/index#general-fields] for notes on _meta @@ -673,6 +688,8 @@ type Prompt struct { // Intended for UI and end-user contexts — optimized to be human-readable and // easily understood, even by those unfamiliar with domain-specific terminology. Title string `json:"title,omitempty"` + // Icons for the prompt, if any. + Icons []Icon `json:"icons,omitempty"` } // Describes an argument that a prompt can accept. @@ -782,6 +799,8 @@ type Resource struct { Title string `json:"title,omitempty"` // The URI of this resource. URI string `json:"uri"` + // Icons for the resource, if any. + Icons []Icon `json:"icons,omitempty"` } type ResourceListChangedParams struct { @@ -948,6 +967,8 @@ type Tool struct { // If not provided, Annotations.Title should be used for display if present, // otherwise Name. Title string `json:"title,omitempty"` + // Icons for the tool, if any. + Icons []Icon `json:"icons,omitempty"` } // Additional properties describing a Tool to clients. @@ -1090,6 +1111,10 @@ type Implementation struct { // easily understood, even by those unfamiliar with domain-specific terminology. Title string `json:"title,omitempty"` Version string `json:"version"` + // WebsiteURL for the server, if any. + WebsiteURL string `json:"websiteUrl,omitempty"` + // Icons for the Server, if any. + Icons []Icon `json:"icons,omitempty"` } // Present if the server supports argument autocompletion suggestions. From 085099e78950fdfb1a04fbafa2d27d20f0cf6929 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 11 Oct 2025 20:37:16 +0530 Subject: [PATCH 02/15] update tests --- mcp/mcp_test.go | 9 ++++++++- mcp/streamable_test.go | 2 +- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/mcp/mcp_test.go b/mcp/mcp_test.go index 0e1ee1f2..13582896 100644 --- a/mcp/mcp_test.go +++ b/mcp/mcp_test.go @@ -1646,7 +1646,14 @@ func TestNoDistributedDeadlock(t *testing.T) { } } -var testImpl = &Implementation{Name: "test", Version: "v1.0.0"} +var testImpl = &Implementation{ + Name: "test", Version: "v1.0.0", Title: "test-mcp-server", + WebsiteURL: "http://example.com", + Icons: []Icon{{ + Source: "", + MIMEType: "image/png", + Sizes: []string{"48x48", "96x96"}, + }}} // This test checks that when we use pointer types for tools, we get the same // schema as when using the non-pointer types. It is too much of a footgun for diff --git a/mcp/streamable_test.go b/mcp/streamable_test.go index 6ccaebf7..540a059c 100644 --- a/mcp/streamable_test.go +++ b/mcp/streamable_test.go @@ -1192,7 +1192,7 @@ func TestStreamableStateless(t *testing.T) { Tools: &ToolCapabilities{ListChanged: true}, }, ProtocolVersion: latestProtocolVersion, - ServerInfo: &Implementation{Name: "test", Version: "v1.0.0"}, + ServerInfo: testImpl, }, nil) // This version of sayHi expects // that request from our client). From 24947b35b2f8aba0b46be09964dc8249bab78930 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sun, 19 Oct 2025 15:59:13 +0530 Subject: [PATCH 03/15] revert tests --- mcp/mcp_test.go | 9 +-------- mcp/streamable_test.go | 2 +- 2 files changed, 2 insertions(+), 9 deletions(-) diff --git a/mcp/mcp_test.go b/mcp/mcp_test.go index c9ec98d4..a78c1525 100644 --- a/mcp/mcp_test.go +++ b/mcp/mcp_test.go @@ -1637,14 +1637,7 @@ func TestNoDistributedDeadlock(t *testing.T) { } } -var testImpl = &Implementation{ - Name: "test", Version: "v1.0.0", Title: "test-mcp-server", - WebsiteURL: "http://example.com", - Icons: []Icon{{ - Source: "", - MIMEType: "image/png", - Sizes: []string{"48x48", "96x96"}, - }}} +var testImpl = &Implementation{Name: "test", Version: "v1.0.0"} // This test checks that when we use pointer types for tools, we get the same // schema as when using the non-pointer types. It is too much of a footgun for diff --git a/mcp/streamable_test.go b/mcp/streamable_test.go index 8688df01..a0893689 100644 --- a/mcp/streamable_test.go +++ b/mcp/streamable_test.go @@ -1288,7 +1288,7 @@ func TestStreamableStateless(t *testing.T) { Tools: &ToolCapabilities{ListChanged: true}, }, ProtocolVersion: latestProtocolVersion, - ServerInfo: testImpl, + ServerInfo: &Implementation{Name: "test", Version: "v1.0.0"}, }, nil) // This version of sayHi expects // that request from our client). From ec8b14aaefbef0b5f1e465a42f2ee9b48f1d1f00 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sun, 19 Oct 2025 16:41:13 +0530 Subject: [PATCH 04/15] add draft protocol version --- mcp/shared.go | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/mcp/shared.go b/mcp/shared.go index e90bcbd8..4fc88b87 100644 --- a/mcp/shared.go +++ b/mcp/shared.go @@ -33,8 +33,10 @@ const ( // // It is the version that the client sends in the initialization request, and // the default version used by the server. - latestProtocolVersion = protocolVersion20250618 - protocolVersion20250618 = "2025-06-18" + latestProtocolVersion = protocolVersion20250618 + + protocolVersionDraft = "draft" // draft protocol version with experimental features for testing + protocolVersion20250618 = "2025-06-18" // latest stable version protocolVersion20250326 = "2025-03-26" protocolVersion20241105 = "2024-11-05" ) @@ -48,6 +50,12 @@ var supportedProtocolVersions = []string{ // negotiatedVersion returns the effective protocol version to use, given a // client version. func negotiatedVersion(clientVersion string) string { + // If client sends protocol version draft, enable draft features. + if clientVersion == protocolVersionDraft { + log.Printf("Using draft protocol version features") + return protocolVersionDraft + } + // In general, prefer to use the clientVersion, but if we don't support the // client's version, use the latest version. // From dc32c434831793eeecf774c7bae98b4d6090abb9 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sun, 19 Oct 2025 16:41:51 +0530 Subject: [PATCH 05/15] Add conformance test for Icon and websiteUrl metadata --- mcp/conformance_test.go | 33 +++- .../conformance/server/version-draft.txtar | 149 ++++++++++++++++++ 2 files changed, 181 insertions(+), 1 deletion(-) create mode 100644 mcp/testdata/conformance/server/version-draft.txtar diff --git a/mcp/conformance_test.go b/mcp/conformance_test.go index 3393efcb..6b2c19bf 100644 --- a/mcp/conformance_test.go +++ b/mcp/conformance_test.go @@ -135,12 +135,23 @@ func incTool(_ context.Context, _ *CallToolRequest, args incInput) (*CallToolRes return nil, incOutput{args.X + 1}, nil } +var iconObj = Icon{Source: "", + MIMEType: "image/png", Sizes: []string{"48x48", "96x96"}} + // runServerTest runs the server conformance test. // It must be executed in a synctest bubble. func runServerTest(t *testing.T, test *conformanceTest) { ctx := t.Context() // Construct the server based on features listed in the test. - s := NewServer(&Implementation{Name: "testServer", Version: "v1.0.0"}, nil) + impl := &Implementation{Name: "testServer", Version: "v1.0.0"} + + // TODO(IAmSurajBobade): Remove this hack once we have a client protocol specific handling. + if test.name == "version-draft.txtar" { + impl.Icons = []Icon{iconObj} + impl.WebsiteURL = "https://modelcontextprotocol.io" + } + + s := NewServer(impl, nil) for _, tn := range test.tools { switch tn { case "greet": @@ -148,6 +159,12 @@ func runServerTest(t *testing.T, test *conformanceTest) { Name: "greet", Description: "say hi", }, sayHi) + case "greetWithIcon": + AddTool(s, &Tool{ + Name: "greetWithIcon", + Description: "say hi", + Icons: []Icon{iconObj}, + }, sayHi) case "structured": AddTool(s, &Tool{Name: "structured"}, structuredTool) case "tomorrow": @@ -167,6 +184,13 @@ func runServerTest(t *testing.T, test *conformanceTest) { switch pn { case "code_review": s.AddPrompt(codeReviewPrompt, codReviewPromptHandler) + case "code_reviewWithIcon": + s.AddPrompt(&Prompt{ + Name: "code_review", + Description: "do a code review", + Arguments: []*PromptArgument{{Name: "Code", Required: true}}, + Icons: []Icon{iconObj}, + }, codReviewPromptHandler) default: t.Fatalf("unknown prompt %q", pn) } @@ -177,6 +201,13 @@ func runServerTest(t *testing.T, test *conformanceTest) { s.AddResource(resource1, readHandler) case "info": s.AddResource(resource3, handleEmbeddedResource) + case "infoWithIcon": + s.AddResource(&Resource{ + Name: "info", + MIMEType: "text/plain", + URI: "embedded:info", + Icons: []Icon{iconObj}, + }, handleEmbeddedResource) default: t.Fatalf("unknown resource %q", rn) } diff --git a/mcp/testdata/conformance/server/version-draft.txtar b/mcp/testdata/conformance/server/version-draft.txtar new file mode 100644 index 00000000..408f5502 --- /dev/null +++ b/mcp/testdata/conformance/server/version-draft.txtar @@ -0,0 +1,149 @@ +Check that when the client sends an arbitrary later version, the server +response with its latest supported version. + +-- tools -- +greetWithIcon + +-- prompts -- +code_reviewWithIcon + +-- resources -- +infoWithIcon + +-- client -- +{ + "jsonrpc": "2.0", + "id": 1, + "method": "initialize", + "params": { + "protocolVersion": "draft", + "capabilities": {}, + "clientInfo": { "name": "ExampleClient", "version": "1.0.0" } + } +} +{ "jsonrpc": "2.0", "method": "notifications/initialized" } +{ "jsonrpc": "2.0", "id": 2, "method": "tools/list" } +{ "jsonrpc": "2.0", "id": 3, "method": "resources/list" } +{ "jsonrpc": "2.0", "id": 4, "method": "prompts/list" } + +-- server -- +{ + "jsonrpc": "2.0", + "id": 1, + "result": { + "capabilities": { + "logging": {}, + "prompts": { + "listChanged": true + }, + "resources": { + "listChanged": true + }, + "tools": { + "listChanged": true + } + }, + "protocolVersion": "draft", + "serverInfo": { + "name": "testServer", + "version": "v1.0.0", + "websiteUrl": "https://modelcontextprotocol.io", + "icons": [ + { + "src": "", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + } +} +{ + "jsonrpc": "2.0", + "id": 2, + "result": { + "tools": [ + { + "description": "say hi", + "inputSchema": { + "type": "object", + "required": [ + "Name" + ], + "properties": { + "Name": { + "type": "string" + } + }, + "additionalProperties": false + }, + "name": "greetWithIcon", + "icons": [ + { + "src": "", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} +{ + "jsonrpc": "2.0", + "id": 3, + "result": { + "resources": [ + { + "mimeType": "text/plain", + "name": "info", + "uri": "embedded:info", + "icons": [ + { + "src": "", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} +{ + "jsonrpc": "2.0", + "id": 4, + "result": { + "prompts": [ + { + "arguments": [ + { + "name": "Code", + "required": true + } + ], + "description": "do a code review", + "name": "code_review", + "icons": [ + { + "src": "", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} + From 1fb33435b309af909efa28548599573de5af4548 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sun, 19 Oct 2025 17:01:30 +0530 Subject: [PATCH 06/15] update test description --- mcp/testdata/conformance/server/version-draft.txtar | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/mcp/testdata/conformance/server/version-draft.txtar b/mcp/testdata/conformance/server/version-draft.txtar index 408f5502..f218fa57 100644 --- a/mcp/testdata/conformance/server/version-draft.txtar +++ b/mcp/testdata/conformance/server/version-draft.txtar @@ -1,5 +1,13 @@ -Check that when the client sends an arbitrary later version, the server -response with its latest supported version. +Check behavior of server with Icons and websiteUrl metadata added as part of SEP-973 specification. + +check modelcontextprotocol/go-sdk/issues/552 for more details. + +Checks following: +- If client sends protocolVersion as "draft", server responds with same +- Test setting websiteUrl, icons for mcp.Implementation +- Test setting icons for mcp.Prompt +- Test setting icons for mcp.Tool +- Test setting icons for mcp.Resource -- tools -- greetWithIcon From 5092e10f95609ac7e994dfada8c5e3887e652f35 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Wed, 29 Oct 2025 12:20:26 +0530 Subject: [PATCH 07/15] address review comments --- mcp/conformance_test.go | 6 +- mcp/shared.go | 7 - .../spec-sep-973-additional-metadata.txtar | 157 ++++++++++++++++++ .../conformance/server/version-draft.txtar | 157 ------------------ 4 files changed, 159 insertions(+), 168 deletions(-) create mode 100644 mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar delete mode 100644 mcp/testdata/conformance/server/version-draft.txtar diff --git a/mcp/conformance_test.go b/mcp/conformance_test.go index 6b2c19bf..be63b52d 100644 --- a/mcp/conformance_test.go +++ b/mcp/conformance_test.go @@ -135,8 +135,7 @@ func incTool(_ context.Context, _ *CallToolRequest, args incInput) (*CallToolRes return nil, incOutput{args.X + 1}, nil } -var iconObj = Icon{Source: "", - MIMEType: "image/png", Sizes: []string{"48x48", "96x96"}} +var iconObj = Icon{Source: "foobar", MIMEType: "image/png", Sizes: []string{"48x48", "96x96"}} // runServerTest runs the server conformance test. // It must be executed in a synctest bubble. @@ -145,8 +144,7 @@ func runServerTest(t *testing.T, test *conformanceTest) { // Construct the server based on features listed in the test. impl := &Implementation{Name: "testServer", Version: "v1.0.0"} - // TODO(IAmSurajBobade): Remove this hack once we have a client protocol specific handling. - if test.name == "version-draft.txtar" { + if test.name == "spec-sep-973-additional-metadata.txtar" { impl.Icons = []Icon{iconObj} impl.WebsiteURL = "https://modelcontextprotocol.io" } diff --git a/mcp/shared.go b/mcp/shared.go index 4fc88b87..30d0bf51 100644 --- a/mcp/shared.go +++ b/mcp/shared.go @@ -35,7 +35,6 @@ const ( // the default version used by the server. latestProtocolVersion = protocolVersion20250618 - protocolVersionDraft = "draft" // draft protocol version with experimental features for testing protocolVersion20250618 = "2025-06-18" // latest stable version protocolVersion20250326 = "2025-03-26" protocolVersion20241105 = "2024-11-05" @@ -50,12 +49,6 @@ var supportedProtocolVersions = []string{ // negotiatedVersion returns the effective protocol version to use, given a // client version. func negotiatedVersion(clientVersion string) string { - // If client sends protocol version draft, enable draft features. - if clientVersion == protocolVersionDraft { - log.Printf("Using draft protocol version features") - return protocolVersionDraft - } - // In general, prefer to use the clientVersion, but if we don't support the // client's version, use the latest version. // diff --git a/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar new file mode 100644 index 00000000..283beb18 --- /dev/null +++ b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar @@ -0,0 +1,157 @@ +Check behavior of server with Icons and websiteUrl metadata added as part of SEP-973 specification. + +check modelcontextprotocol/go-sdk/issues/552 for more details. + +Checks following: +- If client sends protocolVersion as "draft", server responds with same +- Test setting websiteUrl, icons for mcp.Implementation +- Test setting icons for mcp.Prompt +- Test setting icons for mcp.Tool +- Test setting icons for mcp.Resource + +-- tools -- +greetWithIcon + +-- prompts -- +code_reviewWithIcon + +-- resources -- +infoWithIcon + +-- client -- +{ + "jsonrpc": "2.0", + "id": 1, + "method": "initialize", + "params": { + "protocolVersion": "2025-06-18", + "capabilities": {}, + "clientInfo": { "name": "ExampleClient", "version": "1.0.0" } + } +} +{ "jsonrpc": "2.0", "method": "notifications/initialized" } +{ "jsonrpc": "2.0", "id": 2, "method": "tools/list" } +{ "jsonrpc": "2.0", "id": 3, "method": "resources/list" } +{ "jsonrpc": "2.0", "id": 4, "method": "prompts/list" } + +-- server -- +{ + "jsonrpc": "2.0", + "id": 1, + "result": { + "capabilities": { + "logging": {}, + "prompts": { + "listChanged": true + }, + "resources": { + "listChanged": true + }, + "tools": { + "listChanged": true + } + }, + "protocolVersion": "2025-06-18", + "serverInfo": { + "name": "testServer", + "version": "v1.0.0", + "websiteUrl": "https://modelcontextprotocol.io", + "icons": [ + { + "src": "foobar", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + } +} +{ + "jsonrpc": "2.0", + "id": 2, + "result": { + "tools": [ + { + "description": "say hi", + "inputSchema": { + "type": "object", + "required": [ + "Name" + ], + "properties": { + "Name": { + "type": "string" + } + }, + "additionalProperties": false + }, + "name": "greetWithIcon", + "icons": [ + { + "src": "foobar", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} +{ + "jsonrpc": "2.0", + "id": 3, + "result": { + "resources": [ + { + "mimeType": "text/plain", + "name": "info", + "uri": "embedded:info", + "icons": [ + { + "src": "foobar", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} +{ + "jsonrpc": "2.0", + "id": 4, + "result": { + "prompts": [ + { + "arguments": [ + { + "name": "Code", + "required": true + } + ], + "description": "do a code review", + "name": "code_review", + "icons": [ + { + "src": "foobar", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ] + } + ] + } + ] + } +} + diff --git a/mcp/testdata/conformance/server/version-draft.txtar b/mcp/testdata/conformance/server/version-draft.txtar deleted file mode 100644 index f218fa57..00000000 --- a/mcp/testdata/conformance/server/version-draft.txtar +++ /dev/null @@ -1,157 +0,0 @@ -Check behavior of server with Icons and websiteUrl metadata added as part of SEP-973 specification. - -check modelcontextprotocol/go-sdk/issues/552 for more details. - -Checks following: -- If client sends protocolVersion as "draft", server responds with same -- Test setting websiteUrl, icons for mcp.Implementation -- Test setting icons for mcp.Prompt -- Test setting icons for mcp.Tool -- Test setting icons for mcp.Resource - --- tools -- -greetWithIcon - --- prompts -- -code_reviewWithIcon - --- resources -- -infoWithIcon - --- client -- -{ - "jsonrpc": "2.0", - "id": 1, - "method": "initialize", - "params": { - "protocolVersion": "draft", - "capabilities": {}, - "clientInfo": { "name": "ExampleClient", "version": "1.0.0" } - } -} -{ "jsonrpc": "2.0", "method": "notifications/initialized" } -{ "jsonrpc": "2.0", "id": 2, "method": "tools/list" } -{ "jsonrpc": "2.0", "id": 3, "method": "resources/list" } -{ "jsonrpc": "2.0", "id": 4, "method": "prompts/list" } - --- server -- -{ - "jsonrpc": "2.0", - "id": 1, - "result": { - "capabilities": { - "logging": {}, - "prompts": { - "listChanged": true - }, - "resources": { - "listChanged": true - }, - "tools": { - "listChanged": true - } - }, - "protocolVersion": "draft", - "serverInfo": { - "name": "testServer", - "version": "v1.0.0", - "websiteUrl": "https://modelcontextprotocol.io", - "icons": [ - { - "src": "", - "mimeType": "image/png", - "sizes": [ - "48x48", - "96x96" - ] - } - ] - } - } -} -{ - "jsonrpc": "2.0", - "id": 2, - "result": { - "tools": [ - { - "description": "say hi", - "inputSchema": { - "type": "object", - "required": [ - "Name" - ], - "properties": { - "Name": { - "type": "string" - } - }, - "additionalProperties": false - }, - "name": "greetWithIcon", - "icons": [ - { - "src": "", - "mimeType": "image/png", - "sizes": [ - "48x48", - "96x96" - ] - } - ] - } - ] - } -} -{ - "jsonrpc": "2.0", - "id": 3, - "result": { - "resources": [ - { - "mimeType": "text/plain", - "name": "info", - "uri": "embedded:info", - "icons": [ - { - "src": "", - "mimeType": "image/png", - "sizes": [ - "48x48", - "96x96" - ] - } - ] - } - ] - } -} -{ - "jsonrpc": "2.0", - "id": 4, - "result": { - "prompts": [ - { - "arguments": [ - { - "name": "Code", - "required": true - } - ], - "description": "do a code review", - "name": "code_review", - "icons": [ - { - "src": "", - "mimeType": "image/png", - "sizes": [ - "48x48", - "96x96" - ] - } - ] - } - ] - } -} - From 47f6939fe3e00001a415e61739dc4ee11618ad37 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Wed, 29 Oct 2025 15:56:32 +0530 Subject: [PATCH 08/15] add example --- examples/server/everything/main.go | 41 ++++++++++++++++++++++++----- examples/server/everything/mcp.png | Bin 0 -> 2586 bytes 2 files changed, 34 insertions(+), 7 deletions(-) create mode 100644 examples/server/everything/mcp.png diff --git a/examples/server/everything/main.go b/examples/server/everything/main.go index 0b81919f..708cc2de 100644 --- a/examples/server/everything/main.go +++ b/examples/server/everything/main.go @@ -7,6 +7,7 @@ package main import ( "context" + "encoding/base64" "flag" "fmt" "log" @@ -50,19 +51,27 @@ func main() { CompletionHandler: complete, // support completions by setting this handler } - server := mcp.NewServer(&mcp.Implementation{Name: "everything"}, opts) + // Optionally add an icon to the server implementation. + icons, err := iconToBase64DataURL("./mcp.png") + if err != nil { + log.Fatalf("failed to read icon: %v", err) + } + + server := mcp.NewServer(&mcp.Implementation{Name: "everything", WebsiteURL: "https://example.com", Icons: icons}, opts) // Add tools that exercise different features of the protocol. mcp.AddTool(server, &mcp.Tool{Name: "greet", Description: "say hi"}, contentTool) - mcp.AddTool(server, &mcp.Tool{Name: "greet (structured)"}, structuredTool) // returns structured output - mcp.AddTool(server, &mcp.Tool{Name: "ping"}, pingingTool) // performs a ping - mcp.AddTool(server, &mcp.Tool{Name: "log"}, loggingTool) // performs a log - mcp.AddTool(server, &mcp.Tool{Name: "sample"}, samplingTool) // performs sampling - mcp.AddTool(server, &mcp.Tool{Name: "elicit"}, elicitingTool) // performs elicitation - mcp.AddTool(server, &mcp.Tool{Name: "roots"}, rootsTool) // lists roots + mcp.AddTool(server, &mcp.Tool{Name: "greet (structured)"}, structuredTool) // returns structured output + mcp.AddTool(server, &mcp.Tool{Name: "greet (with Icons)", Icons: icons}, structuredTool) // tool with icons + mcp.AddTool(server, &mcp.Tool{Name: "ping"}, pingingTool) // performs a ping + mcp.AddTool(server, &mcp.Tool{Name: "log"}, loggingTool) // performs a log + mcp.AddTool(server, &mcp.Tool{Name: "sample"}, samplingTool) // performs sampling + mcp.AddTool(server, &mcp.Tool{Name: "elicit"}, elicitingTool) // performs elicitation + mcp.AddTool(server, &mcp.Tool{Name: "roots"}, rootsTool) // lists roots // Add a basic prompt. server.AddPrompt(&mcp.Prompt{Name: "greet"}, prompt) + server.AddPrompt(&mcp.Prompt{Name: "greet (with Icons)", Icons: icons}, prompt) // greet prompt with icons // Add an embedded resource. server.AddResource(&mcp.Resource{ @@ -70,6 +79,12 @@ func main() { MIMEType: "text/plain", URI: "embedded:info", }, embeddedResource) + server.AddResource(&mcp.Resource{ // text resource with icons + Name: "info (with Icons)", + MIMEType: "text/plain", + URI: "embedded:info", + Icons: icons, + }, embeddedResource) // Serve over stdio, or streamable HTTP if -http is set. if *httpAddr != "" { @@ -222,3 +237,15 @@ func complete(ctx context.Context, req *mcp.CompleteRequest) (*mcp.CompleteResul }, }, nil } + +func iconToBase64DataURL(path string) ([]mcp.Icon, error) { + data, err := os.ReadFile(path) + if err != nil { + return nil, err + } + return []mcp.Icon{{ + Source: "data:image/png;base64," + base64.StdEncoding.EncodeToString(data), + MIMEType: "image/png", + Sizes: []string{"48x48"}, + }}, nil +} diff --git a/examples/server/everything/mcp.png b/examples/server/everything/mcp.png new file mode 100644 index 0000000000000000000000000000000000000000..8e08571d326bba9110ea3ced50e6769d0e100b21 GIT binary patch literal 2586 zcmV+#3gz{QP)00001b5ch_0Itp) z=>Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91K%fHv1ONa40RR91KmY&$07g+lumAuFgh@m}RA>dgnsuyHI}pZkcXxMp z3xS})Ew~fh-Q5BSF2Nx<1PKr{xCR2jg1fuB^WHD-kTP%1F5BI^%l+eK&e^ZinRYt; zIz8e33H{%hGiSz(851Q+lyLw3#f}|&+O%mGFJ25v&*6fi#ha2%n>LjxQ^uI?INsK3#6(-0IT)%!jXU?3;xMt0ooj7sA*!Stvr$dJh)v8rX!6r?bq>Qt945kBDuU=(z zrCBWB`Sa&{vt!)2adIjWCrhk5wy>;xo@7ArG zY%*oalt1I$zXr#F^XJcJ%$QM{LY7>;dbR5B+qZ8&e*BOxpFe+|GG$8BkevJT=T8an zDF?xUvuDqyPoLfxfe<4`jJ0dms=gO3T9h|$-a>^6u{p?}+O=zE+9peu%-?hrRtHX< zI+Z3(8e;)M%$PAZY}g=8L?$*2CVu?*4<0-)fO+%gnV+IXi+1hWHS?2=5GDu6EvZta zGTkFU6E<(&tU8`BVS>aULoN)4bz**H(cidn!~A3;1n$7GW5<#wPj31~h#NQVwr$%~ zucR7@L9SS_BAL{{#*ZIw0K^_EXY1Cjd-m*sEPed=(ZIY30XlHx$dRN;lbVhZ2#lRO zcdBkij~*>C$fZk{e){yOgq1H}USXU^W`nc_(O{k>Ude$taPZ(ka<=R>L4pK~nNmG` z_;86qE?c(jix)2>?7)Eokt0V|F!hKhZQ8VX^5ltZXk@<~*tc(A!h{KBABhqr+P{Cl z(dY<6h76GyWFqI)t5*{C@Zm%Ewyj#3#~e9wknUu2NA}NwUAuM>Dn=%Vvs9WuKfD-YY8I}6)jp+!JJ2Szz7i{xPKzuxla*(IIv^Kj(G9nNxdxMqeqWQ zvyl7r=_4`7)v8r{`}VDb-Me>>4MD;3=g&WR@?;cVzI^%Q$&(8fEGSp5oWc|+P(Z1& zbzsYuEpg(+k?Pn%j~_oS&0-Vn(W8gNAVb;|-G^*rtAsJ$EO(DNXUUQ!BsU2jF=B-5 z)XD+&&RDTxNiC^Ur)H3(A~-^qE?p!Bxo+LMA3l7Lu$wn;DzXjJo;!E0#3-4WXLB@~ z$sdm&KQh+njvJ$lWTS^Raixw@ueEA|_ zknPNwGfS99vJEdH>pL(OwA4_&dUeyB1Yq3r>cG;aOX#E}euyc-F03Co#y=r%zY8pP`W$8Kv2?XE$Ty!2u3|WCqz~?%cV_9#S2vvtGS= z5`)~SQzurlgt^H!+00G0i57XbL={C1Nx2D_V#SJ??jTgA7wy38*|X(5(k5TNe7A4k zHVqKS>vihXG3p$F!=XBD1x&V!7cZ_3#|(sw=bk-#8hifdz>FC))I1W1ckkXcXoPp~ z-cbTk*7xew%d9E~95C4~S+b09?<*IbM`#TB+h{CRs+7u@P%cFq-DI19?%lh$QQ`<}v5Fy+<6E_ArC`mQH+Nf) zZU1w?qv_7eWy_XHjjFa|Nw(E;L?N@S$Tn7DtLzIIf6UIMjI?z?9{`-5_%|cdIjRh;O3Y;2i<7tpb`_3ztp`HgMoTqsOOU z>i~)v71>jX5+x+#l|5k=8AgJU$wDfi85DP2ap=&YQk75H4+k(;P69S@!d$Up1ui;5*> zjq?aK!dqnr`-Uos!jLCUoTy;-kAXOVb`F`Of9ld6O{WOSk(^?LvIC~5N}?P9r%s)! zV17M@2kp@ao0lwWs_m3xIr78FpsN&+ixersJ%GBjL>os_mdfeVr;TPpkLZ-cq;ISl1_1vbdPfW-i;QIqOk_-llhl>|4_Gy_V zAhZrpiVWcU1548#SO^_pw^Ef3O2w;It@6u0pWqNWK+Qr%4mryG{ee&Wd;&t{0IXBq zBXKq8E{Xay&IaJ&GEfd%z%qvcs#K;qhj;fEPUhFJ2%&O-aF-RXjT<+@Qo~}Xx3-~j zfL&Iev!MQ5Cg6rQI|6@i6K8>8C^B=YBP|QXIu6w$c6{bSO_8FYXVTC*fEymarE`Cc z7T6@yD&!8(hnr3CsSZcVfQ^<7|Aqtfr}!NAMe5p0+T@q~mmI(*7_~X Date: Wed, 29 Oct 2025 15:56:51 +0530 Subject: [PATCH 09/15] update test url --- mcp/conformance_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mcp/conformance_test.go b/mcp/conformance_test.go index be63b52d..17a58eaf 100644 --- a/mcp/conformance_test.go +++ b/mcp/conformance_test.go @@ -146,7 +146,7 @@ func runServerTest(t *testing.T, test *conformanceTest) { if test.name == "spec-sep-973-additional-metadata.txtar" { impl.Icons = []Icon{iconObj} - impl.WebsiteURL = "https://modelcontextprotocol.io" + impl.WebsiteURL = "https://github.com/modelcontextprotocol/go-sdk" } s := NewServer(impl, nil) From 80ef38bc02f4ea77e6913fe102d429a88d969bb1 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Wed, 29 Oct 2025 15:57:17 +0530 Subject: [PATCH 10/15] clean diff --- mcp/shared.go | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/mcp/shared.go b/mcp/shared.go index 30d0bf51..e90bcbd8 100644 --- a/mcp/shared.go +++ b/mcp/shared.go @@ -33,9 +33,8 @@ const ( // // It is the version that the client sends in the initialization request, and // the default version used by the server. - latestProtocolVersion = protocolVersion20250618 - - protocolVersion20250618 = "2025-06-18" // latest stable version + latestProtocolVersion = protocolVersion20250618 + protocolVersion20250618 = "2025-06-18" protocolVersion20250326 = "2025-03-26" protocolVersion20241105 = "2024-11-05" ) From dc75b26d666382fa3509070e7ccb334fc444e850 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Wed, 29 Oct 2025 16:01:26 +0530 Subject: [PATCH 11/15] update website url in test --- .../conformance/server/spec-sep-973-additional-metadata.txtar | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar index 283beb18..e69b2cbf 100644 --- a/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar +++ b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar @@ -55,7 +55,7 @@ infoWithIcon "serverInfo": { "name": "testServer", "version": "v1.0.0", - "websiteUrl": "https://modelcontextprotocol.io", + "websiteUrl": "https://github.com/modelcontextprotocol/go-sdk", "icons": [ { "src": "foobar", From e74e5c9b7c438520e2fc33731f71ee7eaef3e3bf Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 8 Nov 2025 12:08:27 +0530 Subject: [PATCH 12/15] add icons to resource template and links. add theme --- mcp/content.go | 2 ++ mcp/protocol.go | 4 ++++ 2 files changed, 6 insertions(+) diff --git a/mcp/content.go b/mcp/content.go index e53cad14..cfe67410 100644 --- a/mcp/content.go +++ b/mcp/content.go @@ -130,6 +130,8 @@ type ResourceLink struct { Size *int64 Meta Meta Annotations *Annotations + // Icons for the resource link, if any. + Icons []Icon `json:"icons,omitempty"` } func (c *ResourceLink) MarshalJSON() ([]byte, error) { diff --git a/mcp/protocol.go b/mcp/protocol.go index 9514ffbc..4d7fce08 100644 --- a/mcp/protocol.go +++ b/mcp/protocol.go @@ -671,6 +671,8 @@ type Icon struct { MIMEType string `json:"mimeType,omitempty"` // Optional size specification (e.g., ["48x48"], ["any"] for scalable formats like SVG, or ["48x48", "96x96"] for multiple sizes) Sizes []string `json:"sizes,omitempty"` + // Optional Theme of the icon, e.g., "light" or "dark" + Theme string `json:"theme,omitempty"` } // A prompt or prompt template that the server offers. @@ -841,6 +843,8 @@ type ResourceTemplate struct { // A URI template (according to RFC 6570) that can be used to construct resource // URIs. URITemplate string `json:"uriTemplate"` + // Icons for the resource template, if any. + Icons []Icon `json:"icons,omitempty"` } // The sender or recipient of messages and data in a conversation. From 33fc5569c7c6d125bf507f324c148145e9b36354 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 8 Nov 2025 20:51:46 +0530 Subject: [PATCH 13/15] update everything example --- examples/server/everything/main.go | 41 +++++++++++++++++++++++++----- 1 file changed, 34 insertions(+), 7 deletions(-) diff --git a/examples/server/everything/main.go b/examples/server/everything/main.go index 708cc2de..f255ab49 100644 --- a/examples/server/everything/main.go +++ b/examples/server/everything/main.go @@ -61,13 +61,14 @@ func main() { // Add tools that exercise different features of the protocol. mcp.AddTool(server, &mcp.Tool{Name: "greet", Description: "say hi"}, contentTool) - mcp.AddTool(server, &mcp.Tool{Name: "greet (structured)"}, structuredTool) // returns structured output - mcp.AddTool(server, &mcp.Tool{Name: "greet (with Icons)", Icons: icons}, structuredTool) // tool with icons - mcp.AddTool(server, &mcp.Tool{Name: "ping"}, pingingTool) // performs a ping - mcp.AddTool(server, &mcp.Tool{Name: "log"}, loggingTool) // performs a log - mcp.AddTool(server, &mcp.Tool{Name: "sample"}, samplingTool) // performs sampling - mcp.AddTool(server, &mcp.Tool{Name: "elicit"}, elicitingTool) // performs elicitation - mcp.AddTool(server, &mcp.Tool{Name: "roots"}, rootsTool) // lists roots + mcp.AddTool(server, &mcp.Tool{Name: "greet (structured)"}, structuredTool) // returns structured output + mcp.AddTool(server, &mcp.Tool{Name: "greet (with Icons)", Icons: icons}, structuredTool) // tool with icons + mcp.AddTool(server, &mcp.Tool{Name: "greet (content with ResourceLink)"}, resourceLinkContentTool(icons)) // tool that returns content with a resource link + mcp.AddTool(server, &mcp.Tool{Name: "ping"}, pingingTool) // performs a ping + mcp.AddTool(server, &mcp.Tool{Name: "log"}, loggingTool) // performs a log + mcp.AddTool(server, &mcp.Tool{Name: "sample"}, samplingTool) // performs sampling + mcp.AddTool(server, &mcp.Tool{Name: "elicit"}, elicitingTool) // performs elicitation + mcp.AddTool(server, &mcp.Tool{Name: "roots"}, rootsTool) // lists roots // Add a basic prompt. server.AddPrompt(&mcp.Prompt{Name: "greet"}, prompt) @@ -86,6 +87,14 @@ func main() { Icons: icons, }, embeddedResource) + // Add a resource template. + server.AddResourceTemplate(&mcp.ResourceTemplate{ + Name: "Resource template (with Icon)", + MIMEType: "text/plain", + URITemplate: "http://example.com/~{resource_name}/", + Icons: icons, + }, embeddedResource) + // Serve over stdio, or streamable HTTP if -http is set. if *httpAddr != "" { handler := mcp.NewStreamableHTTPHandler(func(*http.Request) *mcp.Server { @@ -155,6 +164,23 @@ func contentTool(ctx context.Context, req *mcp.CallToolRequest, args args) (*mcp }, nil, nil } +// resourceLinkContentTool returns a ResourceLink content with icons. +func resourceLinkContentTool(icons []mcp.Icon) func(ctx context.Context, req *mcp.CallToolRequest, args args) (*mcp.CallToolResult, any, error) { + return func(ctx context.Context, req *mcp.CallToolRequest, args args) (*mcp.CallToolResult, any, error) { + return &mcp.CallToolResult{ + Content: []mcp.Content{ + &mcp.ResourceLink{ + Name: "greeting", + Title: "A friendly greeting", + MIMEType: "text/plain", + URI: "data:text/plain,Hi%20" + url.PathEscape(args.Name), + Icons: icons, + }, + }, + }, nil, nil + } +} + type result struct { Message string `json:"message" jsonschema:"the message to convey"` } @@ -247,5 +273,6 @@ func iconToBase64DataURL(path string) ([]mcp.Icon, error) { Source: "data:image/png;base64," + base64.StdEncoding.EncodeToString(data), MIMEType: "image/png", Sizes: []string{"48x48"}, + Theme: "light", // or "dark" or empty }}, nil } From b4f4f0dff9683ca1f122d4de1b48df01ac43369d Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 8 Nov 2025 20:53:53 +0530 Subject: [PATCH 14/15] support Icon marshal/unmarshal in ResourceLink --- mcp/content.go | 3 +++ mcp/content_test.go | 53 ++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 55 insertions(+), 1 deletion(-) diff --git a/mcp/content.go b/mcp/content.go index cfe67410..fb1a0d1e 100644 --- a/mcp/content.go +++ b/mcp/content.go @@ -145,6 +145,7 @@ func (c *ResourceLink) MarshalJSON() ([]byte, error) { Size: c.Size, Meta: c.Meta, Annotations: c.Annotations, + Icons: c.Icons, }) } @@ -157,6 +158,7 @@ func (c *ResourceLink) fromWire(wire *wireContent) { c.Size = wire.Size c.Meta = wire.Meta c.Annotations = wire.Annotations + c.Icons = wire.Icons } // EmbeddedResource contains embedded resources. @@ -239,6 +241,7 @@ type wireContent struct { Size *int64 `json:"size,omitempty"` Meta Meta `json:"_meta,omitempty"` Annotations *Annotations `json:"annotations,omitempty"` + Icons []Icon `json:"icons,omitempty"` } func contentsFromWire(wires []*wireContent, allow map[string]bool) ([]Content, error) { diff --git a/mcp/content_test.go b/mcp/content_test.go index 9366b0d4..ae13c307 100644 --- a/mcp/content_test.go +++ b/mcp/content_test.go @@ -121,8 +121,9 @@ func TestContent(t *testing.T) { Description: "This resource demonstrates all fields", MIMEType: "text/plain", Meta: mcp.Meta{"custom": "metadata"}, + Icons: []mcp.Icon{{Source: "foobar", MIMEType: "image/png", Sizes: []string{"48x48"}, Theme: "light"}}, }, - `{"type":"resource_link","mimeType":"text/plain","uri":"https://example.com/resource","name":"Example Resource","title":"A comprehensive example resource","description":"This resource demonstrates all fields","_meta":{"custom":"metadata"}}`, + `{"type":"resource_link","mimeType":"text/plain","uri":"https://example.com/resource","name":"Example Resource","title":"A comprehensive example resource","description":"This resource demonstrates all fields","_meta":{"custom":"metadata"},"icons":[{"src":"foobar","mimeType":"image/png","sizes":["48x48"],"theme":"light"}]}`, }, } @@ -192,3 +193,53 @@ func TestEmbeddedResource(t *testing.T) { } } } + +// TestContentUnmarshal tests that unmarshaling JSON into various Content types +// works correctly, including when the Content fields are initially nil. +func TestContentUnmarshal(t *testing.T) { + valInt64 := int64(24) + tests := []struct { + name string + json string + content mcp.Content + expectContent mcp.Content + }{ + { + name: "ResourceLink", + json: `{"type":"resource_link","mimeType":"text/plain","uri":"https://example.com/resource","name":"Example Resource","title":"A comprehensive example resource","description":"This resource demonstrates all fields","_meta":{"custom":"metadata"},"icons":[{"src":"foobar","mimeType":"image/png","sizes":["48x48"],"theme":"light"}], "size":24,"annotations":{"audience":["user","assistant"],"lastModified":"2025-01-12T15:00:58Z","priority":0.5}}`, + content: &mcp.ResourceLink{}, + expectContent: &mcp.ResourceLink{ + URI: "https://example.com/resource", + Name: "Example Resource", + Title: "A comprehensive example resource", + Description: "This resource demonstrates all fields", + MIMEType: "text/plain", + // Meta: mcp.Meta{"custom": "metadata"}, + Size: &valInt64, + Annotations: &mcp.Annotations{Audience: []mcp.Role{"user", "assistant"}, LastModified: "2025-01-12T15:00:58Z", Priority: 0.5}, + Icons: []mcp.Icon{{Source: "foobar", MIMEType: "image/png", Sizes: []string{"48x48"}, Theme: "light"}}, + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + // Test that unmarshaling doesn't panic on nil Content fields + defer func() { + if r := recover(); r != nil { + t.Errorf("UnmarshalJSON panicked: %v", r) + } + }() + + err := json.Unmarshal([]byte(tt.json), tt.content) + if err != nil { + t.Errorf("UnmarshalJSON failed: %v", err) + } + + // Verify that the Content field was properly populated + if cmp.Diff(tt.expectContent, tt.content) != "" { + t.Errorf("Content is not equal: %v", cmp.Diff(tt.expectContent, tt.content)) + } + }) + } +} From b706c83e666ca8b05b5f1bfaea0fbdef2253fac2 Mon Sep 17 00:00:00 2001 From: iamsurajbobade Date: Sat, 8 Nov 2025 20:57:36 +0530 Subject: [PATCH 15/15] update conformance test with ResourceLink and Theme --- mcp/conformance_test.go | 26 ++++++- .../spec-sep-973-additional-metadata.txtar | 70 +++++++++++++++++-- 2 files changed, 90 insertions(+), 6 deletions(-) diff --git a/mcp/conformance_test.go b/mcp/conformance_test.go index 17a58eaf..b2801dbc 100644 --- a/mcp/conformance_test.go +++ b/mcp/conformance_test.go @@ -111,6 +111,19 @@ func structuredTool(ctx context.Context, req *CallToolRequest, args *structuredI return nil, &structuredOutput{"Ack " + args.In}, nil } +func contentTool(ctx context.Context, req *CallToolRequest, args *structuredInput) (*CallToolResult, any, error) { + return &CallToolResult{ + Content: []Content{ + &ResourceLink{ + Name: "Example Resource Link with Icons", + MIMEType: "text/plain", + URI: "https://example.com/resource/" + args.In, + Icons: []Icon{iconObj}, + }, + }, + }, nil, nil +} + type tomorrowInput struct { Now time.Time } @@ -135,7 +148,12 @@ func incTool(_ context.Context, _ *CallToolRequest, args incInput) (*CallToolRes return nil, incOutput{args.X + 1}, nil } -var iconObj = Icon{Source: "foobar", MIMEType: "image/png", Sizes: []string{"48x48", "96x96"}} +var iconObj = Icon{ + Source: "foobar", + MIMEType: "image/png", + Sizes: []string{"48x48", "96x96"}, + Theme: "light", +} // runServerTest runs the server conformance test. // It must be executed in a synctest bubble. @@ -174,6 +192,12 @@ func runServerTest(t *testing.T, test *conformanceTest) { } inSchema.Properties["x"].Default = json.RawMessage(`6`) AddTool(s, &Tool{Name: "inc", InputSchema: inSchema}, incTool) + case "contentTool": + AddTool(s, &Tool{ + Name: "contentTool", + Title: "contentTool", + Description: "return resourceLink content with Icon", + }, contentTool) default: t.Fatalf("unknown tool %q", tn) } diff --git a/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar index e69b2cbf..985bff8f 100644 --- a/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar +++ b/mcp/testdata/conformance/server/spec-sep-973-additional-metadata.txtar @@ -8,9 +8,12 @@ Checks following: - Test setting icons for mcp.Prompt - Test setting icons for mcp.Tool - Test setting icons for mcp.Resource +- Test if tool call returning ResourceLink with icons works as expected + -- tools -- greetWithIcon +contentTool -- prompts -- code_reviewWithIcon @@ -33,6 +36,17 @@ infoWithIcon { "jsonrpc": "2.0", "id": 2, "method": "tools/list" } { "jsonrpc": "2.0", "id": 3, "method": "resources/list" } { "jsonrpc": "2.0", "id": 4, "method": "prompts/list" } +{ + "jsonrpc": "2.0", + "id": 5, + "method": "tools/call", + "params": { + "name": "contentTool", + "arguments": { + "In": "resource1" + } + } +} -- server -- { @@ -63,7 +77,8 @@ infoWithIcon "sizes": [ "48x48", "96x96" - ] + ], + "theme": "light" } ] } @@ -74,6 +89,24 @@ infoWithIcon "id": 2, "result": { "tools": [ + { + "description": "return resourceLink content with Icon", + "inputSchema": { + "type": "object", + "required": [ + "In" + ], + "properties": { + "In": { + "type": "string", + "description": "the input" + } + }, + "additionalProperties": false + }, + "name": "contentTool", + "title": "contentTool" + }, { "description": "say hi", "inputSchema": { @@ -96,7 +129,8 @@ infoWithIcon "sizes": [ "48x48", "96x96" - ] + ], + "theme": "light" } ] } @@ -119,7 +153,8 @@ infoWithIcon "sizes": [ "48x48", "96x96" - ] + ], + "theme": "light" } ] } @@ -147,11 +182,36 @@ infoWithIcon "sizes": [ "48x48", "96x96" - ] + ], + "theme": "light" + } + ] + } + ] + } +} +{ + "jsonrpc": "2.0", + "id": 5, + "result": { + "content": [ + { + "type": "resource_link", + "mimeType": "text/plain", + "uri": "https://example.com/resource/resource1", + "name": "Example Resource Link with Icons", + "icons": [ + { + "src": "foobar", + "mimeType": "image/png", + "sizes": [ + "48x48", + "96x96" + ], + "theme": "light" } ] } ] } } -