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/directions-tool/DirectionsTool.output.schema.ts b/src/tools/directions-tool/DirectionsTool.output.schema.ts index 0523025..3e4016d 100644 --- a/src/tools/directions-tool/DirectionsTool.output.schema.ts +++ b/src/tools/directions-tool/DirectionsTool.output.schema.ts @@ -284,8 +284,8 @@ const RouteStepSchema = z.object({ distance: z.number(), duration: z.number(), weight: z.number(), - duration_typical: z.number().optional(), - weight_typical: z.number().optional(), + duration_typical: z.number().nullable().optional(), + weight_typical: z.number().nullable().optional(), geometry: z.union([z.string(), GeoJSONLineStringSchema]), name: z.string(), ref: z.string().optional(), @@ -315,8 +315,8 @@ const RouteLegSchema = z.object({ distance: z.number(), duration: z.number(), weight: z.number(), - duration_typical: z.number().optional(), - weight_typical: z.number().optional(), + duration_typical: z.number().nullable().optional(), + weight_typical: z.number().nullable().optional(), steps: z.array(RouteStepSchema), summary: z.string(), admins: z.array(AdminSchema), @@ -333,8 +333,8 @@ const RouteSchema = z.object({ distance: z.number(), weight_name: z.enum(['auto', 'pedestrian']).optional(), // Removed by cleanResponseData weight: z.number().optional(), // Removed by cleanResponseData - duration_typical: z.number().optional(), - weight_typical: z.number().optional(), + duration_typical: z.number().nullable().optional(), + weight_typical: z.number().nullable().optional(), geometry: z.union([z.string(), GeoJSONLineStringSchema]).optional(), // Can be removed when geometries='none' legs: z.array(RouteLegSchema).optional(), // Removed by cleanResponseData, replaced with leg_summaries voiceLocale: z.string().optional(), diff --git a/src/tools/directions-tool/DirectionsTool.ts b/src/tools/directions-tool/DirectionsTool.ts index 16fa0bc..37fd31f 100644 --- a/src/tools/directions-tool/DirectionsTool.ts +++ b/src/tools/directions-tool/DirectionsTool.ts @@ -38,10 +38,117 @@ export class DirectionsTool extends MapboxApiBasedTool< httpRequest: params.httpRequest }); } + private formatDuration(seconds: number): string { + const hours = Math.floor(seconds / 3600); + const minutes = Math.round((seconds % 3600) / 60); + if (hours > 0) { + return `${hours}h ${minutes}min`; + } + return `${minutes}min`; + } + + private formatDistance(meters: number): string { + const km = (meters / 1000).toFixed(1); + const miles = (meters / 1609.34).toFixed(1); + return `${km}km (${miles}mi)`; + } + protected async execute( input: z.infer, accessToken: string ): Promise { + // Stage 1: Elicit routing preferences if server available and no preferences set + const excludeOptions: string[] = []; + if ( + this.server && + !input.exclude && + input.coordinates.length === 2 // Only for simple A-to-B routes + ) { + try { + const isDrivingProfile = + input.routing_profile === 'mapbox/driving-traffic' || + input.routing_profile === 'mapbox/driving'; + + const preferenceOptions = [ + { + value: 'fastest', + label: 'Fastest route (tolls OK)' + }, + { + value: 'avoid_tolls', + label: 'Avoid tolls (may be slower)' + } + ]; + + if (isDrivingProfile) { + preferenceOptions.push({ + value: 'avoid_highways', + label: 'Avoid highways/motorways' + }); + } + + preferenceOptions.push({ + value: 'avoid_ferries', + label: 'Avoid ferries' + }); + + const preferencesResult = await this.server.server.elicitInput({ + mode: 'form', + message: 'Choose your routing preferences:', + requestedSchema: { + type: 'object', + properties: { + routePreference: { + type: 'string', + title: 'Route Preference', + description: 'Select your routing priorities', + enum: preferenceOptions.map((o) => o.value), + enumNames: preferenceOptions.map((o) => o.label) + } + }, + required: ['routePreference'] + } + }); + + if ( + preferencesResult.action === 'accept' && + preferencesResult.content?.routePreference + ) { + const preference = + typeof preferencesResult.content.routePreference === 'string' + ? preferencesResult.content.routePreference + : String(preferencesResult.content.routePreference); + + // Map preferences to exclude options + if (preference === 'avoid_tolls') { + excludeOptions.push('toll', 'cash_only_tolls'); + } else if (preference === 'avoid_highways' && isDrivingProfile) { + excludeOptions.push('motorway'); + } else if (preference === 'avoid_ferries') { + excludeOptions.push('ferry'); + } + + // Force alternatives=true to get multiple routes for Stage 2 + input.alternatives = true; + + this.log( + 'info', + `DirectionsTool: User selected preference: ${preference}, exclude: ${excludeOptions.join(',')}` + ); + } + } catch (elicitError) { + this.log( + 'warning', + `DirectionsTool: Stage 1 elicitation failed: ${elicitError instanceof Error ? elicitError.message : 'Unknown error'}` + ); + } + } + + // Apply collected exclusions + if (excludeOptions.length > 0 && !input.exclude) { + input.exclude = excludeOptions.join(','); + } + // Validate exclude parameter against the actual routing_profile // This is needed because some exclusions are only driving specific if (input.exclude) { @@ -268,6 +375,116 @@ export class DirectionsTool extends MapboxApiBasedTool< validatedData = cleanedData as DirectionsResponse; } + // Stage 2: Elicit route selection if multiple routes returned + if ( + this.server && + validatedData.routes && + validatedData.routes.length >= 2 + ) { + try { + const routeOptions = validatedData.routes.map((route, index) => { + const duration = this.formatDuration(route.duration); + const distance = this.formatDistance(route.distance); + const roads = + route.leg_summaries && route.leg_summaries.length > 0 + ? route.leg_summaries[0] + : 'Route'; + + // Build traffic/congestion summary + let trafficInfo = ''; + if (route.congestion_information) { + const congestion = route.congestion_information; + const totalLength = + congestion.length_low + + congestion.length_moderate + + congestion.length_heavy + + congestion.length_severe; + const heavyPercent = Math.round( + ((congestion.length_heavy + congestion.length_severe) / + totalLength) * + 100 + ); + if (heavyPercent > 20) { + trafficInfo = ` ⚠️ Heavy traffic (${heavyPercent}%)`; + } else if (congestion.length_moderate > 0) { + trafficInfo = ' 🟡 Moderate traffic'; + } else { + trafficInfo = ' ✅ Light traffic'; + } + } + + // Count incidents + const incidentCount = route.incidents_summary?.length || 0; + const incidentInfo = + incidentCount > 0 ? ` • ${incidentCount} incident(s)` : ''; + + return { + value: String(index), + label: `${duration} via ${roads} • ${distance}${trafficInfo}${incidentInfo}` + }; + }); + + const routeSelectionResult = await this.server.server.elicitInput({ + mode: 'form', + message: `Found ${validatedData.routes.length} routes. Choose your preferred route:`, + requestedSchema: { + type: 'object', + properties: { + selectedRoute: { + type: 'string', + title: 'Select Route', + description: 'Choose the route that best fits your needs', + enum: routeOptions.map((o) => o.value), + enumNames: routeOptions.map((o) => o.label) + } + }, + required: ['selectedRoute'] + } + }); + + if ( + routeSelectionResult.action === 'accept' && + routeSelectionResult.content?.selectedRoute + ) { + const selectedIndexStr = + typeof routeSelectionResult.content.selectedRoute === 'string' + ? routeSelectionResult.content.selectedRoute + : String(routeSelectionResult.content.selectedRoute); + const selectedIndex = parseInt(selectedIndexStr, 10); + const selectedRoute = validatedData.routes[selectedIndex]; + + // Return only the selected route + const singleRouteResult: DirectionsResponse = { + ...validatedData, + routes: [selectedRoute] + }; + + this.log( + 'info', + `DirectionsTool: User selected route ${selectedIndex}: ${this.formatDuration(selectedRoute.duration)} via ${selectedRoute.leg_summaries?.[0] || 'route'}` + ); + + return { + content: [ + { type: 'text', text: JSON.stringify(singleRouteResult, null, 2) } + ], + structuredContent: singleRouteResult, + isError: false + }; + } else if (routeSelectionResult.action === 'decline') { + this.log( + 'info', + 'DirectionsTool: User declined to select a specific route' + ); + } + } catch (elicitError) { + this.log( + 'warning', + `DirectionsTool: Stage 2 elicitation failed: ${elicitError instanceof Error ? elicitError.message : 'Unknown error'}` + ); + } + } + return { content: [{ type: 'text', text: JSON.stringify(validatedData, null, 2) }], structuredContent: validatedData, diff --git a/src/tools/directions-tool/cleanResponseData.ts b/src/tools/directions-tool/cleanResponseData.ts index 6cbefbb..9078724 100644 --- a/src/tools/directions-tool/cleanResponseData.ts +++ b/src/tools/directions-tool/cleanResponseData.ts @@ -67,8 +67,8 @@ interface RawRoute { distance?: number; weight_name?: string; weight?: number; - duration_typical?: number; - weight_typical?: number; + duration_typical?: number | null; + weight_typical?: number | null; geometry?: unknown; legs?: RawLeg[]; [key: string]: unknown; diff --git a/test/tools/directions-tool/DirectionsTool.test.ts b/test/tools/directions-tool/DirectionsTool.test.ts index 888d088..37c41b7 100644 --- a/test/tools/directions-tool/DirectionsTool.test.ts +++ b/test/tools/directions-tool/DirectionsTool.test.ts @@ -1064,4 +1064,155 @@ describe('DirectionsTool', () => { isError: true }); }); + + describe('elicitation behavior', () => { + it('does not elicit when exclude parameter is already provided', async () => { + const { httpRequest, mockHttpRequest } = setupHttpRequest(); + + await new DirectionsTool({ httpRequest }).run({ + coordinates: [ + { longitude: -74.006, latitude: 40.7128 }, + { longitude: -71.0589, latitude: 42.3601 } + ], + exclude: 'toll' + }); + + const calledUrl = mockHttpRequest.mock.calls[0][0]; + // Should use the provided exclude parameter + expect(calledUrl).toContain('exclude=toll'); + // Should not modify alternatives (defaults to false) + expect(calledUrl).toContain('alternatives=false'); + }); + + it('does not elicit when more than 2 coordinates provided', async () => { + const { httpRequest, mockHttpRequest } = setupHttpRequest(); + + await new DirectionsTool({ httpRequest }).run({ + coordinates: [ + { longitude: -74.006, latitude: 40.7128 }, + { longitude: -73.9, latitude: 41.0 }, + { longitude: -71.0589, latitude: 42.3601 } + ] + }); + + const calledUrl = mockHttpRequest.mock.calls[0][0]; + // Should not set alternatives=true (elicitation skipped for multi-waypoint) + expect(calledUrl).toContain('alternatives=false'); + }); + + it('returns multiple routes when alternatives=true and no elicitation', async () => { + const mockResponse = { + code: 'Ok', + routes: [ + { + duration: 17280, + distance: 366800, + legs: [ + { + summary: 'I-84 East, I-90 East', + duration: 17280, + distance: 366800, + steps: [] + } + ], + leg_summaries: ['I-84 East, I-90 East'], + congestion_information: { + length_low: 300000, + length_moderate: 50000, + length_heavy: 16800, + length_severe: 0 + }, + incidents_summary: [{ type: 'construction' }] + }, + { + duration: 17880, + distance: 348000, + legs: [ + { + summary: 'CT-15 North, I-90 East', + duration: 17880, + distance: 348000, + steps: [] + } + ], + leg_summaries: ['CT-15 North, I-90 East'], + congestion_information: { + length_low: 280000, + length_moderate: 60000, + length_heavy: 8000, + length_severe: 0 + }, + incidents_summary: [ + { type: 'construction' }, + { type: 'accident' }, + { type: 'construction' }, + { type: 'construction' }, + { type: 'construction' } + ] + } + ], + waypoints: [] + }; + + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const result = await new DirectionsTool({ httpRequest }).run({ + coordinates: [ + { longitude: -74.006, latitude: 40.7128 }, + { longitude: -71.0589, latitude: 42.3601 } + ], + alternatives: true + }); + + expect(result.isError).toBe(false); + // Without elicitation, should return all routes + const response = JSON.parse( + (result.content[0] as { type: 'text'; text: string }).text + ); + expect(response.routes).toHaveLength(2); + }); + + it('returns single route when only one route available (no Stage 2 elicitation)', async () => { + const mockResponse = { + code: 'Ok', + routes: [ + { + duration: 17280, + distance: 366800, + legs: [ + { + summary: 'I-84 East, I-90 East', + duration: 17280, + distance: 366800, + steps: [] + } + ], + leg_summaries: ['I-84 East, I-90 East'] + } + ], + waypoints: [] + }; + + const { httpRequest } = setupHttpRequest({ + json: async () => mockResponse + }); + + const result = await new DirectionsTool({ httpRequest }).run({ + coordinates: [ + { longitude: -74.006, latitude: 40.7128 }, + { longitude: -71.0589, latitude: 42.3601 } + ], + alternatives: true + }); + + expect(result.isError).toBe(false); + // Should return the single route + const response = JSON.parse( + (result.content[0] as { type: 'text'; text: string }).text + ); + expect(response.routes).toHaveLength(1); + }); + }); });