@mcp-z/mcp-sheets 1.0.0 → 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +32 -3
- package/bin/server.js +1 -1
- package/dist/cjs/constants.js.map +1 -1
- package/dist/cjs/index.js +13 -13
- package/dist/cjs/index.js.map +1 -1
- package/dist/cjs/lib/create-store.js.map +1 -1
- package/dist/cjs/mcp/index.js +6 -6
- package/dist/cjs/mcp/index.js.map +1 -1
- package/dist/cjs/mcp/prompts/a1-notation.js.map +1 -1
- package/dist/cjs/mcp/prompts/index.js +2 -2
- package/dist/cjs/mcp/prompts/index.js.map +1 -1
- package/dist/cjs/mcp/resources/index.js +2 -2
- package/dist/cjs/mcp/resources/index.js.map +1 -1
- package/dist/cjs/mcp/resources/spreadsheet.js.map +1 -1
- package/dist/cjs/mcp/tools/cells-format.js +8 -8
- package/dist/cjs/mcp/tools/cells-format.js.map +1 -1
- package/dist/cjs/mcp/tools/chart-create.js +8 -8
- package/dist/cjs/mcp/tools/chart-create.js.map +1 -1
- package/dist/cjs/mcp/tools/columns-get.js +3 -3
- package/dist/cjs/mcp/tools/columns-get.js.map +1 -1
- package/dist/cjs/mcp/tools/columns-update.js +10 -10
- package/dist/cjs/mcp/tools/columns-update.js.map +1 -1
- package/dist/cjs/mcp/tools/csv-get-columns.js +2 -2
- package/dist/cjs/mcp/tools/csv-get-columns.js.map +1 -1
- package/dist/cjs/mcp/tools/dimensions-batch-update.js +16 -16
- package/dist/cjs/mcp/tools/dimensions-batch-update.js.map +1 -1
- package/dist/cjs/mcp/tools/dimensions-move.js +5 -5
- package/dist/cjs/mcp/tools/dimensions-move.js.map +1 -1
- package/dist/cjs/mcp/tools/index.js +52 -52
- package/dist/cjs/mcp/tools/index.js.map +1 -1
- package/dist/cjs/mcp/tools/lib/dimension-operations.js.map +1 -1
- package/dist/cjs/mcp/tools/rows-append.js +10 -10
- package/dist/cjs/mcp/tools/rows-append.js.map +1 -1
- package/dist/cjs/mcp/tools/rows-csv-append.js +13 -13
- package/dist/cjs/mcp/tools/rows-csv-append.js.map +1 -1
- package/dist/cjs/mcp/tools/rows-get.js +4 -4
- package/dist/cjs/mcp/tools/rows-get.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-copy-to.js +4 -4
- package/dist/cjs/mcp/tools/sheet-copy-to.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-copy.js +5 -5
- package/dist/cjs/mcp/tools/sheet-copy.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-create.js +4 -4
- package/dist/cjs/mcp/tools/sheet-create.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-delete.js +4 -4
- package/dist/cjs/mcp/tools/sheet-delete.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-find.js +7 -7
- package/dist/cjs/mcp/tools/sheet-find.js.map +1 -1
- package/dist/cjs/mcp/tools/sheet-rename.js +5 -5
- package/dist/cjs/mcp/tools/sheet-rename.js.map +1 -1
- package/dist/cjs/mcp/tools/spreadsheet-copy.js +2 -2
- package/dist/cjs/mcp/tools/spreadsheet-copy.js.map +1 -1
- package/dist/cjs/mcp/tools/spreadsheet-create.js +2 -2
- package/dist/cjs/mcp/tools/spreadsheet-create.js.map +1 -1
- package/dist/cjs/mcp/tools/spreadsheet-find.js +6 -6
- package/dist/cjs/mcp/tools/spreadsheet-find.js.map +1 -1
- package/dist/cjs/mcp/tools/spreadsheet-rename.js +3 -3
- package/dist/cjs/mcp/tools/spreadsheet-rename.js.map +1 -1
- package/dist/cjs/mcp/tools/validation-set.js +8 -8
- package/dist/cjs/mcp/tools/validation-set.js.map +1 -1
- package/dist/cjs/mcp/tools/values-batch-update.js +8 -8
- package/dist/cjs/mcp/tools/values-batch-update.js.map +1 -1
- package/dist/cjs/mcp/tools/values-clear.js +6 -6
- package/dist/cjs/mcp/tools/values-clear.js.map +1 -1
- package/dist/cjs/mcp/tools/values-csv-update.js +8 -8
- package/dist/cjs/mcp/tools/values-csv-update.js.map +1 -1
- package/dist/cjs/mcp/tools/values-replace.js +8 -8
- package/dist/cjs/mcp/tools/values-replace.js.map +1 -1
- package/dist/cjs/mcp/tools/values-search.js +5 -5
- package/dist/cjs/mcp/tools/values-search.js.map +1 -1
- package/dist/cjs/schemas/index.js.map +1 -1
- package/dist/cjs/setup/config.js +11 -1
- package/dist/cjs/setup/config.js.map +1 -1
- package/dist/cjs/setup/http.js +6 -2
- package/dist/cjs/setup/http.js.map +1 -1
- package/dist/cjs/setup/index.js +9 -9
- package/dist/cjs/setup/index.js.map +1 -1
- package/dist/cjs/setup/oauth-google.d.cts +3 -2
- package/dist/cjs/setup/oauth-google.d.ts +3 -2
- package/dist/cjs/setup/oauth-google.js +15 -12
- package/dist/cjs/setup/oauth-google.js.map +1 -1
- package/dist/cjs/setup/runtime.js +9 -9
- package/dist/cjs/setup/runtime.js.map +1 -1
- package/dist/cjs/setup/stdio.js +2 -2
- package/dist/cjs/setup/stdio.js.map +1 -1
- package/dist/cjs/spreadsheet/column-utilities.js.map +1 -1
- package/dist/cjs/spreadsheet/csv-streaming.js.map +1 -1
- package/dist/cjs/spreadsheet/data-operations.js +9 -9
- package/dist/cjs/spreadsheet/data-operations.js.map +1 -1
- package/dist/cjs/spreadsheet/deduplication-utils.js.map +1 -1
- package/dist/cjs/spreadsheet/range-operations.js +2 -2
- package/dist/cjs/spreadsheet/range-operations.js.map +1 -1
- package/dist/cjs/spreadsheet/sheet-operations.js +3 -3
- package/dist/cjs/spreadsheet/sheet-operations.js.map +1 -1
- package/dist/cjs/spreadsheet/spreadsheet-management.js.map +1 -1
- package/dist/esm/constants.js.map +1 -1
- package/dist/esm/index.js.map +1 -1
- package/dist/esm/lib/create-store.js.map +1 -1
- package/dist/esm/mcp/index.js.map +1 -1
- package/dist/esm/mcp/prompts/a1-notation.js.map +1 -1
- package/dist/esm/mcp/prompts/index.js.map +1 -1
- package/dist/esm/mcp/resources/index.js.map +1 -1
- package/dist/esm/mcp/resources/spreadsheet.js.map +1 -1
- package/dist/esm/mcp/tools/cells-format.js.map +1 -1
- package/dist/esm/mcp/tools/chart-create.js.map +1 -1
- package/dist/esm/mcp/tools/columns-get.js.map +1 -1
- package/dist/esm/mcp/tools/columns-update.js.map +1 -1
- package/dist/esm/mcp/tools/csv-get-columns.js.map +1 -1
- package/dist/esm/mcp/tools/dimensions-batch-update.js.map +1 -1
- package/dist/esm/mcp/tools/dimensions-move.js.map +1 -1
- package/dist/esm/mcp/tools/index.js.map +1 -1
- package/dist/esm/mcp/tools/lib/dimension-operations.js.map +1 -1
- package/dist/esm/mcp/tools/rows-append.js.map +1 -1
- package/dist/esm/mcp/tools/rows-csv-append.js.map +1 -1
- package/dist/esm/mcp/tools/rows-get.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-copy-to.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-copy.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-create.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-delete.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-find.js.map +1 -1
- package/dist/esm/mcp/tools/sheet-rename.js.map +1 -1
- package/dist/esm/mcp/tools/spreadsheet-copy.js.map +1 -1
- package/dist/esm/mcp/tools/spreadsheet-create.js.map +1 -1
- package/dist/esm/mcp/tools/spreadsheet-find.js.map +1 -1
- package/dist/esm/mcp/tools/spreadsheet-rename.js.map +1 -1
- package/dist/esm/mcp/tools/validation-set.js.map +1 -1
- package/dist/esm/mcp/tools/values-batch-update.js.map +1 -1
- package/dist/esm/mcp/tools/values-clear.js.map +1 -1
- package/dist/esm/mcp/tools/values-csv-update.js.map +1 -1
- package/dist/esm/mcp/tools/values-replace.js.map +1 -1
- package/dist/esm/mcp/tools/values-search.js.map +1 -1
- package/dist/esm/schemas/index.js.map +1 -1
- package/dist/esm/setup/config.js +12 -2
- package/dist/esm/setup/config.js.map +1 -1
- package/dist/esm/setup/http.js +4 -0
- package/dist/esm/setup/http.js.map +1 -1
- package/dist/esm/setup/index.js.map +1 -1
- package/dist/esm/setup/oauth-google.d.ts +3 -2
- package/dist/esm/setup/oauth-google.js +8 -11
- package/dist/esm/setup/oauth-google.js.map +1 -1
- package/dist/esm/setup/runtime.js.map +1 -1
- package/dist/esm/setup/stdio.js.map +1 -1
- package/dist/esm/spreadsheet/column-utilities.js.map +1 -1
- package/dist/esm/spreadsheet/csv-streaming.js.map +1 -1
- package/dist/esm/spreadsheet/data-operations.js.map +1 -1
- package/dist/esm/spreadsheet/deduplication-utils.js.map +1 -1
- package/dist/esm/spreadsheet/range-operations.js.map +1 -1
- package/dist/esm/spreadsheet/sheet-operations.js.map +1 -1
- package/dist/esm/spreadsheet/spreadsheet-management.js.map +1 -1
- package/dist/esm/types.js.map +1 -1
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["/Users/kevin/Dev/Projects/ai/mcp-z/servers/mcp-sheets/src/mcp/tools/chart-create.ts"],"sourcesContent":["import type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { google, type sheets_v4 } from 'googleapis';\nimport { z } from 'zod';\nimport { SheetGidOutput, SheetGidSchema, SpreadsheetIdOutput, SpreadsheetIdSchema } from '../../schemas/index.js';\nimport { parseA1Notation, rangeReferenceToGridRange } from '../../spreadsheet/range-operations.js';\n\n// Input schema for chart position\nconst ChartPositionSchema = z.object({\n anchorCell: z.string().min(1).describe('A1 notation cell where chart top-left anchors (e.g., \"F2\")'),\n offsetX: z.number().int().default(0).describe('Horizontal pixel offset from anchor'),\n offsetY: z.number().int().default(0).describe('Vertical pixel offset from anchor'),\n});\n\nconst inputSchema = z.object({\n id: SpreadsheetIdSchema,\n gid: SheetGidSchema,\n chartType: z.enum(['PIE', 'BAR', 'COLUMN', 'LINE']).describe('Type of chart to create'),\n dataRange: z.string().min(1).describe('A1 notation range containing chart data including headers (e.g., \"A1:C10\")'),\n title: z.string().optional().describe('Chart title displayed at top'),\n position: ChartPositionSchema,\n legend: z.enum(['BOTTOM', 'RIGHT', 'TOP', 'LEFT', 'NONE']).default('BOTTOM').describe('Legend position'),\n is3D: z.boolean().default(false).describe('Render as 3D chart (PIE only)'),\n});\n\n// Success branch schema\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n id: SpreadsheetIdOutput,\n gid: SheetGidOutput,\n sheetTitle: z.string().describe('Title of the sheet containing the chart'),\n sheetUrl: z.string().describe('URL of the sheet containing the chart'),\n chartId: z.number().int().describe('Unique chart ID for future updates/deletion'),\n anchorCell: z.string().describe('Where chart was anchored'),\n});\n\n// Output schema with auth_required support\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Create charts (pie, bar, column, line) from spreadsheet data ranges. Charts anchor to specific cells with optional pixel offsets. Data range should include headers. PIE charts use 2 columns (labels, values). BAR/COLUMN/LINE charts use first row as headers. Best for visualizing spreadsheet data.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\n// Parse A1 notation anchor cell to row/column indices\nfunction parseAnchorCell(anchorCell: string): { rowIndex: number; columnIndex: number } {\n // Simple A1 notation parser for single cells\n const match = anchorCell.match(/^([A-Z]+)(\\d+)$/);\n if (!match || !match[1] || !match[2]) {\n throw new Error(`Invalid anchor cell format: ${anchorCell}`);\n }\n\n const colLetters = match[1];\n const rowNumber = match[2];\n\n // Convert column letters to index (A=0, B=1, ..., Z=25, AA=26, etc.)\n let columnIndex = 0;\n for (let i = 0; i < colLetters.length; i++) {\n columnIndex = columnIndex * 26 + (colLetters.charCodeAt(i) - 65 + 1);\n }\n columnIndex--; // Convert to 0-based\n\n // Convert row number to index (1-based to 0-based)\n const rowIndex = Number.parseInt(rowNumber, 10) - 1;\n\n return { rowIndex, columnIndex };\n}\n\nasync function handler({ id, gid, chartType, dataRange, title, position, legend = 'BOTTOM', is3D = false }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.info('sheets.chart.create called', {\n id,\n gid,\n chartType,\n dataRange,\n title,\n position,\n legend,\n is3D,\n });\n\n try {\n const sheets = google.sheets({ version: 'v4', auth: extra.authContext.auth });\n\n // Get spreadsheet and sheet info in single API call\n const spreadsheetResponse = await sheets.spreadsheets.get({\n spreadsheetId: id,\n fields: 'sheets.properties.sheetId,sheets.properties.title',\n });\n\n // Find sheet by gid\n const sheet = spreadsheetResponse.data.sheets?.find((s) => String(s.properties?.sheetId) === gid);\n if (!sheet?.properties) {\n logger.info('Sheet not found for chart create', { id, gid, chartType });\n throw new McpError(ErrorCode.InvalidParams, `Sheet not found: ${gid}`);\n }\n\n const sheetTitle = sheet.properties.title ?? gid;\n const sheetId = sheet.properties.sheetId;\n const sheetUrl = `https://docs.google.com/spreadsheets/d/${id}/edit#gid=${sheetId}`;\n\n // Validate 3D only for PIE charts\n // Note: Google Sheets API only supports 3D for PIE charts, not COLUMN/BAR/LINE\n if (is3D && chartType !== 'PIE') {\n logger.info('3D mode not supported for this chart type', {\n chartType,\n is3D,\n });\n throw new McpError(ErrorCode.InvalidParams, `3D mode is only supported for PIE charts, not ${chartType}`);\n }\n\n // Parse anchor cell to row/column indices\n let anchorRowIndex: number;\n let anchorColumnIndex: number;\n try {\n const anchorIndices = parseAnchorCell(position.anchorCell);\n anchorRowIndex = anchorIndices.rowIndex;\n anchorColumnIndex = anchorIndices.columnIndex;\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.info('Failed to parse anchor cell', {\n anchorCell: position.anchorCell,\n error: message,\n });\n throw new McpError(ErrorCode.InvalidParams, `Failed to parse anchor cell: ${message}`);\n }\n\n // Parse data range to grid range\n let dataGridRange: sheets_v4.Schema$GridRange;\n try {\n const rangeRef = parseA1Notation(dataRange);\n dataGridRange = rangeReferenceToGridRange(rangeRef, sheetId);\n\n // Validate that required properties are defined for chart creation\n if (dataGridRange.startColumnIndex === undefined || dataGridRange.startColumnIndex === null) {\n throw new Error('Data range must include column information');\n }\n if (dataGridRange.endColumnIndex === undefined || dataGridRange.endColumnIndex === null) {\n throw new Error('Data range must include column information');\n }\n if (dataGridRange.startRowIndex === undefined || dataGridRange.startRowIndex === null) {\n throw new Error('Data range must include row information');\n }\n if (dataGridRange.endRowIndex === undefined || dataGridRange.endRowIndex === null) {\n throw new Error('Data range must include row information');\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.info('Failed to parse data range', {\n dataRange,\n error: message,\n });\n throw new McpError(ErrorCode.InvalidParams, `Failed to parse data range: ${message}`);\n }\n\n // Extract validated properties with defined types\n const dataStartRowIndex = dataGridRange.startRowIndex as number;\n const dataEndRowIndex = dataGridRange.endRowIndex as number;\n const dataStartColumnIndex = dataGridRange.startColumnIndex as number;\n const dataEndColumnIndex = dataGridRange.endColumnIndex as number;\n const dataSheetId = dataGridRange.sheetId ?? sheetId;\n\n // Build chart spec with properly split domain and series ranges\n let chartSpec: sheets_v4.Schema$ChartSpec;\n\n logger.info('Building chart spec', {\n chartType,\n dataGridRange,\n sheetId,\n });\n\n // Map legend position to API format (add _LEGEND suffix)\n const legendPositionMap: Record<string, string> = {\n BOTTOM: 'BOTTOM_LEGEND',\n RIGHT: 'RIGHT_LEGEND',\n TOP: 'TOP_LEGEND',\n LEFT: 'LEFT_LEGEND',\n NONE: 'NO_LEGEND',\n };\n const apiLegendPosition = legendPositionMap[legend] || 'BOTTOM_LEGEND';\n\n if (chartType === 'PIE') {\n // PIE charts use a separate pieChart property, not basicChart\n const baseSheetId = dataSheetId;\n\n const domainRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex,\n endColumnIndex: dataStartColumnIndex + 1, // First column only (labels)\n };\n\n const seriesRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex + 1, // Second column (values)\n endColumnIndex: dataStartColumnIndex + 2,\n };\n\n logger.info('PIE chart ranges', {\n dataGridRange,\n domainRange,\n seriesRange,\n apiLegendPosition,\n });\n\n chartSpec = {\n pieChart: {\n // Use pieChart for PIE charts\n legendPosition: legend === 'NONE' ? 'NO_LEGEND' : apiLegendPosition,\n domain: {\n // Note: domain/series are direct objects, not arrays\n sourceRange: {\n sources: [domainRange],\n },\n },\n series: {\n sourceRange: {\n sources: [seriesRange],\n },\n },\n threeDimensional: is3D, // 3D is directly on pieChart\n },\n };\n if (title) {\n chartSpec.title = title;\n }\n } else {\n // For BAR, COLUMN, LINE charts: domain is first column, each subsequent column is a series\n const baseSheetId = dataSheetId;\n\n const domainRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex,\n endColumnIndex: dataStartColumnIndex + 1, // First column only\n };\n\n // Calculate number of data columns (excluding the first domain column)\n const numDataColumns = dataEndColumnIndex - dataStartColumnIndex - 1;\n const series = [];\n\n // Create a series for each data column\n for (let i = 0; i < numDataColumns; i++) {\n const seriesRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex + 1 + i,\n endColumnIndex: dataStartColumnIndex + 2 + i,\n };\n\n series.push({\n series: {\n sourceRange: {\n sources: [seriesRange],\n },\n },\n });\n }\n\n // If no data columns were found, default to using the entire range as a fallback\n if (series.length === 0) {\n series.push({\n series: {\n sourceRange: {\n sources: [dataGridRange],\n },\n },\n });\n }\n\n const basicChart: sheets_v4.Schema$BasicChartSpec = {\n chartType, // BAR, COLUMN, LINE are valid for basicChart\n headerCount: 1, // First row is headers\n // Note: Google Sheets API does not support threeDimensional for basicChart (BAR/COLUMN/LINE)\n // threeDimensional is only supported for pieChart\n domains: [\n {\n domain: {\n sourceRange: {\n sources: [domainRange],\n },\n },\n },\n ],\n series,\n };\n if (legend !== 'NONE') {\n basicChart.legendPosition = apiLegendPosition;\n }\n\n chartSpec = { basicChart };\n if (title) {\n chartSpec.title = title;\n }\n }\n\n // Build embedded object position\n const embeddedObjectPosition = {\n overlayPosition: {\n anchorCell: {\n sheetId,\n rowIndex: anchorRowIndex,\n columnIndex: anchorColumnIndex,\n },\n offsetXPixels: position.offsetX,\n offsetYPixels: position.offsetY,\n },\n };\n\n const requestBody = {\n requests: [\n {\n addChart: {\n chart: {\n spec: chartSpec,\n position: embeddedObjectPosition,\n },\n },\n },\n ],\n };\n\n logger.info('sheets.chart.create executing addChart request', {\n spreadsheetId: id,\n sheetTitle,\n chartType,\n dataRange,\n anchorCell: position.anchorCell,\n requestBody: JSON.stringify(requestBody),\n });\n\n // Execute the addChart request\n const response = await sheets.spreadsheets.batchUpdate({\n spreadsheetId: id,\n requestBody,\n });\n\n // Extract chart ID from response\n const replies = response.data.replies || [];\n if (replies.length === 0 || !replies[0]?.addChart?.chart?.chartId) {\n logger.error('Chart creation failed - no chart ID returned', {\n spreadsheetId: id,\n sheetTitle,\n chartType,\n });\n throw new McpError(ErrorCode.InternalError, 'Chart creation failed: no chart ID returned from Google Sheets API');\n }\n\n const chartId = replies[0].addChart.chart.chartId;\n\n logger.info('sheets.chart.create completed successfully', {\n chartId,\n chartType,\n anchorCell: position.anchorCell,\n });\n\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetId),\n sheetTitle,\n sheetUrl,\n chartId,\n anchorCell: position.anchorCell,\n };\n\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (e) {\n const error = e as Error & { response?: { data?: unknown; status?: number } };\n const message = error.message || String(e);\n logger.error('Chart create operation failed', {\n id,\n gid,\n chartType,\n dataRange,\n error: message,\n response: error.response?.data,\n status: error.response?.status,\n });\n\n throw new McpError(ErrorCode.InternalError, `Error creating chart: ${message}`, {\n stack: e instanceof Error ? e.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'chart-create',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","ChartPositionSchema","z","object","anchorCell","string","min","describe","offsetX","number","int","default","offsetY","inputSchema","id","SpreadsheetIdSchema","gid","SheetGidSchema","chartType","enum","dataRange","title","optional","position","legend","is3D","boolean","successBranchSchema","type","literal","SpreadsheetIdOutput","SheetGidOutput","sheetTitle","sheetUrl","chartId","outputSchema","discriminatedUnion","config","description","result","parseAnchorCell","match","Error","colLetters","rowNumber","columnIndex","i","length","charCodeAt","rowIndex","Number","parseInt","handler","extra","logger","sheet","dataGridRange","spreadsheetResponse","replies","sheets","sheetId","anchorRowIndex","anchorColumnIndex","anchorIndices","message","rangeRef","dataStartRowIndex","dataEndRowIndex","dataStartColumnIndex","dataEndColumnIndex","dataSheetId","chartSpec","legendPositionMap","apiLegendPosition","baseSheetId","domainRange","seriesRange","numDataColumns","series","basicChart","embeddedObjectPosition","requestBody","response","e","error","info","google","version","auth","authContext","spreadsheets","get","spreadsheetId","fields","data","find","s","String","properties","McpError","ErrorCode","InvalidParams","parseA1Notation","rangeReferenceToGridRange","startColumnIndex","undefined","endColumnIndex","startRowIndex","endRowIndex","BOTTOM","RIGHT","TOP","LEFT","NONE","pieChart","legendPosition","domain","sourceRange","sources","threeDimensional","push","headerCount","domains","overlayPosition","offsetXPixels","offsetYPixels","requests","addChart","chart","spec","JSON","stringify","batchUpdate","InternalError","content","text","structuredContent","status","stack","name"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["/Users/kevin/Dev/Projects/mcp-z/mcp-sheets/src/mcp/tools/chart-create.ts"],"sourcesContent":["import type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { google, type sheets_v4 } from 'googleapis';\nimport { z } from 'zod';\nimport { SheetGidOutput, SheetGidSchema, SpreadsheetIdOutput, SpreadsheetIdSchema } from '../../schemas/index.ts';\nimport { parseA1Notation, rangeReferenceToGridRange } from '../../spreadsheet/range-operations.ts';\n\n// Input schema for chart position\nconst ChartPositionSchema = z.object({\n anchorCell: z.string().min(1).describe('A1 notation cell where chart top-left anchors (e.g., \"F2\")'),\n offsetX: z.number().int().default(0).describe('Horizontal pixel offset from anchor'),\n offsetY: z.number().int().default(0).describe('Vertical pixel offset from anchor'),\n});\n\nconst inputSchema = z.object({\n id: SpreadsheetIdSchema,\n gid: SheetGidSchema,\n chartType: z.enum(['PIE', 'BAR', 'COLUMN', 'LINE']).describe('Type of chart to create'),\n dataRange: z.string().min(1).describe('A1 notation range containing chart data including headers (e.g., \"A1:C10\")'),\n title: z.string().optional().describe('Chart title displayed at top'),\n position: ChartPositionSchema,\n legend: z.enum(['BOTTOM', 'RIGHT', 'TOP', 'LEFT', 'NONE']).default('BOTTOM').describe('Legend position'),\n is3D: z.boolean().default(false).describe('Render as 3D chart (PIE only)'),\n});\n\n// Success branch schema\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n id: SpreadsheetIdOutput,\n gid: SheetGidOutput,\n sheetTitle: z.string().describe('Title of the sheet containing the chart'),\n sheetUrl: z.string().describe('URL of the sheet containing the chart'),\n chartId: z.number().int().describe('Unique chart ID for future updates/deletion'),\n anchorCell: z.string().describe('Where chart was anchored'),\n});\n\n// Output schema with auth_required support\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Create charts (pie, bar, column, line) from spreadsheet data ranges. Charts anchor to specific cells with optional pixel offsets. Data range should include headers. PIE charts use 2 columns (labels, values). BAR/COLUMN/LINE charts use first row as headers. Best for visualizing spreadsheet data.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\n// Parse A1 notation anchor cell to row/column indices\nfunction parseAnchorCell(anchorCell: string): { rowIndex: number; columnIndex: number } {\n // Simple A1 notation parser for single cells\n const match = anchorCell.match(/^([A-Z]+)(\\d+)$/);\n if (!match || !match[1] || !match[2]) {\n throw new Error(`Invalid anchor cell format: ${anchorCell}`);\n }\n\n const colLetters = match[1];\n const rowNumber = match[2];\n\n // Convert column letters to index (A=0, B=1, ..., Z=25, AA=26, etc.)\n let columnIndex = 0;\n for (let i = 0; i < colLetters.length; i++) {\n columnIndex = columnIndex * 26 + (colLetters.charCodeAt(i) - 65 + 1);\n }\n columnIndex--; // Convert to 0-based\n\n // Convert row number to index (1-based to 0-based)\n const rowIndex = Number.parseInt(rowNumber, 10) - 1;\n\n return { rowIndex, columnIndex };\n}\n\nasync function handler({ id, gid, chartType, dataRange, title, position, legend = 'BOTTOM', is3D = false }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.info('sheets.chart.create called', {\n id,\n gid,\n chartType,\n dataRange,\n title,\n position,\n legend,\n is3D,\n });\n\n try {\n const sheets = google.sheets({ version: 'v4', auth: extra.authContext.auth });\n\n // Get spreadsheet and sheet info in single API call\n const spreadsheetResponse = await sheets.spreadsheets.get({\n spreadsheetId: id,\n fields: 'sheets.properties.sheetId,sheets.properties.title',\n });\n\n // Find sheet by gid\n const sheet = spreadsheetResponse.data.sheets?.find((s) => String(s.properties?.sheetId) === gid);\n if (!sheet?.properties) {\n logger.info('Sheet not found for chart create', { id, gid, chartType });\n throw new McpError(ErrorCode.InvalidParams, `Sheet not found: ${gid}`);\n }\n\n const sheetTitle = sheet.properties.title ?? gid;\n const sheetId = sheet.properties.sheetId;\n const sheetUrl = `https://docs.google.com/spreadsheets/d/${id}/edit#gid=${sheetId}`;\n\n // Validate 3D only for PIE charts\n // Note: Google Sheets API only supports 3D for PIE charts, not COLUMN/BAR/LINE\n if (is3D && chartType !== 'PIE') {\n logger.info('3D mode not supported for this chart type', {\n chartType,\n is3D,\n });\n throw new McpError(ErrorCode.InvalidParams, `3D mode is only supported for PIE charts, not ${chartType}`);\n }\n\n // Parse anchor cell to row/column indices\n let anchorRowIndex: number;\n let anchorColumnIndex: number;\n try {\n const anchorIndices = parseAnchorCell(position.anchorCell);\n anchorRowIndex = anchorIndices.rowIndex;\n anchorColumnIndex = anchorIndices.columnIndex;\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.info('Failed to parse anchor cell', {\n anchorCell: position.anchorCell,\n error: message,\n });\n throw new McpError(ErrorCode.InvalidParams, `Failed to parse anchor cell: ${message}`);\n }\n\n // Parse data range to grid range\n let dataGridRange: sheets_v4.Schema$GridRange;\n try {\n const rangeRef = parseA1Notation(dataRange);\n dataGridRange = rangeReferenceToGridRange(rangeRef, sheetId);\n\n // Validate that required properties are defined for chart creation\n if (dataGridRange.startColumnIndex === undefined || dataGridRange.startColumnIndex === null) {\n throw new Error('Data range must include column information');\n }\n if (dataGridRange.endColumnIndex === undefined || dataGridRange.endColumnIndex === null) {\n throw new Error('Data range must include column information');\n }\n if (dataGridRange.startRowIndex === undefined || dataGridRange.startRowIndex === null) {\n throw new Error('Data range must include row information');\n }\n if (dataGridRange.endRowIndex === undefined || dataGridRange.endRowIndex === null) {\n throw new Error('Data range must include row information');\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.info('Failed to parse data range', {\n dataRange,\n error: message,\n });\n throw new McpError(ErrorCode.InvalidParams, `Failed to parse data range: ${message}`);\n }\n\n // Extract validated properties with defined types\n const dataStartRowIndex = dataGridRange.startRowIndex as number;\n const dataEndRowIndex = dataGridRange.endRowIndex as number;\n const dataStartColumnIndex = dataGridRange.startColumnIndex as number;\n const dataEndColumnIndex = dataGridRange.endColumnIndex as number;\n const dataSheetId = dataGridRange.sheetId ?? sheetId;\n\n // Build chart spec with properly split domain and series ranges\n let chartSpec: sheets_v4.Schema$ChartSpec;\n\n logger.info('Building chart spec', {\n chartType,\n dataGridRange,\n sheetId,\n });\n\n // Map legend position to API format (add _LEGEND suffix)\n const legendPositionMap: Record<string, string> = {\n BOTTOM: 'BOTTOM_LEGEND',\n RIGHT: 'RIGHT_LEGEND',\n TOP: 'TOP_LEGEND',\n LEFT: 'LEFT_LEGEND',\n NONE: 'NO_LEGEND',\n };\n const apiLegendPosition = legendPositionMap[legend] || 'BOTTOM_LEGEND';\n\n if (chartType === 'PIE') {\n // PIE charts use a separate pieChart property, not basicChart\n const baseSheetId = dataSheetId;\n\n const domainRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex,\n endColumnIndex: dataStartColumnIndex + 1, // First column only (labels)\n };\n\n const seriesRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex + 1, // Second column (values)\n endColumnIndex: dataStartColumnIndex + 2,\n };\n\n logger.info('PIE chart ranges', {\n dataGridRange,\n domainRange,\n seriesRange,\n apiLegendPosition,\n });\n\n chartSpec = {\n pieChart: {\n // Use pieChart for PIE charts\n legendPosition: legend === 'NONE' ? 'NO_LEGEND' : apiLegendPosition,\n domain: {\n // Note: domain/series are direct objects, not arrays\n sourceRange: {\n sources: [domainRange],\n },\n },\n series: {\n sourceRange: {\n sources: [seriesRange],\n },\n },\n threeDimensional: is3D, // 3D is directly on pieChart\n },\n };\n if (title) {\n chartSpec.title = title;\n }\n } else {\n // For BAR, COLUMN, LINE charts: domain is first column, each subsequent column is a series\n const baseSheetId = dataSheetId;\n\n const domainRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex,\n endColumnIndex: dataStartColumnIndex + 1, // First column only\n };\n\n // Calculate number of data columns (excluding the first domain column)\n const numDataColumns = dataEndColumnIndex - dataStartColumnIndex - 1;\n const series = [];\n\n // Create a series for each data column\n for (let i = 0; i < numDataColumns; i++) {\n const seriesRange = {\n sheetId: baseSheetId,\n startRowIndex: dataStartRowIndex,\n endRowIndex: dataEndRowIndex,\n startColumnIndex: dataStartColumnIndex + 1 + i,\n endColumnIndex: dataStartColumnIndex + 2 + i,\n };\n\n series.push({\n series: {\n sourceRange: {\n sources: [seriesRange],\n },\n },\n });\n }\n\n // If no data columns were found, default to using the entire range as a fallback\n if (series.length === 0) {\n series.push({\n series: {\n sourceRange: {\n sources: [dataGridRange],\n },\n },\n });\n }\n\n const basicChart: sheets_v4.Schema$BasicChartSpec = {\n chartType, // BAR, COLUMN, LINE are valid for basicChart\n headerCount: 1, // First row is headers\n // Note: Google Sheets API does not support threeDimensional for basicChart (BAR/COLUMN/LINE)\n // threeDimensional is only supported for pieChart\n domains: [\n {\n domain: {\n sourceRange: {\n sources: [domainRange],\n },\n },\n },\n ],\n series,\n };\n if (legend !== 'NONE') {\n basicChart.legendPosition = apiLegendPosition;\n }\n\n chartSpec = { basicChart };\n if (title) {\n chartSpec.title = title;\n }\n }\n\n // Build embedded object position\n const embeddedObjectPosition = {\n overlayPosition: {\n anchorCell: {\n sheetId,\n rowIndex: anchorRowIndex,\n columnIndex: anchorColumnIndex,\n },\n offsetXPixels: position.offsetX,\n offsetYPixels: position.offsetY,\n },\n };\n\n const requestBody = {\n requests: [\n {\n addChart: {\n chart: {\n spec: chartSpec,\n position: embeddedObjectPosition,\n },\n },\n },\n ],\n };\n\n logger.info('sheets.chart.create executing addChart request', {\n spreadsheetId: id,\n sheetTitle,\n chartType,\n dataRange,\n anchorCell: position.anchorCell,\n requestBody: JSON.stringify(requestBody),\n });\n\n // Execute the addChart request\n const response = await sheets.spreadsheets.batchUpdate({\n spreadsheetId: id,\n requestBody,\n });\n\n // Extract chart ID from response\n const replies = response.data.replies || [];\n if (replies.length === 0 || !replies[0]?.addChart?.chart?.chartId) {\n logger.error('Chart creation failed - no chart ID returned', {\n spreadsheetId: id,\n sheetTitle,\n chartType,\n });\n throw new McpError(ErrorCode.InternalError, 'Chart creation failed: no chart ID returned from Google Sheets API');\n }\n\n const chartId = replies[0].addChart.chart.chartId;\n\n logger.info('sheets.chart.create completed successfully', {\n chartId,\n chartType,\n anchorCell: position.anchorCell,\n });\n\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetId),\n sheetTitle,\n sheetUrl,\n chartId,\n anchorCell: position.anchorCell,\n };\n\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (e) {\n const error = e as Error & { response?: { data?: unknown; status?: number } };\n const message = error.message || String(e);\n logger.error('Chart create operation failed', {\n id,\n gid,\n chartType,\n dataRange,\n error: message,\n response: error.response?.data,\n status: error.response?.status,\n });\n\n throw new McpError(ErrorCode.InternalError, `Error creating chart: ${message}`, {\n stack: e instanceof Error ? e.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'chart-create',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","ChartPositionSchema","z","object","anchorCell","string","min","describe","offsetX","number","int","default","offsetY","inputSchema","id","SpreadsheetIdSchema","gid","SheetGidSchema","chartType","enum","dataRange","title","optional","position","legend","is3D","boolean","successBranchSchema","type","literal","SpreadsheetIdOutput","SheetGidOutput","sheetTitle","sheetUrl","chartId","outputSchema","discriminatedUnion","config","description","result","parseAnchorCell","match","Error","colLetters","rowNumber","columnIndex","i","length","charCodeAt","rowIndex","Number","parseInt","handler","extra","logger","sheet","dataGridRange","spreadsheetResponse","replies","sheets","sheetId","anchorRowIndex","anchorColumnIndex","anchorIndices","message","rangeRef","dataStartRowIndex","dataEndRowIndex","dataStartColumnIndex","dataEndColumnIndex","dataSheetId","chartSpec","legendPositionMap","apiLegendPosition","baseSheetId","domainRange","seriesRange","numDataColumns","series","basicChart","embeddedObjectPosition","requestBody","response","e","error","info","google","version","auth","authContext","spreadsheets","get","spreadsheetId","fields","data","find","s","String","properties","McpError","ErrorCode","InvalidParams","parseA1Notation","rangeReferenceToGridRange","startColumnIndex","undefined","endColumnIndex","startRowIndex","endRowIndex","BOTTOM","RIGHT","TOP","LEFT","NONE","pieChart","legendPosition","domain","sourceRange","sources","threeDimensional","push","headerCount","domains","overlayPosition","offsetXPixels","offsetYPixels","requests","addChart","chart","spec","JSON","stringify","batchUpdate","InternalError","content","text","structuredContent","status","stack","name"],"mappings":"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"}
|
|
@@ -12,7 +12,7 @@ var _oauthgoogle = require("@mcp-z/oauth-google");
|
|
|
12
12
|
var _types = require("@modelcontextprotocol/sdk/types.js");
|
|
13
13
|
var _googleapis = require("googleapis");
|
|
14
14
|
var _zod = require("zod");
|
|
15
|
-
var
|
|
15
|
+
var _indexts = require("../../schemas/index.js");
|
|
16
16
|
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
|
|
17
17
|
try {
|
|
18
18
|
var info = gen[key](arg);
|
|
@@ -150,8 +150,8 @@ function _ts_generator(thisArg, body) {
|
|
|
150
150
|
}
|
|
151
151
|
var AuthRequiredBranchSchema = _oauthgoogle.schemas.AuthRequiredBranchSchema;
|
|
152
152
|
var inputSchema = _zod.z.object({
|
|
153
|
-
id:
|
|
154
|
-
gid:
|
|
153
|
+
id: _indexts.SpreadsheetIdSchema,
|
|
154
|
+
gid: _indexts.SheetGidSchema,
|
|
155
155
|
render: _zod.z.enum([
|
|
156
156
|
'FORMATTED_VALUE',
|
|
157
157
|
'UNFORMATTED_VALUE',
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["/Users/kevin/Dev/Projects/
|
|
1
|
+
{"version":3,"sources":["/Users/kevin/Dev/Projects/mcp-z/mcp-sheets/src/mcp/tools/columns-get.ts"],"sourcesContent":["/** Get column names from Google Sheet (peek at first row only) */\n\nimport type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { google } from 'googleapis';\nimport { z } from 'zod';\nimport { SheetGidSchema, SpreadsheetIdSchema } from '../../schemas/index.ts';\n\nconst inputSchema = z.object({\n id: SpreadsheetIdSchema,\n gid: SheetGidSchema,\n render: z.enum(['FORMATTED_VALUE', 'UNFORMATTED_VALUE', 'FORMULA']).optional().describe('How to render cell values. FORMATTED_VALUE (default): calculated with formatting. UNFORMATTED_VALUE: calculated without formatting. FORMULA: show formula text instead of result.'),\n});\n\n// Success branch schema - uses columns: for consistency with standard vocabulary\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n columns: z.array(z.string()).describe('First row values (column names) or empty if no rows'),\n isEmpty: z.boolean().describe('True if sheet has zero rows'),\n});\n\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Get first row from Google Sheet. Returns columns array and isEmpty flag.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\nasync function handler({ id, gid, render }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.debug?.('sheets.columns.get called', { id, gid, render });\n\n try {\n const sheets = google.sheets({ version: 'v4', auth: extra.authContext.auth });\n\n // Get spreadsheet and sheet info in single API call\n const spreadsheetResponse = await sheets.spreadsheets.get({\n spreadsheetId: id,\n fields: 'sheets.properties.sheetId,sheets.properties.title',\n });\n\n // Find sheet by gid\n const sheet = spreadsheetResponse.data.sheets?.find((s) => String(s.properties?.sheetId) === gid);\n if (!sheet?.properties) {\n throw new McpError(ErrorCode.InvalidParams, `Sheet not found: ${gid}`);\n }\n\n const sheetTitle = sheet.properties.title ?? '';\n\n // Read first row only (A1:ZZZ1 should cover most reasonable sheets)\n const range = `'${sheetTitle}'!A1:ZZZ1`;\n const response = await sheets.spreadsheets.values.get({\n spreadsheetId: id,\n range,\n valueRenderOption: render || 'FORMATTED_VALUE',\n });\n\n const rows = response.data.values || [];\n const isEmpty = rows.length === 0;\n\n // Extract first row and convert to strings (column names)\n const firstRow = rows[0] || [];\n const columns = firstRow.map((value) => String(value ?? ''));\n\n const result: Output = {\n type: 'success' as const,\n columns,\n isEmpty,\n };\n\n logger.info?.('sheets.columns.get completed', { id, gid, columnCount: columns.length, isEmpty });\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.error?.('sheets.columns.get error', { error: message });\n throw new McpError(ErrorCode.InternalError, `Error getting sheet columns: ${message}`, {\n stack: error instanceof Error ? error.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'columns-get',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","inputSchema","z","object","id","SpreadsheetIdSchema","gid","SheetGidSchema","render","enum","optional","describe","successBranchSchema","type","literal","columns","array","string","isEmpty","boolean","outputSchema","discriminatedUnion","config","description","result","handler","extra","logger","sheet","spreadsheetResponse","sheets","sheetTitle","range","response","rows","firstRow","error","message","debug","google","version","auth","authContext","spreadsheets","get","spreadsheetId","fields","data","find","s","String","properties","sheetId","McpError","ErrorCode","InvalidParams","title","values","valueRenderOption","length","map","value","info","columnCount","content","text","JSON","stringify","structuredContent","Error","InternalError","stack","undefined","name"],"mappings":"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"}
|
|
@@ -12,8 +12,8 @@ var _oauthgoogle = require("@mcp-z/oauth-google");
|
|
|
12
12
|
var _types = require("@modelcontextprotocol/sdk/types.js");
|
|
13
13
|
var _googleapis = require("googleapis");
|
|
14
14
|
var _zod = require("zod");
|
|
15
|
-
var
|
|
16
|
-
var
|
|
15
|
+
var _indexts = require("../../schemas/index.js");
|
|
16
|
+
var _dataoperationsts = require("../../spreadsheet/data-operations.js");
|
|
17
17
|
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
|
|
18
18
|
try {
|
|
19
19
|
var info = gen[key](arg);
|
|
@@ -152,9 +152,9 @@ function _ts_generator(thisArg, body) {
|
|
|
152
152
|
var AuthRequiredBranchSchema = _oauthgoogle.schemas.AuthRequiredBranchSchema;
|
|
153
153
|
// Input schema for columns update requests with enhanced validation
|
|
154
154
|
var inputSchema = _zod.z.object({
|
|
155
|
-
id:
|
|
156
|
-
gid:
|
|
157
|
-
rows: _zod.z.array(_zod.z.array(
|
|
155
|
+
id: _indexts.SpreadsheetIdSchema,
|
|
156
|
+
gid: _indexts.SheetGidSchema,
|
|
157
|
+
rows: _zod.z.array(_zod.z.array(_indexts.SheetCellSchema)).min(1).max(1000).describe('Array of rows to upsert (max 1000 rows per request). Each row is an array of cell values matching the headers array'),
|
|
158
158
|
headers: _zod.z.array(_zod.z.string().min(1).max(100)).min(1).max(50).describe('Array of column names/headers (max 50 columns). Must match the length of each row in the rows array'),
|
|
159
159
|
updateBy: _zod.z.array(_zod.z.string().min(1).max(100)).min(1).max(10).describe('Array of column names to use as unique keys for matching existing rows (max 10 key columns). These columns must exist in the headers array'),
|
|
160
160
|
behavior: _zod.z.enum([
|
|
@@ -201,8 +201,8 @@ var inputSchema = _zod.z.object({
|
|
|
201
201
|
// Success branch schema
|
|
202
202
|
var successBranchSchema = _zod.z.object({
|
|
203
203
|
type: _zod.z.literal('success'),
|
|
204
|
-
id:
|
|
205
|
-
gid:
|
|
204
|
+
id: _indexts.SpreadsheetIdOutput,
|
|
205
|
+
gid: _indexts.SheetGidOutput,
|
|
206
206
|
spreadsheetTitle: _zod.z.string().describe('Title of the updated spreadsheet'),
|
|
207
207
|
spreadsheetUrl: _zod.z.string().describe('URL of the updated spreadsheet'),
|
|
208
208
|
sheetTitle: _zod.z.string().describe('Title of the updated sheet'),
|
|
@@ -226,7 +226,7 @@ var config = {
|
|
|
226
226
|
};
|
|
227
227
|
// Helper function to ensure consistent key generation
|
|
228
228
|
function generateConsistentRowKey(row, headers, strategy) {
|
|
229
|
-
return (0,
|
|
229
|
+
return (0, _dataoperationsts.generateRowKey)(row, headers, strategy);
|
|
230
230
|
}
|
|
231
231
|
function handler(_0, _1) {
|
|
232
232
|
return _async_to_generator(function(param, extra) {
|
|
@@ -323,7 +323,7 @@ function handler(_0, _1) {
|
|
|
323
323
|
};
|
|
324
324
|
return [
|
|
325
325
|
4,
|
|
326
|
-
(0,
|
|
326
|
+
(0, _dataoperationsts.snapshotHeaderKeysAndPositions)(sheets, id, sheetTitle, updateBy, upsertOptions.keyStrategy)
|
|
327
327
|
];
|
|
328
328
|
case 3:
|
|
329
329
|
_ref1 = _state.sent(), existingKeys = _ref1.keySet;
|
|
@@ -402,7 +402,7 @@ function handler(_0, _1) {
|
|
|
402
402
|
});
|
|
403
403
|
return [
|
|
404
404
|
4,
|
|
405
|
-
(0,
|
|
405
|
+
(0, _dataoperationsts.upsertByKey)(sheets, {
|
|
406
406
|
spreadsheetId: id,
|
|
407
407
|
sheetTitle: sheetTitle,
|
|
408
408
|
rows: processedRows,
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["/Users/kevin/Dev/Projects/ai/mcp-z/servers/mcp-sheets/src/mcp/tools/columns-update.ts"],"sourcesContent":["import type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { google } from 'googleapis';\nimport { z } from 'zod';\nimport { SheetCellSchema, SheetGidOutput, SheetGidSchema, SpreadsheetIdOutput, SpreadsheetIdSchema } from '../../schemas/index.js';\nimport { generateRowKey, type KeyGenerationStrategy, type Row, snapshotHeaderKeysAndPositions, type UpsertOptions, upsertByKey } from '../../spreadsheet/data-operations.js';\n\n// Input schema for columns update requests with enhanced validation\nconst inputSchema = z\n .object({\n id: SpreadsheetIdSchema,\n gid: SheetGidSchema,\n rows: z.array(z.array(SheetCellSchema)).min(1).max(1000).describe('Array of rows to upsert (max 1000 rows per request). Each row is an array of cell values matching the headers array'),\n headers: z.array(z.string().min(1).max(100)).min(1).max(50).describe('Array of column names/headers (max 50 columns). Must match the length of each row in the rows array'),\n updateBy: z.array(z.string().min(1).max(100)).min(1).max(10).describe('Array of column names to use as unique keys for matching existing rows (max 10 key columns). These columns must exist in the headers array'),\n behavior: z\n .enum(['add-or-update', 'update-only', 'add-only'])\n .default('add-or-update')\n .describe('Update behavior: add-or-update (default) adds new rows and updates existing, update-only skips new rows, add-only skips existing rows. BEST FOR: Bulk upsert operations with key matching in structured database or table contexts.'),\n valueInputOption: z.enum(['RAW', 'USER_ENTERED']).default('USER_ENTERED').describe('How input data should be interpreted (RAW = exact values, USER_ENTERED = parsed like user input with formulas, dates, etc.)'),\n })\n .refine(\n (data) => {\n // Validate that all updateBy columns exist in headers\n const missingColumns = data.updateBy.filter((col) => !data.headers.includes(col));\n return missingColumns.length === 0;\n },\n {\n message: 'All updateBy columns must exist in the headers array',\n path: ['updateBy'],\n }\n )\n .refine(\n (data) => {\n // Validate that all rows have the same length as headers\n const invalidRows = data.rows.findIndex((row) => row.length !== data.headers.length);\n return invalidRows === -1;\n },\n {\n message: 'All rows must have the same length as the headers array',\n path: ['rows'],\n }\n )\n .refine(\n (data) => {\n // Validate that updateBy columns are unique\n const uniqueColumns = new Set(data.updateBy);\n return uniqueColumns.size === data.updateBy.length;\n },\n {\n message: 'updateBy columns must be unique',\n path: ['updateBy'],\n }\n );\n\n// Success branch schema\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n id: SpreadsheetIdOutput,\n gid: SheetGidOutput,\n spreadsheetTitle: z.string().describe('Title of the updated spreadsheet'),\n spreadsheetUrl: z.string().describe('URL of the updated spreadsheet'),\n sheetTitle: z.string().describe('Title of the updated sheet'),\n sheetUrl: z.string().describe('URL of the updated sheet'),\n updatedRows: z.number().int().nonnegative().describe('Number of rows that were successfully updated or inserted'),\n insertedKeys: z.array(z.string()).describe('Keys of rows that were successfully inserted (new rows)'),\n rowsSkipped: z.number().int().nonnegative().describe('Number of rows that were skipped based on the update behavior and existing data'),\n headersAdded: z.array(z.string()).describe('Column headers that were added to the sheet (if any were missing)'),\n errors: z.array(z.string()).optional().describe('Any non-fatal errors encountered during the operation'),\n});\n\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Update spreadsheet data by column headers with intelligent upsert logic. Supports adding missing columns, flexible update behaviors, and robust error handling. Uses column names as keys for matching existing rows, enabling context-aware data synchronization workflows.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\n// Helper function to ensure consistent key generation\nfunction generateConsistentRowKey(row: Row, headers: string[], strategy: KeyGenerationStrategy): string {\n return generateRowKey(row, headers, strategy);\n}\n\nasync function handler({ id, gid, rows, headers, updateBy, behavior = 'add-or-update', valueInputOption = 'USER_ENTERED' }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.debug?.('sheets.columns.update called', {\n id,\n gid,\n rowCount: rows.length,\n headerCount: headers.length,\n updateByColumns: updateBy,\n behavior,\n valueInputOption,\n });\n\n try {\n const sheets = google.sheets({ version: 'v4', auth: extra.authContext.auth });\n\n // Note: Basic input validation is now handled by the enhanced Zod schema\n\n // EARLY VALIDATION: Validate that updateBy columns have non-empty values for all rows\n // This validation happens before finding spreadsheet/sheet to fail fast\n const emptyKeyErrors: string[] = [];\n rows.forEach((row, rowIndex) => {\n const keyValues = updateBy.map((keyCol) => {\n const colIndex = headers.indexOf(keyCol);\n return colIndex >= 0 ? String(row[colIndex] ?? '').trim() : '';\n });\n\n if (keyValues.some((val) => val === '')) {\n emptyKeyErrors.push(`Row ${rowIndex + 1} has empty key values for columns: ${updateBy.join(', ')}`);\n }\n });\n\n if (emptyKeyErrors.length > 0) {\n const message = `Silent data loss prevented - empty key columns detected:\\n${emptyKeyErrors.join('\\n')}`;\n logger.error?.('sheets.columns.update error', { error: message });\n throw new McpError(ErrorCode.InvalidParams, message);\n }\n\n // Get spreadsheet and sheet info in single API call\n const spreadsheetResponse = await sheets.spreadsheets.get({\n spreadsheetId: id,\n fields: 'properties.title,spreadsheetUrl,sheets.properties.sheetId,sheets.properties.title',\n });\n\n const spreadsheetData = spreadsheetResponse.data;\n const spreadsheetTitle = spreadsheetData.properties?.title ?? '';\n const spreadsheetUrl = spreadsheetData.spreadsheetUrl ?? '';\n\n // Find sheet by gid\n const sheet = spreadsheetData.sheets?.find((s) => String(s.properties?.sheetId) === gid);\n if (!sheet?.properties) {\n logger.warn?.('Sheet not found for columns update', { id, gid, rowCount: rows.length });\n throw new McpError(ErrorCode.InvalidParams, `Sheet not found: ${gid}`);\n }\n\n const sheetTitle = sheet.properties.title ?? gid;\n const sheetGid = sheet.properties.sheetId ?? 0;\n const sheetUrl = `https://docs.google.com/spreadsheets/d/${id}/edit#gid=${sheetGid}`;\n\n // Configure upsert options based on behavior\n const upsertOptions: UpsertOptions = {\n keyStrategy: {\n keyColumns: updateBy,\n useProviderIdLogic: false, // Use standard key generation for column-based updates\n separator: '\\\\',\n },\n allowUpdates: behavior === 'add-or-update' || behavior === 'update-only',\n batchSize: 50,\n valueInputOption: valueInputOption,\n };\n\n // Get existing keys to implement behavior filtering\n const { keySet: existingKeys } = await snapshotHeaderKeysAndPositions(sheets, id, sheetTitle, updateBy, upsertOptions.keyStrategy);\n\n // Filter rows based on behavior using consistent key generation\n let processedRows = rows;\n if (behavior === 'add-only') {\n // Only include rows that don't exist yet\n processedRows = rows.filter((row) => {\n // Use the same key generation strategy as the upsert function\n const key = generateConsistentRowKey(row, headers, upsertOptions.keyStrategy);\n return key !== '' && !existingKeys.has(key);\n });\n logger.debug?.('Filtered rows for add-only behavior', {\n originalCount: rows.length,\n filteredCount: processedRows.length,\n });\n } else if (behavior === 'update-only') {\n // Only include rows that already exist\n processedRows = rows.filter((row) => {\n // Use the same key generation strategy as the upsert function\n const key = generateConsistentRowKey(row, headers, upsertOptions.keyStrategy);\n return key !== '' && existingKeys.has(key);\n });\n logger.debug?.('Filtered rows for update-only behavior', {\n originalCount: rows.length,\n filteredCount: processedRows.length,\n });\n }\n\n // Early return if no rows to process after behavior filtering\n if (processedRows.length === 0) {\n logger.info?.('No rows to process after behavior filtering', { behavior, originalCount: rows.length });\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetGid),\n spreadsheetTitle: spreadsheetTitle || '',\n spreadsheetUrl: spreadsheetUrl || '',\n sheetTitle,\n sheetUrl,\n updatedRows: 0,\n insertedKeys: [],\n rowsSkipped: rows.length,\n headersAdded: [],\n errors: [`No rows matched behavior '${behavior}' - all ${rows.length} rows were skipped`],\n };\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n }\n\n logger.debug?.('sheets.columns.update executing upsert operation', {\n spreadsheetId: id,\n sheetTitle,\n canonicalHeaders: headers,\n keyColumns: updateBy,\n behavior,\n processedRowsCount: processedRows.length,\n });\n\n // Execute the upsert operation using the shared function\n const upsertResult = await upsertByKey(sheets, {\n spreadsheetId: id,\n sheetTitle,\n rows: processedRows,\n canonicalHeaders: headers,\n options: upsertOptions,\n logger,\n });\n\n // Track headers that were added (if any)\n // This would require additional logic in upsertByKey to return added headers\n const headersAdded: string[] = [];\n\n // Calculate total rows skipped: behavior filter + upsert duplicates\n const rowsFilteredByBehavior = rows.length - processedRows.length;\n const totalRowsSkipped = upsertResult.rowsSkipped + rowsFilteredByBehavior;\n\n logger.debug?.('sheets.columns.update completed successfully', {\n updatedRows: upsertResult.updatedRows,\n insertedCount: upsertResult.inserted.length,\n rowsSkipped: totalRowsSkipped,\n rowsFilteredByBehavior,\n rowsSkippedByUpsert: upsertResult.rowsSkipped,\n errorsCount: upsertResult.errors?.length || 0,\n });\n\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetGid),\n spreadsheetTitle: spreadsheetTitle || '',\n spreadsheetUrl: spreadsheetUrl || '',\n sheetTitle,\n sheetUrl,\n updatedRows: upsertResult.updatedRows,\n insertedKeys: upsertResult.inserted,\n rowsSkipped: totalRowsSkipped,\n headersAdded,\n errors: upsertResult.errors,\n };\n\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.error?.('sheets.columns.update error', { error: message });\n throw new McpError(ErrorCode.InternalError, `Error updating columns: ${message}`, {\n stack: error instanceof Error ? error.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'columns-update',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","inputSchema","z","object","id","SpreadsheetIdSchema","gid","SheetGidSchema","rows","array","SheetCellSchema","min","max","describe","headers","string","updateBy","behavior","enum","default","valueInputOption","refine","data","missingColumns","filter","col","includes","length","message","path","invalidRows","findIndex","row","uniqueColumns","Set","size","successBranchSchema","type","literal","SpreadsheetIdOutput","SheetGidOutput","spreadsheetTitle","spreadsheetUrl","sheetTitle","sheetUrl","updatedRows","number","int","nonnegative","insertedKeys","rowsSkipped","headersAdded","errors","optional","outputSchema","discriminatedUnion","config","description","result","generateConsistentRowKey","strategy","generateRowKey","handler","extra","logger","spreadsheetData","sheet","upsertResult","sheets","emptyKeyErrors","spreadsheetResponse","sheetGid","upsertOptions","existingKeys","processedRows","rowsFilteredByBehavior","totalRowsSkipped","error","debug","rowCount","headerCount","updateByColumns","google","version","auth","authContext","forEach","rowIndex","keyValues","map","keyCol","colIndex","indexOf","String","trim","some","val","push","join","McpError","ErrorCode","InvalidParams","spreadsheets","get","spreadsheetId","fields","properties","title","find","s","sheetId","warn","keyStrategy","keyColumns","useProviderIdLogic","separator","allowUpdates","batchSize","snapshotHeaderKeysAndPositions","keySet","key","has","originalCount","filteredCount","info","content","text","JSON","stringify","structuredContent","canonicalHeaders","processedRowsCount","upsertByKey","options","insertedCount","inserted","rowsSkippedByUpsert","errorsCount","Error","InternalError","stack","undefined","name"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["/Users/kevin/Dev/Projects/mcp-z/mcp-sheets/src/mcp/tools/columns-update.ts"],"sourcesContent":["import type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { google } from 'googleapis';\nimport { z } from 'zod';\nimport { SheetCellSchema, SheetGidOutput, SheetGidSchema, SpreadsheetIdOutput, SpreadsheetIdSchema } from '../../schemas/index.ts';\nimport { generateRowKey, type KeyGenerationStrategy, type Row, snapshotHeaderKeysAndPositions, type UpsertOptions, upsertByKey } from '../../spreadsheet/data-operations.ts';\n\n// Input schema for columns update requests with enhanced validation\nconst inputSchema = z\n .object({\n id: SpreadsheetIdSchema,\n gid: SheetGidSchema,\n rows: z.array(z.array(SheetCellSchema)).min(1).max(1000).describe('Array of rows to upsert (max 1000 rows per request). Each row is an array of cell values matching the headers array'),\n headers: z.array(z.string().min(1).max(100)).min(1).max(50).describe('Array of column names/headers (max 50 columns). Must match the length of each row in the rows array'),\n updateBy: z.array(z.string().min(1).max(100)).min(1).max(10).describe('Array of column names to use as unique keys for matching existing rows (max 10 key columns). These columns must exist in the headers array'),\n behavior: z\n .enum(['add-or-update', 'update-only', 'add-only'])\n .default('add-or-update')\n .describe('Update behavior: add-or-update (default) adds new rows and updates existing, update-only skips new rows, add-only skips existing rows. BEST FOR: Bulk upsert operations with key matching in structured database or table contexts.'),\n valueInputOption: z.enum(['RAW', 'USER_ENTERED']).default('USER_ENTERED').describe('How input data should be interpreted (RAW = exact values, USER_ENTERED = parsed like user input with formulas, dates, etc.)'),\n })\n .refine(\n (data) => {\n // Validate that all updateBy columns exist in headers\n const missingColumns = data.updateBy.filter((col) => !data.headers.includes(col));\n return missingColumns.length === 0;\n },\n {\n message: 'All updateBy columns must exist in the headers array',\n path: ['updateBy'],\n }\n )\n .refine(\n (data) => {\n // Validate that all rows have the same length as headers\n const invalidRows = data.rows.findIndex((row) => row.length !== data.headers.length);\n return invalidRows === -1;\n },\n {\n message: 'All rows must have the same length as the headers array',\n path: ['rows'],\n }\n )\n .refine(\n (data) => {\n // Validate that updateBy columns are unique\n const uniqueColumns = new Set(data.updateBy);\n return uniqueColumns.size === data.updateBy.length;\n },\n {\n message: 'updateBy columns must be unique',\n path: ['updateBy'],\n }\n );\n\n// Success branch schema\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n id: SpreadsheetIdOutput,\n gid: SheetGidOutput,\n spreadsheetTitle: z.string().describe('Title of the updated spreadsheet'),\n spreadsheetUrl: z.string().describe('URL of the updated spreadsheet'),\n sheetTitle: z.string().describe('Title of the updated sheet'),\n sheetUrl: z.string().describe('URL of the updated sheet'),\n updatedRows: z.number().int().nonnegative().describe('Number of rows that were successfully updated or inserted'),\n insertedKeys: z.array(z.string()).describe('Keys of rows that were successfully inserted (new rows)'),\n rowsSkipped: z.number().int().nonnegative().describe('Number of rows that were skipped based on the update behavior and existing data'),\n headersAdded: z.array(z.string()).describe('Column headers that were added to the sheet (if any were missing)'),\n errors: z.array(z.string()).optional().describe('Any non-fatal errors encountered during the operation'),\n});\n\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Update spreadsheet data by column headers with intelligent upsert logic. Supports adding missing columns, flexible update behaviors, and robust error handling. Uses column names as keys for matching existing rows, enabling context-aware data synchronization workflows.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\n// Helper function to ensure consistent key generation\nfunction generateConsistentRowKey(row: Row, headers: string[], strategy: KeyGenerationStrategy): string {\n return generateRowKey(row, headers, strategy);\n}\n\nasync function handler({ id, gid, rows, headers, updateBy, behavior = 'add-or-update', valueInputOption = 'USER_ENTERED' }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.debug?.('sheets.columns.update called', {\n id,\n gid,\n rowCount: rows.length,\n headerCount: headers.length,\n updateByColumns: updateBy,\n behavior,\n valueInputOption,\n });\n\n try {\n const sheets = google.sheets({ version: 'v4', auth: extra.authContext.auth });\n\n // Note: Basic input validation is now handled by the enhanced Zod schema\n\n // EARLY VALIDATION: Validate that updateBy columns have non-empty values for all rows\n // This validation happens before finding spreadsheet/sheet to fail fast\n const emptyKeyErrors: string[] = [];\n rows.forEach((row, rowIndex) => {\n const keyValues = updateBy.map((keyCol) => {\n const colIndex = headers.indexOf(keyCol);\n return colIndex >= 0 ? String(row[colIndex] ?? '').trim() : '';\n });\n\n if (keyValues.some((val) => val === '')) {\n emptyKeyErrors.push(`Row ${rowIndex + 1} has empty key values for columns: ${updateBy.join(', ')}`);\n }\n });\n\n if (emptyKeyErrors.length > 0) {\n const message = `Silent data loss prevented - empty key columns detected:\\n${emptyKeyErrors.join('\\n')}`;\n logger.error?.('sheets.columns.update error', { error: message });\n throw new McpError(ErrorCode.InvalidParams, message);\n }\n\n // Get spreadsheet and sheet info in single API call\n const spreadsheetResponse = await sheets.spreadsheets.get({\n spreadsheetId: id,\n fields: 'properties.title,spreadsheetUrl,sheets.properties.sheetId,sheets.properties.title',\n });\n\n const spreadsheetData = spreadsheetResponse.data;\n const spreadsheetTitle = spreadsheetData.properties?.title ?? '';\n const spreadsheetUrl = spreadsheetData.spreadsheetUrl ?? '';\n\n // Find sheet by gid\n const sheet = spreadsheetData.sheets?.find((s) => String(s.properties?.sheetId) === gid);\n if (!sheet?.properties) {\n logger.warn?.('Sheet not found for columns update', { id, gid, rowCount: rows.length });\n throw new McpError(ErrorCode.InvalidParams, `Sheet not found: ${gid}`);\n }\n\n const sheetTitle = sheet.properties.title ?? gid;\n const sheetGid = sheet.properties.sheetId ?? 0;\n const sheetUrl = `https://docs.google.com/spreadsheets/d/${id}/edit#gid=${sheetGid}`;\n\n // Configure upsert options based on behavior\n const upsertOptions: UpsertOptions = {\n keyStrategy: {\n keyColumns: updateBy,\n useProviderIdLogic: false, // Use standard key generation for column-based updates\n separator: '\\\\',\n },\n allowUpdates: behavior === 'add-or-update' || behavior === 'update-only',\n batchSize: 50,\n valueInputOption: valueInputOption,\n };\n\n // Get existing keys to implement behavior filtering\n const { keySet: existingKeys } = await snapshotHeaderKeysAndPositions(sheets, id, sheetTitle, updateBy, upsertOptions.keyStrategy);\n\n // Filter rows based on behavior using consistent key generation\n let processedRows = rows;\n if (behavior === 'add-only') {\n // Only include rows that don't exist yet\n processedRows = rows.filter((row) => {\n // Use the same key generation strategy as the upsert function\n const key = generateConsistentRowKey(row, headers, upsertOptions.keyStrategy);\n return key !== '' && !existingKeys.has(key);\n });\n logger.debug?.('Filtered rows for add-only behavior', {\n originalCount: rows.length,\n filteredCount: processedRows.length,\n });\n } else if (behavior === 'update-only') {\n // Only include rows that already exist\n processedRows = rows.filter((row) => {\n // Use the same key generation strategy as the upsert function\n const key = generateConsistentRowKey(row, headers, upsertOptions.keyStrategy);\n return key !== '' && existingKeys.has(key);\n });\n logger.debug?.('Filtered rows for update-only behavior', {\n originalCount: rows.length,\n filteredCount: processedRows.length,\n });\n }\n\n // Early return if no rows to process after behavior filtering\n if (processedRows.length === 0) {\n logger.info?.('No rows to process after behavior filtering', { behavior, originalCount: rows.length });\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetGid),\n spreadsheetTitle: spreadsheetTitle || '',\n spreadsheetUrl: spreadsheetUrl || '',\n sheetTitle,\n sheetUrl,\n updatedRows: 0,\n insertedKeys: [],\n rowsSkipped: rows.length,\n headersAdded: [],\n errors: [`No rows matched behavior '${behavior}' - all ${rows.length} rows were skipped`],\n };\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n }\n\n logger.debug?.('sheets.columns.update executing upsert operation', {\n spreadsheetId: id,\n sheetTitle,\n canonicalHeaders: headers,\n keyColumns: updateBy,\n behavior,\n processedRowsCount: processedRows.length,\n });\n\n // Execute the upsert operation using the shared function\n const upsertResult = await upsertByKey(sheets, {\n spreadsheetId: id,\n sheetTitle,\n rows: processedRows,\n canonicalHeaders: headers,\n options: upsertOptions,\n logger,\n });\n\n // Track headers that were added (if any)\n // This would require additional logic in upsertByKey to return added headers\n const headersAdded: string[] = [];\n\n // Calculate total rows skipped: behavior filter + upsert duplicates\n const rowsFilteredByBehavior = rows.length - processedRows.length;\n const totalRowsSkipped = upsertResult.rowsSkipped + rowsFilteredByBehavior;\n\n logger.debug?.('sheets.columns.update completed successfully', {\n updatedRows: upsertResult.updatedRows,\n insertedCount: upsertResult.inserted.length,\n rowsSkipped: totalRowsSkipped,\n rowsFilteredByBehavior,\n rowsSkippedByUpsert: upsertResult.rowsSkipped,\n errorsCount: upsertResult.errors?.length || 0,\n });\n\n const result: Output = {\n type: 'success' as const,\n id,\n gid: String(sheetGid),\n spreadsheetTitle: spreadsheetTitle || '',\n spreadsheetUrl: spreadsheetUrl || '',\n sheetTitle,\n sheetUrl,\n updatedRows: upsertResult.updatedRows,\n insertedKeys: upsertResult.inserted,\n rowsSkipped: totalRowsSkipped,\n headersAdded,\n errors: upsertResult.errors,\n };\n\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.error?.('sheets.columns.update error', { error: message });\n throw new McpError(ErrorCode.InternalError, `Error updating columns: ${message}`, {\n stack: error instanceof Error ? error.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'columns-update',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","inputSchema","z","object","id","SpreadsheetIdSchema","gid","SheetGidSchema","rows","array","SheetCellSchema","min","max","describe","headers","string","updateBy","behavior","enum","default","valueInputOption","refine","data","missingColumns","filter","col","includes","length","message","path","invalidRows","findIndex","row","uniqueColumns","Set","size","successBranchSchema","type","literal","SpreadsheetIdOutput","SheetGidOutput","spreadsheetTitle","spreadsheetUrl","sheetTitle","sheetUrl","updatedRows","number","int","nonnegative","insertedKeys","rowsSkipped","headersAdded","errors","optional","outputSchema","discriminatedUnion","config","description","result","generateConsistentRowKey","strategy","generateRowKey","handler","extra","logger","spreadsheetData","sheet","upsertResult","sheets","emptyKeyErrors","spreadsheetResponse","sheetGid","upsertOptions","existingKeys","processedRows","rowsFilteredByBehavior","totalRowsSkipped","error","debug","rowCount","headerCount","updateByColumns","google","version","auth","authContext","forEach","rowIndex","keyValues","map","keyCol","colIndex","indexOf","String","trim","some","val","push","join","McpError","ErrorCode","InvalidParams","spreadsheets","get","spreadsheetId","fields","properties","title","find","s","sheetId","warn","keyStrategy","keyColumns","useProviderIdLogic","separator","allowUpdates","batchSize","snapshotHeaderKeysAndPositions","keySet","key","has","originalCount","filteredCount","info","content","text","JSON","stringify","structuredContent","canonicalHeaders","processedRowsCount","upsertByKey","options","insertedCount","inserted","rowsSkippedByUpsert","errorsCount","Error","InternalError","stack","undefined","name"],"mappings":"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"}
|
|
@@ -12,7 +12,7 @@ var _oauthgoogle = require("@mcp-z/oauth-google");
|
|
|
12
12
|
var _types = require("@modelcontextprotocol/sdk/types.js");
|
|
13
13
|
var _csvparse = require("csv-parse");
|
|
14
14
|
var _zod = require("zod");
|
|
15
|
-
var
|
|
15
|
+
var _csvstreamingts = require("../../spreadsheet/csv-streaming.js");
|
|
16
16
|
function _async_iterator(iterable) {
|
|
17
17
|
var method, async, sync, retry = 2;
|
|
18
18
|
for("undefined" != typeof Symbol && (async = Symbol.asyncIterator, sync = Symbol.iterator); retry--;){
|
|
@@ -231,7 +231,7 @@ function handler(_0, _1) {
|
|
|
231
231
|
]);
|
|
232
232
|
return [
|
|
233
233
|
4,
|
|
234
|
-
(0,
|
|
234
|
+
(0, _csvstreamingts.getCsvReadStream)(sourceUri)
|
|
235
235
|
];
|
|
236
236
|
case 2:
|
|
237
237
|
readStream = _state.sent();
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["/Users/kevin/Dev/Projects/
|
|
1
|
+
{"version":3,"sources":["/Users/kevin/Dev/Projects/mcp-z/mcp-sheets/src/mcp/tools/csv-get-columns.ts"],"sourcesContent":["/** Get column names from CSV file (peek at first row only) */\n\nimport type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport type { ToolModule } from '@mcp-z/server';\nimport type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';\nimport { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport { parse } from 'csv-parse';\nimport { z } from 'zod';\nimport { getCsvReadStream } from '../../spreadsheet/csv-streaming.ts';\n\nconst inputSchema = z.object({\n sourceUri: z.string().trim().min(1).describe('CSV file URI (file://, http://, https://)'),\n});\n\n// Success branch schema - uses columns: for consistency with standard vocabulary\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n columns: z.array(z.string()).describe('First row values (column names) or empty if no rows'),\n isEmpty: z.boolean().describe('True if CSV has zero rows'),\n});\n\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n description: 'Get first row from CSV file (streaming, no memory overhead). Returns columns array and isEmpty flag.',\n inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\nasync function handler({ sourceUri }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.debug?.('sheets.csv.get-columns called', { sourceUri });\n\n try {\n // Get readable stream from CSV URI (no temp files!)\n const readStream = await getCsvReadStream(sourceUri);\n\n // Create CSV parser without treating first row as column names\n // We just want to read the raw first row\n const parser = readStream.pipe(\n parse({\n columns: false, // Don't treat first row as headers\n skip_empty_lines: true,\n trim: true,\n cast: true, // Auto-convert numbers/booleans\n relax_column_count: true,\n })\n );\n\n // Read only the first row\n let firstRow: unknown[] = [];\n let rowCount = 0;\n\n for await (const row of parser) {\n firstRow = row;\n rowCount++;\n break; // Only read first row\n }\n\n // Convert first row to strings (column names)\n const columns = firstRow.map((value) => String(value ?? ''));\n const isEmpty = rowCount === 0;\n\n const result: Output = {\n type: 'success' as const,\n columns,\n isEmpty,\n };\n\n logger.info?.('sheets.csv.get-columns completed', { sourceUri, columnCount: columns.length, isEmpty });\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result) }],\n structuredContent: { result },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.error?.('sheets.csv.get-columns error', { error: message });\n throw new McpError(ErrorCode.InternalError, `Error getting CSV columns: ${message}`, {\n stack: error instanceof Error ? error.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'csv-get-columns',\n config,\n handler,\n } satisfies ToolModule;\n}\n"],"names":["createTool","AuthRequiredBranchSchema","schemas","inputSchema","z","object","sourceUri","string","trim","min","describe","successBranchSchema","type","literal","columns","array","isEmpty","boolean","outputSchema","discriminatedUnion","config","description","result","handler","extra","logger","readStream","parser","firstRow","rowCount","row","error","message","debug","getCsvReadStream","pipe","parse","skip_empty_lines","cast","relax_column_count","map","value","String","info","columnCount","length","content","text","JSON","stringify","structuredContent","Error","McpError","ErrorCode","InternalError","stack","undefined","name"],"mappings":"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"}
|
|
@@ -12,8 +12,8 @@ var _oauthgoogle = require("@mcp-z/oauth-google");
|
|
|
12
12
|
var _types = require("@modelcontextprotocol/sdk/types.js");
|
|
13
13
|
var _googleapis = require("googleapis");
|
|
14
14
|
var _zod = require("zod");
|
|
15
|
-
var
|
|
16
|
-
var
|
|
15
|
+
var _indexts = require("../../schemas/index.js");
|
|
16
|
+
var _dimensionoperationsts = require("./lib/dimension-operations.js");
|
|
17
17
|
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
|
|
18
18
|
try {
|
|
19
19
|
var info = gen[key](arg);
|
|
@@ -166,15 +166,15 @@ var DimensionRequestSchema = _zod.z.object({
|
|
|
166
166
|
inheritFromBefore: _zod.z.boolean().optional().describe('For insertDimension: whether new rows/columns inherit properties from the row/column before the insertion point')
|
|
167
167
|
});
|
|
168
168
|
var inputSchema = _zod.z.object({
|
|
169
|
-
id:
|
|
170
|
-
gid:
|
|
169
|
+
id: _indexts.SpreadsheetIdSchema,
|
|
170
|
+
gid: _indexts.SheetGidSchema,
|
|
171
171
|
requests: _zod.z.array(DimensionRequestSchema).min(1).describe('Array of dimension update requests')
|
|
172
172
|
});
|
|
173
173
|
// Success branch schema
|
|
174
174
|
var successBranchSchema = _zod.z.object({
|
|
175
175
|
type: _zod.z.literal('success'),
|
|
176
|
-
id:
|
|
177
|
-
gid:
|
|
176
|
+
id: _indexts.SpreadsheetIdOutput,
|
|
177
|
+
gid: _indexts.SheetGidOutput,
|
|
178
178
|
spreadsheetTitle: _zod.z.string().describe('Title of the updated spreadsheet'),
|
|
179
179
|
spreadsheetUrl: _zod.z.string().describe('URL of the updated spreadsheet'),
|
|
180
180
|
sheetTitle: _zod.z.string().describe('Title of the updated sheet'),
|
|
@@ -284,15 +284,15 @@ function handler(_0, _1) {
|
|
|
284
284
|
// Get current sheet dimensions for response calculation
|
|
285
285
|
// Note: Google Sheets API may not always provide gridProperties for older sheets
|
|
286
286
|
// Fall back to Google's documented defaults for new sheets
|
|
287
|
-
currentRowCount = (_ref1 = (_sheet_properties_gridProperties = sheet.properties.gridProperties) === null || _sheet_properties_gridProperties === void 0 ? void 0 : _sheet_properties_gridProperties.rowCount) !== null && _ref1 !== void 0 ? _ref1 :
|
|
288
|
-
currentColumnCount = (_ref2 = (_sheet_properties_gridProperties1 = sheet.properties.gridProperties) === null || _sheet_properties_gridProperties1 === void 0 ? void 0 : _sheet_properties_gridProperties1.columnCount) !== null && _ref2 !== void 0 ? _ref2 :
|
|
287
|
+
currentRowCount = (_ref1 = (_sheet_properties_gridProperties = sheet.properties.gridProperties) === null || _sheet_properties_gridProperties === void 0 ? void 0 : _sheet_properties_gridProperties.rowCount) !== null && _ref1 !== void 0 ? _ref1 : _dimensionoperationsts.DEFAULT_ROW_COUNT;
|
|
288
|
+
currentColumnCount = (_ref2 = (_sheet_properties_gridProperties1 = sheet.properties.gridProperties) === null || _sheet_properties_gridProperties1 === void 0 ? void 0 : _sheet_properties_gridProperties1.columnCount) !== null && _ref2 !== void 0 ? _ref2 : _dimensionoperationsts.DEFAULT_COLUMN_COUNT;
|
|
289
289
|
// Sort operations for optimal execution order to prevent index conflicts
|
|
290
290
|
// Delete operations are processed first (high to low index) to avoid shifting issues
|
|
291
291
|
// Insert/append operations are processed after (low to high index)
|
|
292
|
-
sortedRequests = (0,
|
|
292
|
+
sortedRequests = (0, _dimensionoperationsts.sortOperations)(requests);
|
|
293
293
|
// Build Google Sheets API batch update requests
|
|
294
294
|
batchRequests = sortedRequests.map(function(operation) {
|
|
295
|
-
return (0,
|
|
295
|
+
return (0, _dimensionoperationsts.buildDimensionRequest)(operation, sheetId);
|
|
296
296
|
});
|
|
297
297
|
(_logger_debug1 = logger.debug) === null || _logger_debug1 === void 0 ? void 0 : _logger_debug1.call(logger, 'sheets.dimensions.batchUpdate executing batch request', {
|
|
298
298
|
spreadsheetId: id,
|
|
@@ -309,7 +309,7 @@ function handler(_0, _1) {
|
|
|
309
309
|
dimension: r.dimension,
|
|
310
310
|
startIndex: r.startIndex,
|
|
311
311
|
endIndex: r.endIndex,
|
|
312
|
-
affectedCount: (0,
|
|
312
|
+
affectedCount: (0, _dimensionoperationsts.calculateAffectedCount)(r)
|
|
313
313
|
};
|
|
314
314
|
}),
|
|
315
315
|
currentDimensions: {
|
|
@@ -395,7 +395,7 @@ function handler(_0, _1) {
|
|
|
395
395
|
finalRowCount = currentRowCount;
|
|
396
396
|
finalColumnCount = currentColumnCount;
|
|
397
397
|
operationResults = sortedRequests.map(function(operation, _index) {
|
|
398
|
-
var affectedCount = (0,
|
|
398
|
+
var affectedCount = (0, _dimensionoperationsts.calculateAffectedCount)(operation);
|
|
399
399
|
// Update dimension counts based on operation
|
|
400
400
|
// Operations are applied in sorted order, so this should match actual result
|
|
401
401
|
if (operation.dimension === 'ROWS') {
|
|
@@ -424,20 +424,20 @@ function handler(_0, _1) {
|
|
|
424
424
|
};
|
|
425
425
|
});
|
|
426
426
|
// Validate final dimensions are within Google Sheets limits
|
|
427
|
-
if (finalRowCount >
|
|
427
|
+
if (finalRowCount > _dimensionoperationsts.MAX_ROW_COUNT) {
|
|
428
428
|
;
|
|
429
429
|
(_logger_warn1 = logger.warn) === null || _logger_warn1 === void 0 ? void 0 : _logger_warn1.call(logger, 'Final row count exceeds Google Sheets maximum', {
|
|
430
430
|
finalRowCount: finalRowCount,
|
|
431
|
-
maxRowCount:
|
|
431
|
+
maxRowCount: _dimensionoperationsts.MAX_ROW_COUNT,
|
|
432
432
|
spreadsheetId: id,
|
|
433
433
|
sheetTitle: sheetTitle
|
|
434
434
|
});
|
|
435
435
|
}
|
|
436
|
-
if (finalColumnCount >
|
|
436
|
+
if (finalColumnCount > _dimensionoperationsts.MAX_COLUMN_COUNT) {
|
|
437
437
|
;
|
|
438
438
|
(_logger_warn2 = logger.warn) === null || _logger_warn2 === void 0 ? void 0 : _logger_warn2.call(logger, 'Final column count exceeds Google Sheets maximum', {
|
|
439
439
|
finalColumnCount: finalColumnCount,
|
|
440
|
-
maxColumnCount:
|
|
440
|
+
maxColumnCount: _dimensionoperationsts.MAX_COLUMN_COUNT,
|
|
441
441
|
spreadsheetId: id,
|
|
442
442
|
sheetTitle: sheetTitle
|
|
443
443
|
});
|