import type { FastMCP } from 'fastmcp'; import { UserError } from 'fastmcp'; import { z } from 'zod'; import { docs_v1 } from 'googleapis'; import { getDocsClient } from '../../clients.js'; import { DocumentIdParameter } from '../../types.js'; import * as GDocsHelpers from '../../googleDocsApiHelpers.js'; export function buildInsertSectionBreakRequest(params: { index: number; sectionType: 'NEXT_PAGE' | 'CONTINUOUS'; tabId?: string; }): docs_v1.Schema$Request { const location: docs_v1.Schema$Location = { index: params.index }; if (params.tabId) { location.tabId = params.tabId; } return { insertSectionBreak: { location, sectionType: params.sectionType, }, }; } export function register(server: FastMCP) { server.addTool({ name: 'insertSectionBreak', description: 'Inserts a section break at a character index in the document. A section break starts a new section whose style (page orientation, margins, columns, page numbering) can then be customized with updateSectionStyle. Use sectionType="NEXT_PAGE" when you want the new section to start on a fresh page (required for mixing portrait and landscape pages in a single document) and "CONTINUOUS" when the new section should begin inline without a page break.', parameters: DocumentIdParameter.extend({ index: z .number() .int() .min(1) .describe( "1-based character index within the document body where the section break will be inserted. Use readDocument with format='json' to inspect indices." ), sectionType: z .enum(['NEXT_PAGE', 'CONTINUOUS']) .default('NEXT_PAGE') .describe( 'The type of section break. NEXT_PAGE starts the new section on the next page (required to change page orientation). CONTINUOUS starts the new section inline without a page break. Defaults to NEXT_PAGE.' ), tabId: z .string() .optional() .describe( 'The ID of the specific tab to insert into. Use listDocumentTabs to get tab IDs. If not specified, inserts into the first tab.' ), }), execute: async (args, { log }) => { const docs = await getDocsClient(); log.info( `Inserting ${args.sectionType} section break in doc ${args.documentId} at index ${args.index}${args.tabId ? ` (tab: ${args.tabId})` : ''}` ); try { if (args.tabId) { const docInfo = await docs.documents.get({ documentId: args.documentId, includeTabsContent: true, fields: 'tabs(tabProperties,documentTab(body))', }); const targetTab = GDocsHelpers.findTabById(docInfo.data, args.tabId); if (!targetTab) { throw new UserError(`Tab with ID "${args.tabId}" not found in document.`); } if (!targetTab.documentTab) { throw new UserError( `Tab "${args.tabId}" does not have content (may not be a document tab).` ); } } const request = buildInsertSectionBreakRequest({ index: args.index, sectionType: args.sectionType, tabId: args.tabId, }); await GDocsHelpers.executeBatchUpdate(docs, args.documentId, [request]); return `Successfully inserted ${args.sectionType} section break at index ${args.index}${args.tabId ? ` in tab ${args.tabId}` : ''}.`; } catch (error: any) { log.error( `Error inserting section break in doc ${args.documentId}: ${error.message || error}` ); if (error instanceof UserError) throw error; throw new UserError(`Failed to insert section break: ${error.message || 'Unknown error'}`); } }, }); }