cowork-os 0.3.21 → 0.3.23

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 (170) hide show
  1. package/README.md +293 -6
  2. package/connectors/README.md +20 -0
  3. package/connectors/asana-mcp/README.md +24 -0
  4. package/connectors/asana-mcp/dist/index.js +427 -0
  5. package/connectors/asana-mcp/package.json +15 -0
  6. package/connectors/asana-mcp/src/index.ts +553 -0
  7. package/connectors/asana-mcp/tsconfig.json +13 -0
  8. package/connectors/hubspot-mcp/README.md +35 -0
  9. package/connectors/hubspot-mcp/dist/index.js +454 -0
  10. package/connectors/hubspot-mcp/package.json +15 -0
  11. package/connectors/hubspot-mcp/src/index.ts +562 -0
  12. package/connectors/hubspot-mcp/tsconfig.json +13 -0
  13. package/connectors/jira-mcp/README.md +49 -0
  14. package/connectors/jira-mcp/dist/index.js +588 -0
  15. package/connectors/jira-mcp/package.json +15 -0
  16. package/connectors/jira-mcp/src/index.ts +711 -0
  17. package/connectors/jira-mcp/tsconfig.json +13 -0
  18. package/connectors/linear-mcp/README.md +22 -0
  19. package/connectors/linear-mcp/dist/index.js +402 -0
  20. package/connectors/linear-mcp/package.json +15 -0
  21. package/connectors/linear-mcp/src/index.ts +522 -0
  22. package/connectors/linear-mcp/tsconfig.json +13 -0
  23. package/connectors/okta-mcp/README.md +24 -0
  24. package/connectors/okta-mcp/dist/index.js +411 -0
  25. package/connectors/okta-mcp/package.json +15 -0
  26. package/connectors/okta-mcp/src/index.ts +520 -0
  27. package/connectors/okta-mcp/tsconfig.json +13 -0
  28. package/connectors/salesforce-mcp/README.md +47 -0
  29. package/connectors/salesforce-mcp/dist/index.js +584 -0
  30. package/connectors/salesforce-mcp/package.json +15 -0
  31. package/connectors/salesforce-mcp/src/index.ts +722 -0
  32. package/connectors/salesforce-mcp/tsconfig.json +13 -0
  33. package/connectors/servicenow-mcp/README.md +26 -0
  34. package/connectors/servicenow-mcp/dist/index.js +400 -0
  35. package/connectors/servicenow-mcp/package.json +15 -0
  36. package/connectors/servicenow-mcp/src/index.ts +500 -0
  37. package/connectors/servicenow-mcp/tsconfig.json +13 -0
  38. package/connectors/templates/mcp-connector/README.md +31 -0
  39. package/connectors/templates/mcp-connector/package.json +15 -0
  40. package/connectors/templates/mcp-connector/src/index.ts +330 -0
  41. package/connectors/templates/mcp-connector/tsconfig.json +13 -0
  42. package/connectors/zendesk-mcp/README.md +40 -0
  43. package/connectors/zendesk-mcp/dist/index.js +431 -0
  44. package/connectors/zendesk-mcp/package.json +15 -0
  45. package/connectors/zendesk-mcp/src/index.ts +543 -0
  46. package/connectors/zendesk-mcp/tsconfig.json +13 -0
  47. package/dist/electron/electron/agent/daemon.js +25 -0
  48. package/dist/electron/electron/agent/executor.js +181 -26
  49. package/dist/electron/electron/agent/llm/anthropic-compatible-provider.js +177 -0
  50. package/dist/electron/electron/agent/llm/github-copilot-provider.js +97 -0
  51. package/dist/electron/electron/agent/llm/groq-provider.js +33 -0
  52. package/dist/electron/electron/agent/llm/index.js +11 -1
  53. package/dist/electron/electron/agent/llm/kimi-provider.js +33 -0
  54. package/dist/electron/electron/agent/llm/openai-compatible-provider.js +116 -0
  55. package/dist/electron/electron/agent/llm/openai-compatible.js +111 -0
  56. package/dist/electron/electron/agent/llm/openai-oauth.js +2 -1
  57. package/dist/electron/electron/agent/llm/openrouter-provider.js +1 -1
  58. package/dist/electron/electron/agent/llm/provider-factory.js +318 -4
  59. package/dist/electron/electron/agent/llm/types.js +66 -1
  60. package/dist/electron/electron/agent/llm/xai-provider.js +33 -0
  61. package/dist/electron/electron/agent/tools/box-tools.js +231 -0
  62. package/dist/electron/electron/agent/tools/builtin-settings.js +28 -0
  63. package/dist/electron/electron/agent/tools/dropbox-tools.js +237 -0
  64. package/dist/electron/electron/agent/tools/google-drive-tools.js +227 -0
  65. package/dist/electron/electron/agent/tools/notion-tools.js +312 -0
  66. package/dist/electron/electron/agent/tools/onedrive-tools.js +217 -0
  67. package/dist/electron/electron/agent/tools/registry.js +541 -0
  68. package/dist/electron/electron/agent/tools/sharepoint-tools.js +243 -0
  69. package/dist/electron/electron/agent/tools/shell-tools.js +12 -3
  70. package/dist/electron/electron/agent/tools/x-tools.js +1 -1
  71. package/dist/electron/electron/gateway/index.js +1 -0
  72. package/dist/electron/electron/gateway/router.js +123 -143
  73. package/dist/electron/electron/ipc/canvas-handlers.js +5 -0
  74. package/dist/electron/electron/ipc/handlers.js +627 -158
  75. package/dist/electron/electron/main.js +63 -0
  76. package/dist/electron/electron/mcp/oauth/connector-oauth.js +333 -0
  77. package/dist/electron/electron/mcp/registry/MCPRegistryManager.js +503 -154
  78. package/dist/electron/electron/memory/MemoryService.js +1 -1
  79. package/dist/electron/electron/preload.js +74 -1
  80. package/dist/electron/electron/settings/box-manager.js +54 -0
  81. package/dist/electron/electron/settings/dropbox-manager.js +54 -0
  82. package/dist/electron/electron/settings/google-drive-manager.js +54 -0
  83. package/dist/electron/electron/settings/notion-manager.js +56 -0
  84. package/dist/electron/electron/settings/onedrive-manager.js +54 -0
  85. package/dist/electron/electron/settings/sharepoint-manager.js +54 -0
  86. package/dist/electron/electron/utils/box-api.js +153 -0
  87. package/dist/electron/electron/utils/dropbox-api.js +144 -0
  88. package/dist/electron/electron/utils/env-migration.js +19 -0
  89. package/dist/electron/electron/utils/google-drive-api.js +152 -0
  90. package/dist/electron/electron/utils/notion-api.js +103 -0
  91. package/dist/electron/electron/utils/onedrive-api.js +113 -0
  92. package/dist/electron/electron/utils/sharepoint-api.js +109 -0
  93. package/dist/electron/electron/utils/validation.js +82 -3
  94. package/dist/electron/electron/utils/x-cli.js +1 -1
  95. package/dist/electron/shared/channelMessages.js +284 -3
  96. package/dist/electron/shared/llm-provider-catalog.js +198 -0
  97. package/dist/electron/shared/types.js +88 -1
  98. package/package.json +12 -2
  99. package/src/electron/agent/executor.ts +205 -28
  100. package/src/electron/agent/llm/anthropic-compatible-provider.ts +214 -0
  101. package/src/electron/agent/llm/github-copilot-provider.ts +117 -0
  102. package/src/electron/agent/llm/groq-provider.ts +39 -0
  103. package/src/electron/agent/llm/index.ts +5 -0
  104. package/src/electron/agent/llm/kimi-provider.ts +39 -0
  105. package/src/electron/agent/llm/openai-compatible-provider.ts +153 -0
  106. package/src/electron/agent/llm/openai-compatible.ts +133 -0
  107. package/src/electron/agent/llm/openai-oauth.ts +2 -1
  108. package/src/electron/agent/llm/openrouter-provider.ts +2 -1
  109. package/src/electron/agent/llm/provider-factory.ts +414 -6
  110. package/src/electron/agent/llm/types.ts +90 -1
  111. package/src/electron/agent/llm/xai-provider.ts +39 -0
  112. package/src/electron/agent/tools/box-tools.ts +239 -0
  113. package/src/electron/agent/tools/builtin-settings.ts +34 -0
  114. package/src/electron/agent/tools/dropbox-tools.ts +237 -0
  115. package/src/electron/agent/tools/google-drive-tools.ts +228 -0
  116. package/src/electron/agent/tools/notion-tools.ts +330 -0
  117. package/src/electron/agent/tools/onedrive-tools.ts +217 -0
  118. package/src/electron/agent/tools/registry.ts +565 -0
  119. package/src/electron/agent/tools/sharepoint-tools.ts +247 -0
  120. package/src/electron/agent/tools/shell-tools.ts +11 -3
  121. package/src/electron/agent/tools/x-tools.ts +1 -1
  122. package/src/electron/database/SecureSettingsRepository.ts +7 -1
  123. package/src/electron/gateway/index.ts +1 -0
  124. package/src/electron/gateway/router.ts +134 -149
  125. package/src/electron/ipc/canvas-handlers.ts +10 -0
  126. package/src/electron/ipc/handlers.ts +673 -153
  127. package/src/electron/main.ts +35 -0
  128. package/src/electron/mcp/oauth/connector-oauth.ts +448 -0
  129. package/src/electron/mcp/registry/MCPRegistryManager.ts +343 -12
  130. package/src/electron/memory/MemoryService.ts +5 -1
  131. package/src/electron/preload.ts +167 -4
  132. package/src/electron/settings/box-manager.ts +58 -0
  133. package/src/electron/settings/dropbox-manager.ts +58 -0
  134. package/src/electron/settings/google-drive-manager.ts +58 -0
  135. package/src/electron/settings/notion-manager.ts +60 -0
  136. package/src/electron/settings/onedrive-manager.ts +58 -0
  137. package/src/electron/settings/sharepoint-manager.ts +58 -0
  138. package/src/electron/utils/box-api.ts +184 -0
  139. package/src/electron/utils/dropbox-api.ts +171 -0
  140. package/src/electron/utils/env-migration.ts +22 -0
  141. package/src/electron/utils/google-drive-api.ts +183 -0
  142. package/src/electron/utils/notion-api.ts +126 -0
  143. package/src/electron/utils/onedrive-api.ts +137 -0
  144. package/src/electron/utils/sharepoint-api.ts +132 -0
  145. package/src/electron/utils/validation.ts +102 -1
  146. package/src/electron/utils/x-cli.ts +1 -1
  147. package/src/renderer/App.tsx +20 -2
  148. package/src/renderer/components/BoxSettings.tsx +203 -0
  149. package/src/renderer/components/BrowserView.tsx +101 -0
  150. package/src/renderer/components/BuiltinToolsSettings.tsx +105 -0
  151. package/src/renderer/components/CanvasPreview.tsx +68 -1
  152. package/src/renderer/components/ConnectorEnvModal.tsx +116 -0
  153. package/src/renderer/components/ConnectorSetupModal.tsx +566 -0
  154. package/src/renderer/components/ConnectorsSettings.tsx +397 -0
  155. package/src/renderer/components/DropboxSettings.tsx +202 -0
  156. package/src/renderer/components/GoogleDriveSettings.tsx +201 -0
  157. package/src/renderer/components/MCPSettings.tsx +56 -0
  158. package/src/renderer/components/MainContent.tsx +270 -34
  159. package/src/renderer/components/NotionSettings.tsx +231 -0
  160. package/src/renderer/components/Onboarding/Onboarding.tsx +13 -1
  161. package/src/renderer/components/OnboardingModal.tsx +70 -1
  162. package/src/renderer/components/OneDriveSettings.tsx +212 -0
  163. package/src/renderer/components/Settings.tsx +611 -8
  164. package/src/renderer/components/SharePointSettings.tsx +224 -0
  165. package/src/renderer/components/Sidebar.tsx +25 -9
  166. package/src/renderer/hooks/useOnboardingFlow.ts +21 -0
  167. package/src/renderer/styles/index.css +438 -25
  168. package/src/shared/channelMessages.ts +367 -4
  169. package/src/shared/llm-provider-catalog.ts +217 -0
  170. package/src/shared/types.ts +226 -1
