@usejunior/docx-mcp 0.0.1 → 0.1.2

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.
Files changed (164) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +208 -28
  3. package/dist/.tsbuildinfo +1 -0
  4. package/dist/cli/commands/compare.d.ts +20 -0
  5. package/dist/cli/commands/compare.d.ts.map +1 -0
  6. package/dist/cli/commands/compare.js +54 -0
  7. package/dist/cli/commands/compare.js.map +1 -0
  8. package/dist/cli/commands/serve.d.ts +2 -0
  9. package/dist/cli/commands/serve.d.ts.map +1 -0
  10. package/dist/cli/commands/serve.js +5 -0
  11. package/dist/cli/commands/serve.js.map +1 -0
  12. package/dist/cli/index.d.ts +13 -0
  13. package/dist/cli/index.d.ts.map +1 -0
  14. package/dist/cli/index.js +145 -0
  15. package/dist/cli/index.js.map +1 -0
  16. package/dist/cli.d.ts +3 -0
  17. package/dist/cli.d.ts.map +1 -0
  18. package/dist/cli.js +8 -0
  19. package/dist/cli.js.map +1 -0
  20. package/dist/conformance/harness.d.ts +73 -0
  21. package/dist/conformance/harness.d.ts.map +1 -0
  22. package/dist/conformance/harness.js +609 -0
  23. package/dist/conformance/harness.js.map +1 -0
  24. package/dist/error_utils.d.ts +3 -0
  25. package/dist/error_utils.d.ts.map +1 -0
  26. package/dist/error_utils.js +19 -0
  27. package/dist/error_utils.js.map +1 -0
  28. package/dist/index.d.ts +3 -0
  29. package/dist/index.d.ts.map +1 -0
  30. package/dist/index.js +3 -0
  31. package/dist/index.js.map +1 -0
  32. package/dist/server.d.ts +80 -0
  33. package/dist/server.d.ts.map +1 -0
  34. package/dist/server.js +112 -0
  35. package/dist/server.js.map +1 -0
  36. package/dist/session/manager.d.ts +84 -0
  37. package/dist/session/manager.d.ts.map +1 -0
  38. package/dist/session/manager.js +155 -0
  39. package/dist/session/manager.js.map +1 -0
  40. package/dist/tool_catalog.d.ts +439 -0
  41. package/dist/tool_catalog.d.ts.map +1 -0
  42. package/dist/tool_catalog.js +314 -0
  43. package/dist/tool_catalog.js.map +1 -0
  44. package/dist/tools/accept_changes.d.ts +7 -0
  45. package/dist/tools/accept_changes.d.ts.map +1 -0
  46. package/dist/tools/accept_changes.js +21 -0
  47. package/dist/tools/accept_changes.js.map +1 -0
  48. package/dist/tools/add_comment.d.ts +13 -0
  49. package/dist/tools/add_comment.d.ts.map +1 -0
  50. package/dist/tools/add_comment.js +77 -0
  51. package/dist/tools/add_comment.js.map +1 -0
  52. package/dist/tools/add_footnote.d.ts +10 -0
  53. package/dist/tools/add_footnote.d.ts.map +1 -0
  54. package/dist/tools/add_footnote.js +45 -0
  55. package/dist/tools/add_footnote.js.map +1 -0
  56. package/dist/tools/apply_plan.d.ts +9 -0
  57. package/dist/tools/apply_plan.d.ts.map +1 -0
  58. package/dist/tools/apply_plan.js +346 -0
  59. package/dist/tools/apply_plan.js.map +1 -0
  60. package/dist/tools/clear_session.d.ts +9 -0
  61. package/dist/tools/clear_session.d.ts.map +1 -0
  62. package/dist/tools/clear_session.js +66 -0
  63. package/dist/tools/clear_session.js.map +1 -0
  64. package/dist/tools/compare_documents.d.ts +12 -0
  65. package/dist/tools/compare_documents.d.ts.map +1 -0
  66. package/dist/tools/compare_documents.js +121 -0
  67. package/dist/tools/compare_documents.js.map +1 -0
  68. package/dist/tools/comparison_defaults.d.ts +9 -0
  69. package/dist/tools/comparison_defaults.d.ts.map +1 -0
  70. package/dist/tools/comparison_defaults.js +8 -0
  71. package/dist/tools/comparison_defaults.js.map +1 -0
  72. package/dist/tools/delete_comment.d.ts +8 -0
  73. package/dist/tools/delete_comment.d.ts.map +1 -0
  74. package/dist/tools/delete_comment.js +28 -0
  75. package/dist/tools/delete_comment.js.map +1 -0
  76. package/dist/tools/delete_footnote.d.ts +8 -0
  77. package/dist/tools/delete_footnote.d.ts.map +1 -0
  78. package/dist/tools/delete_footnote.js +34 -0
  79. package/dist/tools/delete_footnote.js.map +1 -0
  80. package/dist/tools/docx_archive_guard.d.ts +9 -0
  81. package/dist/tools/docx_archive_guard.d.ts.map +1 -0
  82. package/dist/tools/docx_archive_guard.js +64 -0
  83. package/dist/tools/docx_archive_guard.js.map +1 -0
  84. package/dist/tools/download.d.ts +20 -0
  85. package/dist/tools/download.d.ts.map +1 -0
  86. package/dist/tools/download.js +252 -0
  87. package/dist/tools/download.js.map +1 -0
  88. package/dist/tools/duplicate_document.d.ts +8 -0
  89. package/dist/tools/duplicate_document.d.ts.map +1 -0
  90. package/dist/tools/duplicate_document.js +90 -0
  91. package/dist/tools/duplicate_document.js.map +1 -0
  92. package/dist/tools/extract_revisions.d.ts +9 -0
  93. package/dist/tools/extract_revisions.d.ts.map +1 -0
  94. package/dist/tools/extract_revisions.js +52 -0
  95. package/dist/tools/extract_revisions.js.map +1 -0
  96. package/dist/tools/format_layout.d.ts +35 -0
  97. package/dist/tools/format_layout.d.ts.map +1 -0
  98. package/dist/tools/format_layout.js +332 -0
  99. package/dist/tools/format_layout.js.map +1 -0
  100. package/dist/tools/get_comments.d.ts +7 -0
  101. package/dist/tools/get_comments.d.ts.map +1 -0
  102. package/dist/tools/get_comments.js +32 -0
  103. package/dist/tools/get_comments.js.map +1 -0
  104. package/dist/tools/get_footnotes.d.ts +7 -0
  105. package/dist/tools/get_footnotes.d.ts.map +1 -0
  106. package/dist/tools/get_footnotes.js +25 -0
  107. package/dist/tools/get_footnotes.js.map +1 -0
  108. package/dist/tools/get_session_status.d.ts +7 -0
  109. package/dist/tools/get_session_status.d.ts.map +1 -0
  110. package/dist/tools/get_session_status.js +41 -0
  111. package/dist/tools/get_session_status.js.map +1 -0
  112. package/dist/tools/grep.d.ts +13 -0
  113. package/dist/tools/grep.d.ts.map +1 -0
  114. package/dist/tools/grep.js +129 -0
  115. package/dist/tools/grep.js.map +1 -0
  116. package/dist/tools/has_tracked_changes.d.ts +7 -0
  117. package/dist/tools/has_tracked_changes.d.ts.map +1 -0
  118. package/dist/tools/has_tracked_changes.js +74 -0
  119. package/dist/tools/has_tracked_changes.js.map +1 -0
  120. package/dist/tools/init_plan.d.ts +9 -0
  121. package/dist/tools/init_plan.d.ts.map +1 -0
  122. package/dist/tools/init_plan.js +44 -0
  123. package/dist/tools/init_plan.js.map +1 -0
  124. package/dist/tools/insert_paragraph.d.ts +12 -0
  125. package/dist/tools/insert_paragraph.d.ts.map +1 -0
  126. package/dist/tools/insert_paragraph.js +514 -0
  127. package/dist/tools/insert_paragraph.js.map +1 -0
  128. package/dist/tools/merge_plans.d.ts +7 -0
  129. package/dist/tools/merge_plans.d.ts.map +1 -0
  130. package/dist/tools/merge_plans.js +408 -0
  131. package/dist/tools/merge_plans.js.map +1 -0
  132. package/dist/tools/open_document.d.ts +7 -0
  133. package/dist/tools/open_document.d.ts.map +1 -0
  134. package/dist/tools/open_document.js +102 -0
  135. package/dist/tools/open_document.js.map +1 -0
  136. package/dist/tools/path_policy.d.ts +13 -0
  137. package/dist/tools/path_policy.d.ts.map +1 -0
  138. package/dist/tools/path_policy.js +109 -0
  139. package/dist/tools/path_policy.js.map +1 -0
  140. package/dist/tools/preview.d.ts +5 -0
  141. package/dist/tools/preview.d.ts.map +1 -0
  142. package/dist/tools/preview.js +11 -0
  143. package/dist/tools/preview.js.map +1 -0
  144. package/dist/tools/read_file.d.ts +12 -0
  145. package/dist/tools/read_file.d.ts.map +1 -0
  146. package/dist/tools/read_file.js +113 -0
  147. package/dist/tools/read_file.js.map +1 -0
  148. package/dist/tools/replace_text.d.ts +13 -0
  149. package/dist/tools/replace_text.d.ts.map +1 -0
  150. package/dist/tools/replace_text.js +628 -0
  151. package/dist/tools/replace_text.js.map +1 -0
  152. package/dist/tools/session_resolution.d.ts +29 -0
  153. package/dist/tools/session_resolution.d.ts.map +1 -0
  154. package/dist/tools/session_resolution.js +158 -0
  155. package/dist/tools/session_resolution.js.map +1 -0
  156. package/dist/tools/types.d.ts +15 -0
  157. package/dist/tools/types.d.ts.map +1 -0
  158. package/dist/tools/types.js +7 -0
  159. package/dist/tools/types.js.map +1 -0
  160. package/dist/tools/update_footnote.d.ts +9 -0
  161. package/dist/tools/update_footnote.d.ts.map +1 -0
  162. package/dist/tools/update_footnote.js +39 -0
  163. package/dist/tools/update_footnote.js.map +1 -0
  164. package/package.json +74 -6
