Skip to main content
Glama

mcp-metabase-server P

by easecloudio
card-tools.ts7.07 kB
/** * Card/Question-related tool handlers */ import { MetabaseClient } from "../client/metabase-client.js"; import { ErrorCode, McpError } from "../types/errors.js"; import { Tool } from "@modelcontextprotocol/sdk/types.js"; export class CardToolHandlers { constructor(private client: MetabaseClient) {} getToolSchemas(): Tool[] { return [ { name: "list_cards", description: "List all questions/cards in Metabase", inputSchema: { type: "object", properties: {}, }, }, { name: "create_card", description: "Create a new Metabase question (card)", inputSchema: { type: "object", properties: { name: { type: "string", description: "Name of the card" }, dataset_query: { type: "object", description: "The query for the card (e.g., MBQL or native query)", }, display: { type: "string", description: "Display type (e.g., 'table', 'line', 'bar')", }, visualization_settings: { type: "object", description: "Settings for the visualization", }, collection_id: { type: "number", description: "Optional ID of the collection to save the card in", }, description: { type: "string", description: "Optional description for the card", }, }, required: [ "name", "dataset_query", "display", "visualization_settings", ], }, }, { name: "update_card", description: "Update an existing Metabase question (card)", inputSchema: { type: "object", properties: { card_id: { type: "number", description: "ID of the card to update", }, name: { type: "string", description: "New name for the card" }, dataset_query: { type: "object", description: "New query for the card", }, display: { type: "string", description: "New display type" }, visualization_settings: { type: "object", description: "New visualization settings", }, collection_id: { type: "number", description: "New collection ID" }, description: { type: "string", description: "New description" }, archived: { type: "boolean", description: "Set to true to archive the card", }, }, required: ["card_id"], }, }, { name: "delete_card", description: "Delete a Metabase question (card)", inputSchema: { type: "object", properties: { card_id: { type: "number", description: "ID of the card to delete", }, hard_delete: { type: "boolean", description: "Set to true for hard delete, false (default) for archive", default: false, }, }, required: ["card_id"], }, }, { name: "execute_card", description: "Execute a Metabase question/card and get results", inputSchema: { type: "object", properties: { card_id: { type: "number", description: "ID of the card/question to execute", }, parameters: { type: "object", description: "Optional parameters for the query", }, }, required: ["card_id"], }, }, ]; } async handleTool(name: string, args: any): Promise<any> { switch (name) { case "list_cards": return await this.listCards(); case "create_card": return await this.createCard(args); case "update_card": return await this.updateCard(args); case "delete_card": return await this.deleteCard(args); case "execute_card": return await this.executeCard(args); default: throw new McpError( ErrorCode.MethodNotFound, `Unknown card tool: ${name}` ); } } private async listCards(): Promise<any> { const cards = await this.client.getCards(); return { content: [ { type: "text", text: JSON.stringify(cards, null, 2), }, ], }; } private async createCard(args: any): Promise<any> { const { name, dataset_query, display, visualization_settings, collection_id, description, } = args; if (!name || !dataset_query || !display || !visualization_settings) { throw new McpError( ErrorCode.InvalidParams, "Missing required fields: name, dataset_query, display, visualization_settings" ); } const cardData: any = { name, dataset_query, display, visualization_settings, }; if (collection_id !== undefined) cardData.collection_id = collection_id; if (description !== undefined) cardData.description = description; const card = await this.client.createCard(cardData); return { content: [ { type: "text", text: JSON.stringify(card, null, 2), }, ], }; } private async updateCard(args: any): Promise<any> { const { card_id, ...updateFields } = args; if (!card_id) { throw new McpError(ErrorCode.InvalidParams, "Card ID is required"); } if (Object.keys(updateFields).length === 0) { throw new McpError( ErrorCode.InvalidParams, "No fields provided for update" ); } const card = await this.client.updateCard(card_id, updateFields); return { content: [ { type: "text", text: JSON.stringify(card, null, 2), }, ], }; } private async deleteCard(args: any): Promise<any> { const { card_id, hard_delete = false } = args; if (!card_id) { throw new McpError(ErrorCode.InvalidParams, "Card ID is required"); } await this.client.deleteCard(card_id, hard_delete); return { content: [ { type: "text", text: hard_delete ? `Card ${card_id} permanently deleted.` : `Card ${card_id} archived.`, }, ], }; } private async executeCard(args: any): Promise<any> { const { card_id, parameters = {} } = args; if (!card_id) { throw new McpError(ErrorCode.InvalidParams, "Card ID is required"); } const result = await this.client.executeCard(card_id, parameters); return { content: [ { type: "text", text: JSON.stringify(result, null, 2), }, ], }; } }

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/easecloudio/mcp-metabase-server'

If you have feedback or need assistance with the MCP directory API, please join our Discord server