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/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. 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. // diff --git a/mcp/testdata/conformance/server/version-draft.txtar b/mcp/testdata/conformance/server/version-draft.txtar new file mode 100644 index 00000000..f218fa57 --- /dev/null +++ b/mcp/testdata/conformance/server/version-draft.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": "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" + ] + } + ] + } + ] + } +} +