diff --git a/assets/mapbox-logo-black.svg b/assets/mapbox-logo-black.svg new file mode 100644 index 0000000..a0cbd94 --- /dev/null +++ b/assets/mapbox-logo-black.svg @@ -0,0 +1,38 @@ + + + +Mapbox_Logo_08 + + + + + + + + + + + + + + + diff --git a/assets/mapbox-logo-white.svg b/assets/mapbox-logo-white.svg new file mode 100644 index 0000000..8d62aef --- /dev/null +++ b/assets/mapbox-logo-white.svg @@ -0,0 +1,42 @@ + + + + +Mapbox_Logo_08 + + + + + + + + + + + + + + + diff --git a/src/index.ts b/src/index.ts index 50d53a7..c4313fe 100644 --- a/src/index.ts +++ b/src/index.ts @@ -66,7 +66,21 @@ const allResources = getAllResources(); const server = new McpServer( { name: versionInfo.name, - version: versionInfo.version + version: versionInfo.version, + icons: [ + { + src: '', + mimeType: 'image/svg+xml', + sizes: ['800x180'], + theme: 'light' + }, + { + src: '', + mimeType: 'image/svg+xml', + sizes: ['800x180'], + theme: 'dark' + } + ] }, { capabilities: { diff --git a/src/tools/search-and-geocode-tool/SearchAndGeocodeTool.ts b/src/tools/search-and-geocode-tool/SearchAndGeocodeTool.ts index ebabf3b..87948b2 100644 --- a/src/tools/search-and-geocode-tool/SearchAndGeocodeTool.ts +++ b/src/tools/search-and-geocode-tool/SearchAndGeocodeTool.ts @@ -211,6 +211,89 @@ export class SearchAndGeocodeTool extends MapboxApiBasedTool< `SearchAndGeocodeTool: Successfully completed search, found ${data.features?.length || 0} results` ); + // Check if we have multiple results that might be ambiguous + if ( + this.server && + data.features && + data.features.length >= 2 && + data.features.length <= 10 + ) { + // Use elicitation to let user choose which result they want + try { + const options = data.features.map((feature, index) => { + const props = feature.properties || {}; + let label = props.name || 'Unknown'; + if (props.place_formatted) { + label += ` - ${props.place_formatted}`; + } else if (props.full_address) { + label += ` - ${props.full_address}`; + } + return { value: String(index), label }; + }); + + // Create a JSON Schema with enum for the selection + const result = await this.server.server.elicitInput({ + mode: 'form', + message: `Found ${data.features.length} results for "${input.q}". Please select the correct location:`, + requestedSchema: { + type: 'object', + properties: { + selectedIndex: { + type: 'string', + title: 'Select Location', + description: 'Choose the correct location from the results', + enum: options.map((o) => o.value), + // Include labels for better UX (some clients may support this) + enumNames: options.map((o) => o.label) + } + }, + required: ['selectedIndex'] + } + }); + + if (result.action === 'accept' && result.content?.selectedIndex) { + const selectedIndexStr = + typeof result.content.selectedIndex === 'string' + ? result.content.selectedIndex + : String(result.content.selectedIndex); + const selectedIndex = parseInt(selectedIndexStr, 10); + const selectedFeature = data.features[selectedIndex]; + + // Return only the selected result + const singleResult: SearchBoxResponse = { + ...data, + features: [selectedFeature] + }; + + return { + content: [ + { + type: 'text', + text: this.formatGeoJsonToText( + singleResult as MapboxFeatureCollection + ) + } + ], + structuredContent: singleResult, + isError: false + }; + } else if (result.action === 'decline') { + // User declined to select - return all results as before + this.log( + 'info', + 'SearchAndGeocodeTool: User declined to select a specific result' + ); + } + } catch (elicitError) { + // If elicitation fails, fall back to returning all results + this.log( + 'warning', + `SearchAndGeocodeTool: Elicitation failed: ${elicitError instanceof Error ? elicitError.message : 'Unknown error'}` + ); + } + } + + // Default behavior: return all results return { content: [ { diff --git a/test/tools/search-and-geocode-tool/SearchAndGeocodeTool.test.ts b/test/tools/search-and-geocode-tool/SearchAndGeocodeTool.test.ts index 9bda04e..5d8ee20 100644 --- a/test/tools/search-and-geocode-tool/SearchAndGeocodeTool.test.ts +++ b/test/tools/search-and-geocode-tool/SearchAndGeocodeTool.test.ts @@ -391,4 +391,230 @@ describe('SearchAndGeocodeTool', () => { isError: true }); }); + + describe('Elicitation behavior', () => { + const createMockServer = (elicitResponse?: { + action: 'accept' | 'decline'; + content?: Record; + }) => { + return { + server: { + elicitInput: vi.fn().mockResolvedValue( + elicitResponse || { + action: 'accept', + content: { selectedIndex: '0' } + } + ), + sendLoggingMessage: vi.fn() + }, + registerTool: vi.fn() + } as any; + }; + + const createMultipleResultsResponse = (count: number) => ({ + type: 'FeatureCollection', + features: Array.from({ length: count }, (_, i) => ({ + type: 'Feature', + properties: { + name: `Springfield #${i + 1}`, + place_formatted: `Springfield, State ${i + 1}, United States` + }, + geometry: { + type: 'Point', + coordinates: [-73.0 - i, 42.0 + i] + } + })) + }); + + it('triggers elicitation when 2-10 results returned', async () => { + const mockResponse = createMultipleResultsResponse(5); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer(); + tool.installTo(mockServer); + + await tool.run({ q: 'Springfield' }); + + expect(mockServer.server.elicitInput).toHaveBeenCalledOnce(); + expect(mockServer.server.elicitInput).toHaveBeenCalledWith({ + mode: 'form', + message: + 'Found 5 results for "Springfield". Please select the correct location:', + requestedSchema: expect.objectContaining({ + type: 'object', + properties: expect.objectContaining({ + selectedIndex: expect.objectContaining({ + type: 'string', + title: 'Select Location', + enum: ['0', '1', '2', '3', '4'], + enumNames: expect.arrayContaining([ + expect.stringContaining('Springfield #1'), + expect.stringContaining('Springfield #2') + ]) + }) + }), + required: ['selectedIndex'] + }) + }); + }); + + it('does not trigger elicitation with only 1 result', async () => { + const mockResponse = createMultipleResultsResponse(1); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer(); + tool.installTo(mockServer); + + const result = await tool.run({ q: 'Paris' }); + + expect(mockServer.server.elicitInput).not.toHaveBeenCalled(); + expect(result.isError).toBe(false); + expect((result.structuredContent as any).features).toHaveLength(1); + }); + + it('does not trigger elicitation with more than 10 results', async () => { + const mockResponse = createMultipleResultsResponse(15); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer(); + tool.installTo(mockServer); + + const result = await tool.run({ q: 'Main Street' }); + + expect(mockServer.server.elicitInput).not.toHaveBeenCalled(); + expect(result.isError).toBe(false); + expect((result.structuredContent as any).features).toHaveLength(15); + }); + + it('returns only selected result when user accepts elicitation', async () => { + const mockResponse = createMultipleResultsResponse(3); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer({ + action: 'accept', + content: { selectedIndex: '1' } // Select second item + }); + tool.installTo(mockServer); + + const result = await tool.run({ q: 'Springfield' }); + + expect(result.isError).toBe(false); + const features = (result.structuredContent as any).features; + expect(features).toHaveLength(1); + expect(features[0].properties.name).toBe('Springfield #2'); + }); + + it('returns all results when user declines elicitation', async () => { + const mockResponse = createMultipleResultsResponse(4); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer({ + action: 'decline' + }); + tool.installTo(mockServer); + + const result = await tool.run({ q: 'Springfield' }); + + expect(result.isError).toBe(false); + const features = (result.structuredContent as any).features; + expect(features).toHaveLength(4); + }); + + it('falls back to all results when elicitation fails', async () => { + const mockResponse = createMultipleResultsResponse(3); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = { + server: { + elicitInput: vi + .fn() + .mockRejectedValue(new Error('Elicitation not supported')), + sendLoggingMessage: vi.fn() + }, + registerTool: vi.fn() + } as any; + tool.installTo(mockServer); + + const result = await tool.run({ q: 'Springfield' }); + + expect(result.isError).toBe(false); + const features = (result.structuredContent as any).features; + expect(features).toHaveLength(3); + }); + + it('handles elicitation gracefully when server is not installed', async () => { + const mockResponse = createMultipleResultsResponse(5); + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + // Don't install to server - tool.server will be null + + const result = await tool.run({ q: 'Springfield' }); + + expect(result.isError).toBe(false); + const features = (result.structuredContent as any).features; + expect(features).toHaveLength(5); + }); + + it('builds correct enumNames with location labels', async () => { + const mockResponse = { + type: 'FeatureCollection', + features: [ + { + type: 'Feature', + properties: { + name: 'Springfield', + place_formatted: 'Springfield, Illinois, United States' + }, + geometry: { type: 'Point', coordinates: [-89.6501, 39.7817] } + }, + { + type: 'Feature', + properties: { + name: 'Springfield', + full_address: '123 Main St, Springfield, MA 01103' + }, + geometry: { type: 'Point', coordinates: [-72.5301, 42.1015] } + } + ] + }; + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const tool = new SearchAndGeocodeTool({ httpRequest }); + const mockServer = createMockServer(); + tool.installTo(mockServer); + + await tool.run({ q: 'Springfield' }); + + const elicitCall = mockServer.server.elicitInput.mock.calls[0][0]; + expect( + elicitCall.requestedSchema.properties.selectedIndex.enumNames + ).toEqual([ + 'Springfield - Springfield, Illinois, United States', + 'Springfield - 123 Main St, Springfield, MA 01103' + ]); + }); + }); });