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 @@
+
+
+
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 @@
+
+
+
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'
+ ]);
+ });
+ });
});