@@ -0,0 +1,29 @@
1
+ import { type Session, SessionManager } from '../session/manager.js';
2
+ import { type ToolResponse } from './types.js';
3
+ export type SessionResolutionMode = 'opened_new_session' | 'reused_existing_session' | 'explicit_session';
4
+ export type ResolvedSession = {
5
+ ok: true;
6
+ session: Session;
7
+ metadata: Record<string, unknown>;
8
+ };
9
+ export type SessionResolutionOutcome = ResolvedSession | {
10
+ ok: false;
11
+ response: ToolResponse;
12
+ };
13
+ export declare function validateAndLoadDocxFromPath(manager: SessionManager, filePath: string): Promise<{
14
+ ok: true;
15
+ normalizedPath: string;
16
+ filename: string;
17
+ content: Buffer;
18
+ } | {
19
+ ok: false;
20
+ response: ToolResponse;
21
+ }>;
22
+ export declare function mergeSessionResolutionMetadata(extra: Record<string, unknown>, metadata: Record<string, unknown>): Record<string, unknown>;
23
+ export declare function resolveSessionForTool(manager: SessionManager, params: {
24
+ session_id?: unknown;
25
+ file_path?: unknown;
26
+ }, opts: {
27
+ toolName: string;
28
+ }): Promise<SessionResolutionOutcome>;
29
+ //# sourceMappingURL=session_resolution.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session_resolution.d.ts","sourceRoot":"","sources":["../../src/tools/session_resolution.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,KAAK,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACrE,OAAO,EAAO,KAAK,YAAY,EAAE,MAAM,YAAY,CAAC;AAMpD,MAAM,MAAM,qBAAqB,GAC7B,oBAAoB,GACpB,yBAAyB,GACzB,kBAAkB,CAAC;AAEvB,MAAM,MAAM,eAAe,GAAG;IAC5B,EAAE,EAAE,IAAI,CAAC;IACT,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACnC,CAAC;AAEF,MAAM,MAAM,wBAAwB,GAChC,eAAe,GACf;IACE,EAAE,EAAE,KAAK,CAAC;IACV,QAAQ,EAAE,YAAY,CAAC;CACxB,CAAC;AAmBN,wBAAsB,2BAA2B,CAC/C,OAAO,EAAE,cAAc,EACvB,QAAQ,EAAE,MAAM,GACf,OAAO,CACN;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,cAAc,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,GACvE;IAAE,EAAE,EAAE,KAAK,CAAC;IAAC,QAAQ,EAAE,YAAY,CAAA;CAAE,CACxC,CAkEA;AAED,wBAAgB,8BAA8B,CAC5C,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC9B,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAChC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAEzB;AAED,wBAAsB,qBAAqB,CACzC,OAAO,EAAE,cAAc,EACvB,MAAM,EAAE;IAAE,UAAU,CAAC,EAAE,OAAO,CAAC;IAAC,SAAS,CAAC,EAAE,OAAO,CAAA;CAAE,EACrD,IAAI,EAAE;IAAE,QAAQ,EAAE,MAAM,CAAA;CAAE,GACzB,OAAO,CAAC,wBAAwB,CAAC,CAwGnC"}
@@ -0,0 +1,158 @@
1
+ import fs from 'node:fs/promises';
2
+ import { errorMessage } from "../error_utils.js";
3
+ import path from 'node:path';
4
+ import { err } from './types.js';
5
+ import { enforceReadPathPolicy } from './path_policy.js';
6
+ import { validateDocxArchiveSafety } from './docx_archive_guard.js';
7
+ const MAX_DOCX_BYTES = 50 * 1024 * 1024;
8
+ function mapSessionLookupError(message, sessionId) {
9
+ if (message.startsWith('INVALID_SESSION_ID:')) {
10
+ return err('INVALID_SESSION_ID', message.replace(/^INVALID_SESSION_ID:/, 'Invalid session id: '), 'Session IDs must match format: ses_[12 alphanumeric chars]');
11
+ }
12
+ if (message.startsWith('SESSION_NOT_FOUND:')) {
13
+ return err('SESSION_NOT_FOUND', `Session not found: ${sessionId}`);
14
+ }
15
+ if (message.startsWith('SESSION_EXPIRED:')) {
16
+ return err('SESSION_EXPIRED', `Session expired: ${sessionId}`);
17
+ }
18
+ return err('SESSION_RESOLUTION_ERROR', `Failed to resolve session: ${message}`);
19
+ }
20
+ export async function validateAndLoadDocxFromPath(manager, filePath) {
21
+ const normalizedPath = manager.normalizePath(filePath);
22
+ const stat = await fs.stat(normalizedPath).catch(() => null);
23
+ if (!stat || !stat.isFile()) {
24
+ return {
25
+ ok: false,
26
+ response: err('FILE_NOT_FOUND', `File not found: ${filePath}`, 'Copy the file to ~/Downloads/ or ~/Documents/ first, then pass that path.'),
27
+ };
28
+ }
29
+ if (path.extname(normalizedPath).toLowerCase() !== '.docx') {
30
+ return {
31
+ ok: false,
32
+ response: err('INVALID_FILE_TYPE', `Invalid file type: ${path.extname(normalizedPath)}`, 'Only .docx files are supported.'),
33
+ };
34
+ }
35
+ const policy = await enforceReadPathPolicy(filePath);
36
+ if (!policy.ok) {
37
+ return {
38
+ ok: false,
39
+ response: policy.response,
40
+ };
41
+ }
42
+ const safePath = policy.normalizedPath;
43
+ const safeStat = await fs.stat(safePath).catch(() => null);
44
+ if (!safeStat || !safeStat.isFile()) {
45
+ return {
46
+ ok: false,
47
+ response: err('FILE_NOT_FOUND', `File not found: ${filePath}`, 'Copy the file to ~/Downloads/ or ~/Documents/ first, then pass that path.'),
48
+ };
49
+ }
50
+ if (safeStat.size > MAX_DOCX_BYTES) {
51
+ return {
52
+ ok: false,
53
+ response: err('VALIDATION_ERROR', 'File too large', 'Check file type (.docx only) and size (max 50MB).'),
54
+ };
55
+ }
56
+ const content = await fs.readFile(safePath);
57
+ const archiveGuard = await validateDocxArchiveSafety(content);
58
+ if (!archiveGuard.ok) {
59
+ return {
60
+ ok: false,
61
+ response: archiveGuard.response,
62
+ };
63
+ }
64
+ return {
65
+ ok: true,
66
+ normalizedPath: safePath,
67
+ filename: path.basename(safePath),
68
+ content: content,
69
+ };
70
+ }
71
+ export function mergeSessionResolutionMetadata(extra, metadata) {
72
+ return { ...extra, ...metadata };
73
+ }
74
+ export async function resolveSessionForTool(manager, params, opts) {
75
+ const sessionId = typeof params.session_id === 'string' ? params.session_id.trim() : '';
76
+ const filePath = typeof params.file_path === 'string' ? params.file_path.trim() : '';
77
+ if (!sessionId && !filePath) {
78
+ return {
79
+ ok: false,
80
+ response: err('MISSING_SESSION_CONTEXT', `Tool '${opts.toolName}' requires session_id or file_path.`, "Provide an existing session_id, or pass file_path to auto-open/reuse an editing session."),
81
+ };
82
+ }
83
+ if (sessionId) {
84
+ let session;
85
+ try {
86
+ session = manager.getSession(sessionId);
87
+ }
88
+ catch (e) {
89
+ return {
90
+ ok: false,
91
+ response: mapSessionLookupError(errorMessage(e), sessionId),
92
+ };
93
+ }
94
+ if (filePath) {
95
+ const requestedPath = manager.normalizePath(filePath);
96
+ const sessionPath = manager.normalizePath(session.originalPath);
97
+ if (requestedPath !== sessionPath) {
98
+ return {
99
+ ok: false,
100
+ response: err('SESSION_FILE_CONFLICT', `session_id '${sessionId}' is bound to '${sessionPath}', but file_path resolves to '${requestedPath}'.`, 'Use either session_id alone, or provide a file_path that matches the same session document.'),
101
+ };
102
+ }
103
+ }
104
+ manager.touch(session);
105
+ return {
106
+ ok: true,
107
+ session,
108
+ metadata: {
109
+ session_resolution: 'explicit_session',
110
+ resolved_session_id: session.sessionId,
111
+ resolved_file_path: manager.normalizePath(session.originalPath),
112
+ },
113
+ };
114
+ }
115
+ const normalizedPath = manager.normalizePath(filePath);
116
+ const existing = manager.getMostRecentlyUsedSessionForPath(normalizedPath);
117
+ if (existing) {
118
+ const reuseLastUsed = existing.lastAccessedAt.toISOString();
119
+ manager.touch(existing);
120
+ return {
121
+ ok: true,
122
+ session: existing,
123
+ metadata: {
124
+ session_resolution: 'reused_existing_session',
125
+ reused_existing_session: true,
126
+ warning: `Using existing editing session ${existing.sessionId} for ${normalizedPath}.`,
127
+ resolved_session_id: existing.sessionId,
128
+ resolved_file_path: normalizedPath,
129
+ reused_session_context: {
130
+ edit_revision: existing.editRevision,
131
+ edit_count: existing.editCount,
132
+ created_at: existing.createdAt.toISOString(),
133
+ last_used_at: reuseLastUsed,
134
+ },
135
+ },
136
+ };
137
+ }
138
+ const loaded = await validateAndLoadDocxFromPath(manager, filePath);
139
+ if (!loaded.ok) {
140
+ return { ok: false, response: loaded.response };
141
+ }
142
+ const session = await manager.createSession(loaded.content, loaded.filename, loaded.normalizedPath);
143
+ // Normalize: merge runs + simplify redlines BEFORE bookmark allocation.
144
+ session.normalizationStats = session.doc.normalize();
145
+ // Backfill missing intrinsic paragraph IDs before first read/edit.
146
+ session.doc.insertParagraphBookmarks(`mcp_${session.sessionId}`);
147
+ manager.touch(session);
148
+ return {
149
+ ok: true,
150
+ session,
151
+ metadata: {
152
+ session_resolution: 'opened_new_session',
153
+ resolved_session_id: session.sessionId,
154
+ resolved_file_path: loaded.normalizedPath,
155
+ },
156
+ };
157
+ }
158
+ //# sourceMappingURL=session_resolution.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session_resolution.js","sourceRoot":"","sources":["../../src/tools/session_resolution.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ export type ToolResponse = {
2
+ success: true;
3
+ [key: string]: unknown;
4
+ } | {
5
+ success: false;
6
+ error: {
7
+ code: string;
8
+ message: string;
9
+ hint?: string;
10
+ };
11
+ [key: string]: unknown;
12
+ };
13
+ export declare function ok(extra?: Record<string, unknown>): ToolResponse;
14
+ export declare function err(code: string, message: string, hint?: string): ToolResponse;
15
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/tools/types.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,YAAY,GACpB;IAAE,OAAO,EAAE,IAAI,CAAC;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;CAAE,GACzC;IAAE,OAAO,EAAE,KAAK,CAAC;IAAC,KAAK,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;CAAE,CAAC;AAExG,wBAAgB,EAAE,CAAC,KAAK,GAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAM,GAAG,YAAY,CAEpE;AAED,wBAAgB,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,GAAG,YAAY,CAE9E"}
@@ -0,0 +1,7 @@
1
+ export function ok(extra = {}) {
2
+ return { success: true, ...extra };
3
+ }
4
+ export function err(code, message, hint) {
5
+ return { success: false, error: { code, message, hint } };
6
+ }
7
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/tools/types.ts"],"names":[],"mappings":"AAIA,MAAM,UAAU,EAAE,CAAC,QAAiC,EAAE;IACpD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,GAAG,KAAK,EAAE,CAAC;AACrC,CAAC;AAED,MAAM,UAAU,GAAG,CAAC,IAAY,EAAE,OAAe,EAAE,IAAa;IAC9D,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,EAAE,CAAC;AAC5D,CAAC"}
@@ -0,0 +1,9 @@
1
+ import { SessionManager } from '../session/manager.js';
2
+ import { type ToolResponse } from './types.js';
3
+ export declare function updateFootnote(manager: SessionManager, params: {
4
+ session_id?: string;
5
+ file_path?: string;
6
+ note_id?: number;
7
+ new_text?: string;
8
+ }): Promise<ToolResponse>;
9
+ //# sourceMappingURL=update_footnote.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update_footnote.d.ts","sourceRoot":"","sources":["../../src/tools/update_footnote.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAGvD,OAAO,EAAW,KAAK,YAAY,EAAE,MAAM,YAAY,CAAC;AAExD,wBAAsB,cAAc,CAClC,OAAO,EAAE,cAAc,EACvB,MAAM,EAAE;IACN,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB,GACA,OAAO,CAAC,YAAY,CAAC,CAoCvB"}
@@ -0,0 +1,39 @@
1
+ import { errorMessage } from "../error_utils.js";
2
+ import { resolveSessionForTool, mergeSessionResolutionMetadata } from './session_resolution.js';
3
+ import { ok, err } from './types.js';
4
+ export async function updateFootnote(manager, params) {
5
+ const resolved = await resolveSessionForTool(manager, params, { toolName: 'update_footnote' });
6
+ if (!resolved.ok)
7
+ return resolved.response;
8
+ const { session, metadata } = resolved;
9
+ if (params.note_id == null) {
10
+ return err('MISSING_PARAMETER', 'note_id is required.', 'Provide the footnote ID to update.');
11
+ }
12
+ if (!params.new_text) {
13
+ return err('MISSING_PARAMETER', 'new_text is required.', 'Provide the new footnote text.');
14
+ }
15
+ try {
16
+ // Verify footnote exists before updating
17
+ const existing = await session.doc.getFootnote(params.note_id);
18
+ if (!existing) {
19
+ return err('NOTE_NOT_FOUND', `Footnote ID ${params.note_id} not found`, 'Use get_footnotes to list available footnotes.');
20
+ }
21
+ await session.doc.updateFootnoteText({
22
+ noteId: params.note_id,
23
+ newText: params.new_text,
24
+ });
25
+ manager.markEdited(session);
26
+ return ok(mergeSessionResolutionMetadata({
27
+ note_id: params.note_id,
28
+ session_id: session.sessionId,
29
+ }, metadata));
30
+ }
31
+ catch (e) {
32
+ const msg = errorMessage(e);
33
+ if (msg.includes('not found')) {
34
+ return err('NOTE_NOT_FOUND', msg, 'Use get_footnotes to list available footnotes.');
35
+ }
36
+ return err('FOOTNOTE_ERROR', msg);
37
+ }
38
+ }
39
+ //# sourceMappingURL=update_footnote.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update_footnote.js","sourceRoot":"","sources":["../../src/tools/update_footnote.ts"],"names":[],"mappings":"AACA,OAAO,EAAa,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAC5D,OAAO,EAAE,qBAAqB,EAAE,8BAA8B,EAAE,MAAM,yBAAyB,CAAC;AAChG,OAAO,EAAE,EAAE,EAAE,GAAG,EAAqB,MAAM,YAAY,CAAC;AAExD,MAAM,CAAC,KAAK,UAAU,cAAc,CAClC,OAAuB,EACvB,MAKC;IAED,MAAM,QAAQ,GAAG,MAAM,qBAAqB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,QAAQ,EAAE,iBAAiB,EAAE,CAAC,CAAC;IAC/F,IAAI,CAAC,QAAQ,CAAC,EAAE;QAAE,OAAO,QAAQ,CAAC,QAAQ,CAAC;IAC3C,MAAM,EAAE,OAAO,EAAE,QAAQ,EAAE,GAAG,QAAQ,CAAC;IAEvC,IAAI,MAAM,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;QAC3B,OAAO,GAAG,CAAC,mBAAmB,EAAE,sBAAsB,EAAE,oCAAoC,CAAC,CAAC;IAChG,CAAC;IACD,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QACrB,OAAO,GAAG,CAAC,mBAAmB,EAAE,uBAAuB,EAAE,gCAAgC,CAAC,CAAC;IAC7F,CAAC;IAED,IAAI,CAAC;QACH,yCAAyC;QACzC,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAC/D,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO,GAAG,CAAC,gBAAgB,EAAE,eAAe,MAAM,CAAC,OAAO,YAAY,EAAE,gDAAgD,CAAC,CAAC;QAC5H,CAAC;QAED,MAAM,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC;YACnC,MAAM,EAAE,MAAM,CAAC,OAAO;YACtB,OAAO,EAAE,MAAM,CAAC,QAAQ;SACzB,CAAC,CAAC;QAEH,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAC5B,OAAO,EAAE,CAAC,8BAA8B,CAAC;YACvC,OAAO,EAAE,MAAM,CAAC,OAAO;YACvB,UAAU,EAAE,OAAO,CAAC,SAAS;SAC9B,EAAE,QAAQ,CAAC,CAAC,CAAC;IAChB,CAAC;IAAC,OAAO,CAAU,EAAE,CAAC;QACpB,MAAM,GAAG,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC;QAC5B,IAAI,GAAG,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;YAC9B,OAAO,GAAG,CAAC,gBAAgB,EAAE,GAAG,EAAE,gDAAgD,CAAC,CAAC;QACtF,CAAC;QACD,OAAO,GAAG,CAAC,gBAAgB,EAAE,GAAG,CAAC,CAAC;IACpC,CAAC;AACH,CAAC"}
package/package.json CHANGED
@@ -1,10 +1,78 @@
1
1
  {
2
2
  "name": "@usejunior/docx-mcp",
3
- "version": "0.0.1",
4
- "description": "OIDC trusted publishing setup package for @usejunior/docx-mcp",
3
+ "version": "0.1.2",
4
+ "description": "Safe Docx MCP server (TypeScript)",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "types": "dist/index.d.ts",
8
+ "bin": {
9
+ "safedocx": "dist/cli.js",
10
+ "safe-docx": "dist/cli.js"
11
+ },
12
+ "scripts": {
13
+ "clean:dist": "node -e \"const { rmSync } = require('node:fs'); rmSync('dist', { recursive: true, force: true });\"",
14
+ "build": "npm run build -w @usejunior/docx-core &&npm run clean:dist && tsc -p tsconfig.build.json",
15
+ "dev": "tsx watch src/cli.ts",
16
+ "check:spec-coverage": "node scripts/validate_openspec_coverage.mjs",
17
+ "conformance:discover": "tsx scripts/discover_docx_fixtures.ts --out conformance/discovery.report.json",
18
+ "conformance:run": "tsx scripts/run_conformance_harness.ts --manifest conformance/fixtures.manifest.json --report conformance/last-report.json",
19
+ "conformance:smoke": "tsx scripts/run_conformance_harness.ts --manifest conformance/fixtures.manifest.json --mode smoke --report conformance/smoke-report.json",
20
+ "docs:generate:tools": "node --import tsx scripts/generate_tool_reference.ts",
21
+ "test:conformance": "vitest run src/conformance/harness.test.ts",
22
+ "test": "npm run build -w @usejunior/docx-core &&npm run check:spec-coverage && node ../../node_modules/vitest/vitest.mjs",
23
+ "test:run": "npm run build -w @usejunior/docx-core &&npm run check:spec-coverage && node ../../node_modules/vitest/vitest.mjs run",
24
+ "test:coverage": "npm run build -w @usejunior/docx-core &&npm run check:spec-coverage && node ../../node_modules/vitest/vitest.mjs run --coverage --coverage.reporter=text-summary --coverage.reporter=json-summary --coverage.reporter=lcov",
25
+ "prepack": "npm run build",
26
+ "lint": "tsc -b --pretty false"
27
+ },
28
+ "dependencies": {
29
+ "@usejunior/docx-core": "^0.1.2",
30
+ "@modelcontextprotocol/sdk": "^1.0.0",
31
+ "zod": "^4.3.6"
32
+ },
33
+ "devDependencies": {
34
+ "@vitest/coverage-v8": "^3.2.4",
35
+ "@vitest/runner": "^3.2.4",
36
+ "allure-vitest": "^3.4.5",
37
+ "@types/node": "^22.10.0",
38
+ "tsx": "^4.19.2",
39
+ "typescript": "^5.7.2",
40
+ "vitest": "^3.2.4"
41
+ },
42
+ "engines": {
43
+ "node": ">=18.0.0"
44
+ },
45
+ "repository": {
46
+ "type": "git",
47
+ "url": "https://github.com/UseJunior/safe-docx"
48
+ },
49
+ "publishConfig": {
50
+ "access": "public"
51
+ },
52
+ "files": [
53
+ "dist"
54
+ ],
55
+ "license": "MIT",
5
56
  "keywords": [
6
- "oidc",
7
- "trusted-publishing",
8
- "setup"
9
- ]
57
+ "mcp",
58
+ "mcp-server",
59
+ "model-context-protocol",
60
+ "docx",
61
+ "word",
62
+ "document-editing",
63
+ "tracked-changes",
64
+ "document-comparison",
65
+ "ai",
66
+ "llm",
67
+ "formatting-preservation",
68
+ "ooxml",
69
+ "serverless",
70
+ "cloudflare-workers",
71
+ "edge",
72
+ "vercel"
73
+ ],
74
+ "homepage": "https://github.com/UseJunior/safe-docx/tree/main/packages/docx-mcp",
75
+ "bugs": {
76
+ "url": "https://github.com/UseJunior/safe-docx/issues"
77
+ }
10
78
  }