@@ -0,0 +1,228 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import mime from 'mime-types';
4
+ import { Workspace } from '../../../shared/types';
5
+ import { AgentDaemon } from '../daemon';
6
+ import { GoogleDriveSettingsManager } from '../../settings/google-drive-manager';
7
+ import { googleDriveRequest, googleDriveUpload } from '../../utils/google-drive-api';
8
+
9
+ type GoogleDriveAction =
10
+ | 'get_current_user'
11
+ | 'list_files'
12
+ | 'get_file'
13
+ | 'create_folder'
14
+ | 'upload_file'
15
+ | 'delete_file';
16
+
17
+ interface GoogleDriveActionInput {
18
+ action: GoogleDriveAction;
19
+ query?: string;
20
+ page_size?: number;
21
+ page_token?: string;
22
+ fields?: string;
23
+ file_id?: string;
24
+ parent_id?: string;
25
+ name?: string;
26
+ file_path?: string;
27
+ }
28
+
29
+ const DEFAULT_LIST_FIELDS =
30
+ 'nextPageToken, files(id,name,mimeType,modifiedTime,parents,webViewLink,size)';
31
+ const DEFAULT_FILE_FIELDS =
32
+ 'id,name,mimeType,modifiedTime,parents,webViewLink,size';
33
+
34
+ export class GoogleDriveTools {
35
+ constructor(
36
+ private workspace: Workspace,
37
+ private daemon: AgentDaemon,
38
+ private taskId: string
39
+ ) {}
40
+
41
+ setWorkspace(workspace: Workspace): void {
42
+ this.workspace = workspace;
43
+ }
44
+
45
+ static isEnabled(): boolean {
46
+ return GoogleDriveSettingsManager.loadSettings().enabled;
47
+ }
48
+
49
+ private async requireApproval(summary: string, details: Record<string, unknown>): Promise<void> {
50
+ const approved = await this.daemon.requestApproval(
51
+ this.taskId,
52
+ 'external_service',
53
+ summary,
54
+ details
55
+ );
56
+
57
+ if (!approved) {
58
+ throw new Error('User denied Google Drive action');
59
+ }
60
+ }
61
+
62
+ private resolveFilePath(inputPath: string): string {
63
+ if (!this.workspace.permissions.read) {
64
+ throw new Error('Read permission not granted for uploads');
65
+ }
66
+
67
+ const workspaceRoot = path.resolve(this.workspace.path);
68
+ const allowedPaths = this.workspace.permissions.allowedPaths || [];
69
+ const canReadOutside = this.workspace.isTemp || this.workspace.permissions.unrestrictedFileAccess;
70
+
71
+ const isPathAllowed = (absolutePath: string): boolean => {
72
+ if (allowedPaths.length === 0) return false;
73
+ const normalizedPath = path.normalize(absolutePath);
74
+ return allowedPaths.some((allowed) => {
75
+ const normalizedAllowed = path.normalize(allowed);
76
+ return normalizedPath === normalizedAllowed || normalizedPath.startsWith(normalizedAllowed + path.sep);
77
+ });
78
+ };
79
+
80
+ const candidate = path.isAbsolute(inputPath)
81
+ ? path.normalize(inputPath)
82
+ : path.resolve(workspaceRoot, inputPath);
83
+
84
+ const relative = path.relative(workspaceRoot, candidate);
85
+ const isInsideWorkspace = !(relative.startsWith('..') || path.isAbsolute(relative));
86
+ if (!isInsideWorkspace && !canReadOutside && !isPathAllowed(candidate)) {
87
+ throw new Error('File path must be inside the workspace or in Allowed Paths');
88
+ }
89
+ if (!fs.existsSync(candidate)) {
90
+ throw new Error(`File not found: ${inputPath}`);
91
+ }
92
+ const stats = fs.statSync(candidate);
93
+ if (!stats.isFile()) {
94
+ throw new Error(`Path is not a file: ${inputPath}`);
95
+ }
96
+ return candidate;
97
+ }
98
+
99
+ async executeAction(input: GoogleDriveActionInput): Promise<any> {
100
+ const settings = GoogleDriveSettingsManager.loadSettings();
101
+ if (!settings.enabled) {
102
+ throw new Error('Google Drive integration is disabled. Enable it in Settings > Integrations > Google Drive.');
103
+ }
104
+
105
+ const action = input.action;
106
+ if (!action) {
107
+ throw new Error('Missing required "action" parameter');
108
+ }
109
+
110
+ let result;
111
+
112
+ switch (action) {
113
+ case 'get_current_user': {
114
+ result = await googleDriveRequest(settings, {
115
+ method: 'GET',
116
+ path: '/about',
117
+ query: { fields: 'user' },
118
+ });
119
+ break;
120
+ }
121
+ case 'list_files': {
122
+ const query = input.query || 'trashed = false';
123
+ result = await googleDriveRequest(settings, {
124
+ method: 'GET',
125
+ path: '/files',
126
+ query: {
127
+ q: query,
128
+ pageSize: input.page_size,
129
+ pageToken: input.page_token,
130
+ fields: input.fields || DEFAULT_LIST_FIELDS,
131
+ },
132
+ });
133
+ break;
134
+ }
135
+ case 'get_file': {
136
+ if (!input.file_id) throw new Error('Missing file_id for get_file');
137
+ result = await googleDriveRequest(settings, {
138
+ method: 'GET',
139
+ path: `/files/${input.file_id}`,
140
+ query: {
141
+ fields: input.fields || DEFAULT_FILE_FIELDS,
142
+ },
143
+ });
144
+ break;
145
+ }
146
+ case 'create_folder': {
147
+ if (!input.name) throw new Error('Missing name for create_folder');
148
+ await this.requireApproval('Create a Google Drive folder', {
149
+ action: 'create_folder',
150
+ parent_id: input.parent_id || 'root',
151
+ name: input.name,
152
+ });
153
+ result = await googleDriveRequest(settings, {
154
+ method: 'POST',
155
+ path: '/files',
156
+ query: { fields: DEFAULT_FILE_FIELDS },
157
+ body: {
158
+ name: input.name,
159
+ mimeType: 'application/vnd.google-apps.folder',
160
+ parents: input.parent_id ? [input.parent_id] : undefined,
161
+ },
162
+ });
163
+ break;
164
+ }
165
+ case 'upload_file': {
166
+ if (!input.file_path) throw new Error('Missing file_path for upload_file');
167
+ const resolved = this.resolveFilePath(input.file_path);
168
+ const data = fs.readFileSync(resolved);
169
+ const fileName = input.name || path.basename(resolved);
170
+ const contentType = (mime.lookup(fileName) || 'application/octet-stream') as string;
171
+ await this.requireApproval(`Upload file to Google Drive: ${fileName}`, {
172
+ action: 'upload_file',
173
+ parent_id: input.parent_id || 'root',
174
+ file: fileName,
175
+ });
176
+ const created = await googleDriveRequest(settings, {
177
+ method: 'POST',
178
+ path: '/files',
179
+ query: { fields: DEFAULT_FILE_FIELDS },
180
+ body: {
181
+ name: fileName,
182
+ parents: input.parent_id ? [input.parent_id] : undefined,
183
+ },
184
+ });
185
+ const fileId = created.data?.id;
186
+ if (!fileId) {
187
+ throw new Error('Failed to create Google Drive file record');
188
+ }
189
+ const uploaded = await googleDriveUpload(settings, fileId, data, contentType);
190
+ result = {
191
+ status: uploaded.status,
192
+ data: uploaded.data || created.data,
193
+ raw: uploaded.raw,
194
+ };
195
+ break;
196
+ }
197
+ case 'delete_file': {
198
+ if (!input.file_id) throw new Error('Missing file_id for delete_file');
199
+ await this.requireApproval('Delete a Google Drive file', {
200
+ action: 'delete_file',
201
+ file_id: input.file_id,
202
+ });
203
+ result = await googleDriveRequest(settings, {
204
+ method: 'DELETE',
205
+ path: `/files/${input.file_id}`,
206
+ });
207
+ break;
208
+ }
209
+ default:
210
+ throw new Error(`Unsupported action: ${action}`);
211
+ }
212
+
213
+ this.daemon.logEvent(this.taskId, 'tool_result', {
214
+ tool: 'google_drive_action',
215
+ action,
216
+ status: result?.status,
217
+ hasData: result?.data ? true : false,
218
+ });
219
+
220
+ return {
221
+ success: true,
222
+ action,
223
+ status: result?.status,
224
+ data: result?.data,
225
+ raw: result?.raw,
226
+ };
227
+ }
228
+ }
@@ -0,0 +1,330 @@
1
+ import { Workspace } from '../../../shared/types';
2
+ import { AgentDaemon } from '../daemon';
3
+ import { NotionSettingsManager } from '../../settings/notion-manager';
4
+ import { notionRequest } from '../../utils/notion-api';
5
+
6
+ type NotionAction =
7
+ | 'search'
8
+ | 'list_users'
9
+ | 'get_user'
10
+ | 'get_page'
11
+ | 'get_page_property'
12
+ | 'get_database'
13
+ | 'get_block'
14
+ | 'get_block_children'
15
+ | 'update_block'
16
+ | 'delete_block'
17
+ | 'create_page'
18
+ | 'update_page'
19
+ | 'append_blocks'
20
+ | 'query_data_source'
21
+ | 'get_data_source'
22
+ | 'create_data_source'
23
+ | 'update_data_source';
24
+
25
+ interface NotionActionInput {
26
+ action: NotionAction;
27
+ query?: string;
28
+ user_id?: string;
29
+ page_id?: string;
30
+ property_id?: string;
31
+ block_id?: string;
32
+ block_type?: string;
33
+ block?: Record<string, any>;
34
+ data_source_id?: string;
35
+ database_id?: string;
36
+ parent_page_id?: string;
37
+ properties?: Record<string, any>;
38
+ children?: any[];
39
+ filter?: any;
40
+ sort?: any;
41
+ sorts?: any[];
42
+ start_cursor?: string;
43
+ page_size?: number;
44
+ archived?: boolean;
45
+ icon?: any;
46
+ cover?: any;
47
+ title?: string;
48
+ is_inline?: boolean;
49
+ payload?: Record<string, any>;
50
+ }
51
+
52
+ export class NotionTools {
53
+ constructor(
54
+ private workspace: Workspace,
55
+ private daemon: AgentDaemon,
56
+ private taskId: string
57
+ ) {}
58
+
59
+ setWorkspace(workspace: Workspace): void {
60
+ this.workspace = workspace;
61
+ }
62
+
63
+ static isEnabled(): boolean {
64
+ return NotionSettingsManager.loadSettings().enabled;
65
+ }
66
+
67
+ private async requireApproval(summary: string, details: Record<string, unknown>): Promise<void> {
68
+ const approved = await this.daemon.requestApproval(
69
+ this.taskId,
70
+ 'external_service',
71
+ summary,
72
+ details
73
+ );
74
+
75
+ if (!approved) {
76
+ throw new Error('User denied Notion action');
77
+ }
78
+ }
79
+
80
+ private buildPagination(input: NotionActionInput): Record<string, any> {
81
+ const body: Record<string, any> = {};
82
+ if (input.start_cursor) body.start_cursor = input.start_cursor;
83
+ if (typeof input.page_size === 'number') body.page_size = input.page_size;
84
+ return body;
85
+ }
86
+
87
+ private buildTitle(title?: string): Array<{ text: { content: string } }> | undefined {
88
+ if (!title) return undefined;
89
+ const trimmed = title.trim();
90
+ if (!trimmed) return undefined;
91
+ return [{ text: { content: trimmed } }];
92
+ }
93
+
94
+ private buildParent(input: NotionActionInput): Record<string, string> {
95
+ if (input.database_id) {
96
+ return { database_id: input.database_id };
97
+ }
98
+ if (input.parent_page_id) {
99
+ return { page_id: input.parent_page_id };
100
+ }
101
+ throw new Error('Missing parent identifier (database_id or parent_page_id)');
102
+ }
103
+
104
+ async executeAction(input: NotionActionInput): Promise<any> {
105
+ const settings = NotionSettingsManager.loadSettings();
106
+ if (!settings.enabled) {
107
+ throw new Error('Notion integration is disabled. Enable it in Settings > Integrations > Notion.');
108
+ }
109
+
110
+ const action = input.action;
111
+ if (!action) {
112
+ throw new Error('Missing required "action" parameter');
113
+ }
114
+
115
+ let result;
116
+
117
+ switch (action) {
118
+ case 'search': {
119
+ const body = input.payload ? { ...input.payload } : {};
120
+ if (!input.payload) {
121
+ if (input.query) body.query = input.query;
122
+ if (input.filter) body.filter = input.filter;
123
+ if (input.sort) body.sort = input.sort;
124
+ else if (input.sorts) body.sort = input.sorts;
125
+ Object.assign(body, this.buildPagination(input));
126
+ }
127
+ result = await notionRequest(settings, { method: 'POST', path: '/search', body });
128
+ break;
129
+ }
130
+ case 'list_users': {
131
+ const params = new URLSearchParams();
132
+ if (input.start_cursor) params.set('start_cursor', input.start_cursor);
133
+ if (typeof input.page_size === 'number') params.set('page_size', String(input.page_size));
134
+ const suffix = params.toString() ? `?${params.toString()}` : '';
135
+ result = await notionRequest(settings, { method: 'GET', path: `/users${suffix}` });
136
+ break;
137
+ }
138
+ case 'get_user': {
139
+ if (!input.user_id) throw new Error('Missing user_id for get_user');
140
+ result = await notionRequest(settings, { method: 'GET', path: `/users/${input.user_id}` });
141
+ break;
142
+ }
143
+ case 'get_page': {
144
+ if (!input.page_id) throw new Error('Missing page_id for get_page');
145
+ result = await notionRequest(settings, { method: 'GET', path: `/pages/${input.page_id}` });
146
+ break;
147
+ }
148
+ case 'get_page_property': {
149
+ if (!input.page_id) throw new Error('Missing page_id for get_page_property');
150
+ if (!input.property_id) throw new Error('Missing property_id for get_page_property');
151
+ const params = new URLSearchParams();
152
+ if (input.start_cursor) params.set('start_cursor', input.start_cursor);
153
+ if (typeof input.page_size === 'number') params.set('page_size', String(input.page_size));
154
+ const suffix = params.toString() ? `?${params.toString()}` : '';
155
+ result = await notionRequest(settings, { method: 'GET', path: `/pages/${input.page_id}/properties/${input.property_id}${suffix}` });
156
+ break;
157
+ }
158
+ case 'get_database': {
159
+ if (!input.database_id) throw new Error('Missing database_id for get_database');
160
+ result = await notionRequest(settings, { method: 'GET', path: `/databases/${input.database_id}` });
161
+ break;
162
+ }
163
+ case 'get_block': {
164
+ if (!input.block_id) throw new Error('Missing block_id for get_block');
165
+ result = await notionRequest(settings, { method: 'GET', path: `/blocks/${input.block_id}` });
166
+ break;
167
+ }
168
+ case 'get_block_children': {
169
+ if (!input.block_id) throw new Error('Missing block_id for get_block_children');
170
+ const params = new URLSearchParams();
171
+ if (input.start_cursor) params.set('start_cursor', input.start_cursor);
172
+ if (typeof input.page_size === 'number') params.set('page_size', String(input.page_size));
173
+ const suffix = params.toString() ? `?${params.toString()}` : '';
174
+ result = await notionRequest(settings, { method: 'GET', path: `/blocks/${input.block_id}/children${suffix}` });
175
+ break;
176
+ }
177
+ case 'update_block': {
178
+ if (!input.block_id) throw new Error('Missing block_id for update_block');
179
+ const body = input.payload ? { ...input.payload } : {};
180
+ if (!input.payload) {
181
+ if (typeof input.archived === 'boolean') body.archived = input.archived;
182
+ if (input.block_type && input.block) {
183
+ body[input.block_type] = input.block;
184
+ }
185
+ if (Object.keys(body).length === 0) {
186
+ throw new Error('Missing update payload (archived or block content)');
187
+ }
188
+ }
189
+ await this.requireApproval('Update a Notion block', {
190
+ action: 'update_block',
191
+ block_id: input.block_id,
192
+ });
193
+ result = await notionRequest(settings, { method: 'PATCH', path: `/blocks/${input.block_id}`, body });
194
+ break;
195
+ }
196
+ case 'delete_block': {
197
+ if (!input.block_id) throw new Error('Missing block_id for delete_block');
198
+ await this.requireApproval('Delete a Notion block', {
199
+ action: 'delete_block',
200
+ block_id: input.block_id,
201
+ });
202
+ result = await notionRequest(settings, { method: 'DELETE', path: `/blocks/${input.block_id}` });
203
+ break;
204
+ }
205
+ case 'get_data_source': {
206
+ if (!input.data_source_id) throw new Error('Missing data_source_id for get_data_source');
207
+ result = await notionRequest(settings, { method: 'GET', path: `/data_sources/${input.data_source_id}` });
208
+ break;
209
+ }
210
+ case 'query_data_source': {
211
+ if (!input.data_source_id) throw new Error('Missing data_source_id for query_data_source');
212
+ const body = input.payload ? { ...input.payload } : {};
213
+ if (!input.payload) {
214
+ if (input.filter) body.filter = input.filter;
215
+ if (input.sorts) body.sorts = input.sorts;
216
+ Object.assign(body, this.buildPagination(input));
217
+ }
218
+ result = await notionRequest(settings, { method: 'POST', path: `/data_sources/${input.data_source_id}/query`, body });
219
+ break;
220
+ }
221
+ case 'create_page': {
222
+ const body = input.payload ? { ...input.payload } : {};
223
+ if (!input.payload) {
224
+ body.parent = this.buildParent(input);
225
+ if (!input.properties) throw new Error('Missing properties for create_page');
226
+ body.properties = input.properties;
227
+ if (input.children) body.children = input.children;
228
+ if (input.icon) body.icon = input.icon;
229
+ if (input.cover) body.cover = input.cover;
230
+ }
231
+ await this.requireApproval('Create a Notion page', {
232
+ action: 'create_page',
233
+ parent: body.parent,
234
+ });
235
+ result = await notionRequest(settings, { method: 'POST', path: '/pages', body });
236
+ break;
237
+ }
238
+ case 'update_page': {
239
+ if (!input.page_id) throw new Error('Missing page_id for update_page');
240
+ const body = input.payload ? { ...input.payload } : {};
241
+ if (!input.payload) {
242
+ if (input.properties) body.properties = input.properties;
243
+ if (typeof input.archived === 'boolean') body.archived = input.archived;
244
+ if (input.icon) body.icon = input.icon;
245
+ if (input.cover) body.cover = input.cover;
246
+ if (Object.keys(body).length === 0) {
247
+ throw new Error('Missing update payload (properties, archived, icon, or cover)');
248
+ }
249
+ }
250
+ await this.requireApproval('Update a Notion page', {
251
+ action: 'update_page',
252
+ page_id: input.page_id,
253
+ });
254
+ result = await notionRequest(settings, { method: 'PATCH', path: `/pages/${input.page_id}`, body });
255
+ break;
256
+ }
257
+ case 'append_blocks': {
258
+ if (!input.block_id) throw new Error('Missing block_id for append_blocks');
259
+ const body = input.payload ? { ...input.payload } : {};
260
+ if (!input.payload) {
261
+ if (!input.children || input.children.length === 0) {
262
+ throw new Error('Missing children for append_blocks');
263
+ }
264
+ body.children = input.children;
265
+ }
266
+ await this.requireApproval('Append blocks in Notion', {
267
+ action: 'append_blocks',
268
+ block_id: input.block_id,
269
+ count: Array.isArray(body.children) ? body.children.length : undefined,
270
+ });
271
+ result = await notionRequest(settings, { method: 'PATCH', path: `/blocks/${input.block_id}/children`, body });
272
+ break;
273
+ }
274
+ case 'create_data_source': {
275
+ const body = input.payload ? { ...input.payload } : {};
276
+ if (!input.payload) {
277
+ if (!input.parent_page_id) throw new Error('Missing parent_page_id for create_data_source');
278
+ if (!input.properties) throw new Error('Missing properties for create_data_source');
279
+ const title = this.buildTitle(input.title);
280
+ if (!title) throw new Error('Missing title for create_data_source');
281
+ body.parent = { page_id: input.parent_page_id };
282
+ body.title = title;
283
+ body.properties = input.properties;
284
+ if (typeof input.is_inline === 'boolean') body.is_inline = input.is_inline;
285
+ }
286
+ await this.requireApproval('Create a Notion data source', {
287
+ action: 'create_data_source',
288
+ parent: body.parent,
289
+ });
290
+ result = await notionRequest(settings, { method: 'POST', path: '/data_sources', body });
291
+ break;
292
+ }
293
+ case 'update_data_source': {
294
+ if (!input.data_source_id) throw new Error('Missing data_source_id for update_data_source');
295
+ const body = input.payload ? { ...input.payload } : {};
296
+ if (!input.payload) {
297
+ if (input.properties) body.properties = input.properties;
298
+ const title = this.buildTitle(input.title);
299
+ if (title) body.title = title;
300
+ if (Object.keys(body).length === 0) {
301
+ throw new Error('Missing update payload (properties or title)');
302
+ }
303
+ }
304
+ await this.requireApproval('Update a Notion data source', {
305
+ action: 'update_data_source',
306
+ data_source_id: input.data_source_id,
307
+ });
308
+ result = await notionRequest(settings, { method: 'PATCH', path: `/data_sources/${input.data_source_id}`, body });
309
+ break;
310
+ }
311
+ default:
312
+ throw new Error(`Unsupported action: ${action}`);
313
+ }
314
+
315
+ this.daemon.logEvent(this.taskId, 'tool_result', {
316
+ tool: 'notion_action',
317
+ action,
318
+ status: result?.status,
319
+ hasData: !!result?.data,
320
+ });
321
+
322
+ return {
323
+ success: true,
324
+ action,
325
+ status: result?.status,
326
+ data: result?.data,
327
+ raw: result?.raw,
328
+ };
329
+ }
330
+ }