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/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);
+ });
+ });
});