@digitalpresence/cliclaw 0.1.0

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 (338) hide show
  1. package/dist/__tests__/calendar.integration.test.d.ts +2 -0
  2. package/dist/__tests__/calendar.integration.test.d.ts.map +1 -0
  3. package/dist/__tests__/calendar.integration.test.js +98 -0
  4. package/dist/__tests__/calendar.integration.test.js.map +1 -0
  5. package/dist/__tests__/forms.integration.test.d.ts +2 -0
  6. package/dist/__tests__/forms.integration.test.d.ts.map +1 -0
  7. package/dist/__tests__/forms.integration.test.js +243 -0
  8. package/dist/__tests__/forms.integration.test.js.map +1 -0
  9. package/dist/__tests__/gdrive.integration.test.d.ts +2 -0
  10. package/dist/__tests__/gdrive.integration.test.d.ts.map +1 -0
  11. package/dist/__tests__/gdrive.integration.test.js +186 -0
  12. package/dist/__tests__/gdrive.integration.test.js.map +1 -0
  13. package/dist/__tests__/gmail.integration.test.d.ts +2 -0
  14. package/dist/__tests__/gmail.integration.test.d.ts.map +1 -0
  15. package/dist/__tests__/gmail.integration.test.js +197 -0
  16. package/dist/__tests__/gmail.integration.test.js.map +1 -0
  17. package/dist/__tests__/gslides.integration.test.d.ts +2 -0
  18. package/dist/__tests__/gslides.integration.test.d.ts.map +1 -0
  19. package/dist/__tests__/gslides.integration.test.js +124 -0
  20. package/dist/__tests__/gslides.integration.test.js.map +1 -0
  21. package/dist/__tests__/sheets.integration.test.d.ts +2 -0
  22. package/dist/__tests__/sheets.integration.test.d.ts.map +1 -0
  23. package/dist/__tests__/sheets.integration.test.js +150 -0
  24. package/dist/__tests__/sheets.integration.test.js.map +1 -0
  25. package/dist/agent/crud.d.ts +6 -0
  26. package/dist/agent/crud.d.ts.map +1 -0
  27. package/dist/agent/crud.js +41 -0
  28. package/dist/agent/crud.js.map +1 -0
  29. package/dist/agent/memory.d.ts +14 -0
  30. package/dist/agent/memory.d.ts.map +1 -0
  31. package/dist/agent/memory.js +66 -0
  32. package/dist/agent/memory.js.map +1 -0
  33. package/dist/agent/permissions.d.ts +4 -0
  34. package/dist/agent/permissions.d.ts.map +1 -0
  35. package/dist/agent/permissions.js +32 -0
  36. package/dist/agent/permissions.js.map +1 -0
  37. package/dist/calendar/accounts.d.ts +3 -0
  38. package/dist/calendar/accounts.d.ts.map +1 -0
  39. package/dist/calendar/accounts.js +21 -0
  40. package/dist/calendar/accounts.js.map +1 -0
  41. package/dist/calendar/auth.d.ts +3 -0
  42. package/dist/calendar/auth.d.ts.map +1 -0
  43. package/dist/calendar/auth.js +41 -0
  44. package/dist/calendar/auth.js.map +1 -0
  45. package/dist/calendar/calendars.d.ts +3 -0
  46. package/dist/calendar/calendars.d.ts.map +1 -0
  47. package/dist/calendar/calendars.js +28 -0
  48. package/dist/calendar/calendars.js.map +1 -0
  49. package/dist/calendar/create.d.ts +3 -0
  50. package/dist/calendar/create.d.ts.map +1 -0
  51. package/dist/calendar/create.js +31 -0
  52. package/dist/calendar/create.js.map +1 -0
  53. package/dist/calendar/delete.d.ts +3 -0
  54. package/dist/calendar/delete.d.ts.map +1 -0
  55. package/dist/calendar/delete.js +21 -0
  56. package/dist/calendar/delete.js.map +1 -0
  57. package/dist/calendar/events.d.ts +3 -0
  58. package/dist/calendar/events.d.ts.map +1 -0
  59. package/dist/calendar/events.js +37 -0
  60. package/dist/calendar/events.js.map +1 -0
  61. package/dist/calendar/get.d.ts +3 -0
  62. package/dist/calendar/get.d.ts.map +1 -0
  63. package/dist/calendar/get.js +21 -0
  64. package/dist/calendar/get.js.map +1 -0
  65. package/dist/calendar/update.d.ts +3 -0
  66. package/dist/calendar/update.d.ts.map +1 -0
  67. package/dist/calendar/update.js +31 -0
  68. package/dist/calendar/update.js.map +1 -0
  69. package/dist/cli.d.ts +3 -0
  70. package/dist/cli.d.ts.map +1 -0
  71. package/dist/cli.js +37 -0
  72. package/dist/cli.js.map +1 -0
  73. package/dist/commands/agent.d.ts +6 -0
  74. package/dist/commands/agent.d.ts.map +1 -0
  75. package/dist/commands/agent.js +107 -0
  76. package/dist/commands/agent.js.map +1 -0
  77. package/dist/commands/calendar.d.ts +9 -0
  78. package/dist/commands/calendar.d.ts.map +1 -0
  79. package/dist/commands/calendar.js +104 -0
  80. package/dist/commands/calendar.js.map +1 -0
  81. package/dist/commands/cron.d.ts +6 -0
  82. package/dist/commands/cron.d.ts.map +1 -0
  83. package/dist/commands/cron.js +103 -0
  84. package/dist/commands/cron.js.map +1 -0
  85. package/dist/commands/forms.d.ts +9 -0
  86. package/dist/commands/forms.d.ts.map +1 -0
  87. package/dist/commands/forms.js +139 -0
  88. package/dist/commands/forms.js.map +1 -0
  89. package/dist/commands/gdrive.d.ts +9 -0
  90. package/dist/commands/gdrive.d.ts.map +1 -0
  91. package/dist/commands/gdrive.js +198 -0
  92. package/dist/commands/gdrive.js.map +1 -0
  93. package/dist/commands/gmail.d.ts +9 -0
  94. package/dist/commands/gmail.d.ts.map +1 -0
  95. package/dist/commands/gmail.js +231 -0
  96. package/dist/commands/gmail.js.map +1 -0
  97. package/dist/commands/gslides.d.ts +9 -0
  98. package/dist/commands/gslides.d.ts.map +1 -0
  99. package/dist/commands/gslides.js +167 -0
  100. package/dist/commands/gslides.js.map +1 -0
  101. package/dist/commands/sheets.d.ts +9 -0
  102. package/dist/commands/sheets.d.ts.map +1 -0
  103. package/dist/commands/sheets.js +174 -0
  104. package/dist/commands/sheets.js.map +1 -0
  105. package/dist/cron/daemon.d.ts +3 -0
  106. package/dist/cron/daemon.d.ts.map +1 -0
  107. package/dist/cron/daemon.js +127 -0
  108. package/dist/cron/daemon.js.map +1 -0
  109. package/dist/cron/handlers.d.ts +6 -0
  110. package/dist/cron/handlers.d.ts.map +1 -0
  111. package/dist/cron/handlers.js +58 -0
  112. package/dist/cron/handlers.js.map +1 -0
  113. package/dist/cron/logger.d.ts +4 -0
  114. package/dist/cron/logger.d.ts.map +1 -0
  115. package/dist/cron/logger.js +11 -0
  116. package/dist/cron/logger.js.map +1 -0
  117. package/dist/cron/progress.d.ts +25 -0
  118. package/dist/cron/progress.d.ts.map +1 -0
  119. package/dist/cron/progress.js +71 -0
  120. package/dist/cron/progress.js.map +1 -0
  121. package/dist/cron/ralph-wiggum.d.ts +18 -0
  122. package/dist/cron/ralph-wiggum.d.ts.map +1 -0
  123. package/dist/cron/ralph-wiggum.js +143 -0
  124. package/dist/cron/ralph-wiggum.js.map +1 -0
  125. package/dist/forms/accounts.d.ts +3 -0
  126. package/dist/forms/accounts.d.ts.map +1 -0
  127. package/dist/forms/accounts.js +20 -0
  128. package/dist/forms/accounts.js.map +1 -0
  129. package/dist/forms/auth.d.ts +3 -0
  130. package/dist/forms/auth.d.ts.map +1 -0
  131. package/dist/forms/auth.js +41 -0
  132. package/dist/forms/auth.js.map +1 -0
  133. package/dist/forms/create.d.ts +3 -0
  134. package/dist/forms/create.d.ts.map +1 -0
  135. package/dist/forms/create.js +28 -0
  136. package/dist/forms/create.js.map +1 -0
  137. package/dist/forms/get.d.ts +3 -0
  138. package/dist/forms/get.d.ts.map +1 -0
  139. package/dist/forms/get.js +21 -0
  140. package/dist/forms/get.js.map +1 -0
  141. package/dist/forms/list.d.ts +3 -0
  142. package/dist/forms/list.d.ts.map +1 -0
  143. package/dist/forms/list.js +33 -0
  144. package/dist/forms/list.js.map +1 -0
  145. package/dist/forms/questions.d.ts +6 -0
  146. package/dist/forms/questions.d.ts.map +1 -0
  147. package/dist/forms/questions.js +179 -0
  148. package/dist/forms/questions.js.map +1 -0
  149. package/dist/forms/responses.d.ts +4 -0
  150. package/dist/forms/responses.d.ts.map +1 -0
  151. package/dist/forms/responses.js +40 -0
  152. package/dist/forms/responses.js.map +1 -0
  153. package/dist/forms/update.d.ts +3 -0
  154. package/dist/forms/update.d.ts.map +1 -0
  155. package/dist/forms/update.js +44 -0
  156. package/dist/forms/update.js.map +1 -0
  157. package/dist/gdrive/about.d.ts +3 -0
  158. package/dist/gdrive/about.d.ts.map +1 -0
  159. package/dist/gdrive/about.js +31 -0
  160. package/dist/gdrive/about.js.map +1 -0
  161. package/dist/gdrive/accounts.d.ts +3 -0
  162. package/dist/gdrive/accounts.d.ts.map +1 -0
  163. package/dist/gdrive/accounts.js +20 -0
  164. package/dist/gdrive/accounts.js.map +1 -0
  165. package/dist/gdrive/auth.d.ts +3 -0
  166. package/dist/gdrive/auth.d.ts.map +1 -0
  167. package/dist/gdrive/auth.js +51 -0
  168. package/dist/gdrive/auth.js.map +1 -0
  169. package/dist/gdrive/files.d.ts +12 -0
  170. package/dist/gdrive/files.d.ts.map +1 -0
  171. package/dist/gdrive/files.js +174 -0
  172. package/dist/gdrive/files.js.map +1 -0
  173. package/dist/gdrive/folders.d.ts +4 -0
  174. package/dist/gdrive/folders.d.ts.map +1 -0
  175. package/dist/gdrive/folders.js +46 -0
  176. package/dist/gdrive/folders.js.map +1 -0
  177. package/dist/gdrive/search.d.ts +3 -0
  178. package/dist/gdrive/search.d.ts.map +1 -0
  179. package/dist/gdrive/search.js +23 -0
  180. package/dist/gdrive/search.js.map +1 -0
  181. package/dist/gdrive/sharing.d.ts +5 -0
  182. package/dist/gdrive/sharing.d.ts.map +1 -0
  183. package/dist/gdrive/sharing.js +54 -0
  184. package/dist/gdrive/sharing.js.map +1 -0
  185. package/dist/gmail/accounts.d.ts +3 -0
  186. package/dist/gmail/accounts.d.ts.map +1 -0
  187. package/dist/gmail/accounts.js +18 -0
  188. package/dist/gmail/accounts.js.map +1 -0
  189. package/dist/gmail/auth.d.ts +3 -0
  190. package/dist/gmail/auth.d.ts.map +1 -0
  191. package/dist/gmail/auth.js +50 -0
  192. package/dist/gmail/auth.js.map +1 -0
  193. package/dist/gmail/drafts.d.ts +7 -0
  194. package/dist/gmail/drafts.d.ts.map +1 -0
  195. package/dist/gmail/drafts.js +103 -0
  196. package/dist/gmail/drafts.js.map +1 -0
  197. package/dist/gmail/get.d.ts +4 -0
  198. package/dist/gmail/get.d.ts.map +1 -0
  199. package/dist/gmail/get.js +140 -0
  200. package/dist/gmail/get.js.map +1 -0
  201. package/dist/gmail/inbox.d.ts +4 -0
  202. package/dist/gmail/inbox.d.ts.map +1 -0
  203. package/dist/gmail/inbox.js +45 -0
  204. package/dist/gmail/inbox.js.map +1 -0
  205. package/dist/gmail/labels.d.ts +5 -0
  206. package/dist/gmail/labels.d.ts.map +1 -0
  207. package/dist/gmail/labels.js +45 -0
  208. package/dist/gmail/labels.js.map +1 -0
  209. package/dist/gmail/modify.d.ts +5 -0
  210. package/dist/gmail/modify.d.ts.map +1 -0
  211. package/dist/gmail/modify.js +68 -0
  212. package/dist/gmail/modify.js.map +1 -0
  213. package/dist/gmail/send.d.ts +5 -0
  214. package/dist/gmail/send.d.ts.map +1 -0
  215. package/dist/gmail/send.js +310 -0
  216. package/dist/gmail/send.js.map +1 -0
  217. package/dist/gmail/threads.d.ts +4 -0
  218. package/dist/gmail/threads.d.ts.map +1 -0
  219. package/dist/gmail/threads.js +47 -0
  220. package/dist/gmail/threads.js.map +1 -0
  221. package/dist/gslides/accounts.d.ts +3 -0
  222. package/dist/gslides/accounts.d.ts.map +1 -0
  223. package/dist/gslides/accounts.js +20 -0
  224. package/dist/gslides/accounts.js.map +1 -0
  225. package/dist/gslides/auth.d.ts +3 -0
  226. package/dist/gslides/auth.d.ts.map +1 -0
  227. package/dist/gslides/auth.js +50 -0
  228. package/dist/gslides/auth.js.map +1 -0
  229. package/dist/gslides/presentations.d.ts +29 -0
  230. package/dist/gslides/presentations.d.ts.map +1 -0
  231. package/dist/gslides/presentations.js +320 -0
  232. package/dist/gslides/presentations.js.map +1 -0
  233. package/dist/lib/config.d.ts +6 -0
  234. package/dist/lib/config.d.ts.map +1 -0
  235. package/dist/lib/config.js +12 -0
  236. package/dist/lib/config.js.map +1 -0
  237. package/dist/lib/media-utils.d.ts +5 -0
  238. package/dist/lib/media-utils.d.ts.map +1 -0
  239. package/dist/lib/media-utils.js +79 -0
  240. package/dist/lib/media-utils.js.map +1 -0
  241. package/dist/lib/output.d.ts +4 -0
  242. package/dist/lib/output.d.ts.map +1 -0
  243. package/dist/lib/output.js +12 -0
  244. package/dist/lib/output.js.map +1 -0
  245. package/dist/sheets/accounts.d.ts +3 -0
  246. package/dist/sheets/accounts.d.ts.map +1 -0
  247. package/dist/sheets/accounts.js +20 -0
  248. package/dist/sheets/accounts.js.map +1 -0
  249. package/dist/sheets/auth.d.ts +3 -0
  250. package/dist/sheets/auth.d.ts.map +1 -0
  251. package/dist/sheets/auth.js +56 -0
  252. package/dist/sheets/auth.js.map +1 -0
  253. package/dist/sheets/cells.d.ts +6 -0
  254. package/dist/sheets/cells.d.ts.map +1 -0
  255. package/dist/sheets/cells.js +89 -0
  256. package/dist/sheets/cells.js.map +1 -0
  257. package/dist/sheets/format.d.ts +8 -0
  258. package/dist/sheets/format.d.ts.map +1 -0
  259. package/dist/sheets/format.js +97 -0
  260. package/dist/sheets/format.js.map +1 -0
  261. package/dist/sheets/sheets-tab.d.ts +6 -0
  262. package/dist/sheets/sheets-tab.d.ts.map +1 -0
  263. package/dist/sheets/sheets-tab.js +88 -0
  264. package/dist/sheets/sheets-tab.js.map +1 -0
  265. package/dist/sheets/spreadsheets.d.ts +6 -0
  266. package/dist/sheets/spreadsheets.d.ts.map +1 -0
  267. package/dist/sheets/spreadsheets.js +88 -0
  268. package/dist/sheets/spreadsheets.js.map +1 -0
  269. package/package.json +33 -0
  270. package/src/__tests__/calendar.integration.test.ts +152 -0
  271. package/src/__tests__/forms.integration.test.ts +403 -0
  272. package/src/__tests__/gdrive.integration.test.ts +253 -0
  273. package/src/__tests__/gmail.integration.test.ts +294 -0
  274. package/src/__tests__/gslides.integration.test.ts +195 -0
  275. package/src/__tests__/sheets.integration.test.ts +234 -0
  276. package/src/agent/crud.ts +54 -0
  277. package/src/agent/memory.ts +95 -0
  278. package/src/agent/permissions.ts +54 -0
  279. package/src/calendar/accounts.ts +25 -0
  280. package/src/calendar/auth.ts +45 -0
  281. package/src/calendar/calendars.ts +32 -0
  282. package/src/calendar/create.ts +44 -0
  283. package/src/calendar/delete.ts +27 -0
  284. package/src/calendar/events.ts +45 -0
  285. package/src/calendar/get.ts +27 -0
  286. package/src/calendar/update.ts +44 -0
  287. package/src/cli.ts +41 -0
  288. package/src/commands/agent.ts +128 -0
  289. package/src/commands/calendar.ts +127 -0
  290. package/src/commands/cron.ts +126 -0
  291. package/src/commands/forms.ts +158 -0
  292. package/src/commands/gdrive.ts +225 -0
  293. package/src/commands/gmail.ts +290 -0
  294. package/src/commands/gslides.ts +188 -0
  295. package/src/commands/sheets.ts +193 -0
  296. package/src/cron/daemon.ts +143 -0
  297. package/src/cron/handlers.ts +78 -0
  298. package/src/cron/logger.ts +20 -0
  299. package/src/cron/progress.ts +90 -0
  300. package/src/cron/ralph-wiggum.ts +172 -0
  301. package/src/forms/accounts.ts +24 -0
  302. package/src/forms/auth.ts +45 -0
  303. package/src/forms/create.ts +34 -0
  304. package/src/forms/get.ts +26 -0
  305. package/src/forms/list.ts +38 -0
  306. package/src/forms/questions.ts +209 -0
  307. package/src/forms/responses.ts +50 -0
  308. package/src/forms/update.ts +57 -0
  309. package/src/gdrive/about.ts +36 -0
  310. package/src/gdrive/accounts.ts +24 -0
  311. package/src/gdrive/auth.ts +55 -0
  312. package/src/gdrive/files.ts +237 -0
  313. package/src/gdrive/folders.ts +58 -0
  314. package/src/gdrive/search.ts +30 -0
  315. package/src/gdrive/sharing.ts +72 -0
  316. package/src/gmail/accounts.ts +22 -0
  317. package/src/gmail/auth.ts +53 -0
  318. package/src/gmail/drafts.ts +166 -0
  319. package/src/gmail/get.ts +195 -0
  320. package/src/gmail/inbox.ts +78 -0
  321. package/src/gmail/labels.ts +69 -0
  322. package/src/gmail/modify.ts +89 -0
  323. package/src/gmail/send.ts +424 -0
  324. package/src/gmail/threads.ts +68 -0
  325. package/src/gslides/accounts.ts +24 -0
  326. package/src/gslides/auth.ts +54 -0
  327. package/src/gslides/presentations.ts +384 -0
  328. package/src/lib/config.ts +14 -0
  329. package/src/lib/media-utils.ts +82 -0
  330. package/src/lib/output.ts +13 -0
  331. package/src/sheets/accounts.ts +24 -0
  332. package/src/sheets/auth.ts +60 -0
  333. package/src/sheets/cells.ts +112 -0
  334. package/src/sheets/format.ts +114 -0
  335. package/src/sheets/sheets-tab.ts +109 -0
  336. package/src/sheets/spreadsheets.ts +106 -0
  337. package/tsconfig.json +8 -0
  338. package/vitest.config.ts +7 -0
@@ -0,0 +1,166 @@
1
+ import { google } from "googleapis";
2
+ import type { OAuthClientManager } from "@digitalpresence/cliclaw-auth";
3
+ import { outputJson, outputAuthRequired } from "../lib/output.js";
4
+
5
+ function encodeDraftMessage(headers: Record<string, string>, body: string): string {
6
+ const headerLines = Object.entries(headers)
7
+ .filter(([, v]) => v)
8
+ .map(([k, v]) => `${k}: ${v}`)
9
+ .join("\r\n");
10
+ const raw = `${headerLines}\r\n\r\n${body}`;
11
+ return Buffer.from(raw).toString("base64url");
12
+ }
13
+
14
+ export async function handleDraftList(
15
+ clientManager: OAuthClientManager,
16
+ account: string,
17
+ maxResults: number,
18
+ ): Promise<void> {
19
+ if (!clientManager.listAccounts().includes(account)) {
20
+ outputAuthRequired();
21
+ }
22
+
23
+ const client = clientManager.getClient(account);
24
+ const gmail = google.gmail({ version: "v1", auth: client });
25
+
26
+ const listRes = await gmail.users.drafts.list({ userId: "me", maxResults });
27
+ const drafts = listRes.data.drafts ?? [];
28
+
29
+ if (drafts.length === 0) {
30
+ outputJson([]);
31
+ return;
32
+ }
33
+
34
+ const summaries = await Promise.all(
35
+ drafts.map(async (draft) => {
36
+ const detail = await gmail.users.drafts.get({
37
+ userId: "me",
38
+ id: draft.id!,
39
+ format: "metadata",
40
+ });
41
+ const headers = detail.data.message?.payload?.headers ?? [];
42
+ const get = (name: string) =>
43
+ headers.find((h) => h.name?.toLowerCase() === name.toLowerCase())?.value ?? "";
44
+ return {
45
+ id: draft.id ?? "",
46
+ messageId: detail.data.message?.id ?? "",
47
+ subject: get("Subject"),
48
+ to: get("To"),
49
+ snippet: detail.data.message?.snippet ?? "",
50
+ };
51
+ }),
52
+ );
53
+
54
+ outputJson(summaries);
55
+ }
56
+
57
+ export async function handleDraftCreate(
58
+ clientManager: OAuthClientManager,
59
+ account: string,
60
+ to: string,
61
+ subject: string,
62
+ body: string,
63
+ cc?: string,
64
+ bcc?: string,
65
+ ): Promise<void> {
66
+ if (!clientManager.listAccounts().includes(account)) {
67
+ outputAuthRequired();
68
+ }
69
+
70
+ const client = clientManager.getClient(account);
71
+ const gmail = google.gmail({ version: "v1", auth: client });
72
+
73
+ const raw = encodeDraftMessage(
74
+ {
75
+ To: to,
76
+ Subject: subject,
77
+ Cc: cc ?? "",
78
+ Bcc: bcc ?? "",
79
+ "Content-Type": "text/plain; charset=utf-8",
80
+ "MIME-Version": "1.0",
81
+ },
82
+ body,
83
+ );
84
+
85
+ const res = await gmail.users.drafts.create({
86
+ userId: "me",
87
+ requestBody: { message: { raw } },
88
+ });
89
+
90
+ outputJson({ id: res.data.id, messageId: res.data.message?.id, success: true });
91
+ }
92
+
93
+ export async function handleDraftUpdate(
94
+ clientManager: OAuthClientManager,
95
+ account: string,
96
+ draftId: string,
97
+ to: string,
98
+ subject: string,
99
+ body: string,
100
+ cc?: string,
101
+ bcc?: string,
102
+ ): Promise<void> {
103
+ if (!clientManager.listAccounts().includes(account)) {
104
+ outputAuthRequired();
105
+ }
106
+
107
+ const client = clientManager.getClient(account);
108
+ const gmail = google.gmail({ version: "v1", auth: client });
109
+
110
+ const raw = encodeDraftMessage(
111
+ {
112
+ To: to,
113
+ Subject: subject,
114
+ Cc: cc ?? "",
115
+ Bcc: bcc ?? "",
116
+ "Content-Type": "text/plain; charset=utf-8",
117
+ "MIME-Version": "1.0",
118
+ },
119
+ body,
120
+ );
121
+
122
+ const res = await gmail.users.drafts.update({
123
+ userId: "me",
124
+ id: draftId,
125
+ requestBody: { message: { raw } },
126
+ });
127
+
128
+ outputJson({ id: res.data.id, messageId: res.data.message?.id, success: true });
129
+ }
130
+
131
+ export async function handleDraftDelete(
132
+ clientManager: OAuthClientManager,
133
+ account: string,
134
+ draftId: string,
135
+ ): Promise<void> {
136
+ if (!clientManager.listAccounts().includes(account)) {
137
+ outputAuthRequired();
138
+ }
139
+
140
+ const client = clientManager.getClient(account);
141
+ const gmail = google.gmail({ version: "v1", auth: client });
142
+
143
+ await gmail.users.drafts.delete({ userId: "me", id: draftId });
144
+
145
+ outputJson({ draft_id: draftId, success: true });
146
+ }
147
+
148
+ export async function handleDraftSend(
149
+ clientManager: OAuthClientManager,
150
+ account: string,
151
+ draftId: string,
152
+ ): Promise<void> {
153
+ if (!clientManager.listAccounts().includes(account)) {
154
+ outputAuthRequired();
155
+ }
156
+
157
+ const client = clientManager.getClient(account);
158
+ const gmail = google.gmail({ version: "v1", auth: client });
159
+
160
+ const res = await gmail.users.drafts.send({
161
+ userId: "me",
162
+ requestBody: { id: draftId },
163
+ });
164
+
165
+ outputJson({ id: res.data.id, threadId: res.data.threadId, success: true });
166
+ }
@@ -0,0 +1,195 @@
1
+ import { writeFileSync } from "fs";
2
+ import * as path from "path";
3
+ import { google } from "googleapis";
4
+ import type { OAuthClientManager } from "@digitalpresence/cliclaw-auth";
5
+ import { ensureDir, sanitizeFilename, extToMime } from "../lib/media-utils.js";
6
+ import { outputJson, outputAuthRequired, outputError } from "../lib/output.js";
7
+
8
+ interface MessagePart {
9
+ partId?: string | null;
10
+ mimeType?: string | null;
11
+ filename?: string | null;
12
+ headers?: Array<{ name?: string | null; value?: string | null }> | null;
13
+ body?: { data?: string | null; attachmentId?: string | null; size?: number | null } | null;
14
+ parts?: MessagePart[] | null;
15
+ }
16
+
17
+ interface AttachmentMeta {
18
+ filename: string;
19
+ mimeType: string;
20
+ size: number;
21
+ attachmentId: string;
22
+ partId: string;
23
+ }
24
+
25
+ function extractBody(parts: MessagePart[] | null | undefined): { text: string; html: string } {
26
+ let text = "";
27
+ let html = "";
28
+
29
+ function traverse(part: MessagePart) {
30
+ if (part.mimeType === "text/plain" && part.body?.data) {
31
+ text += Buffer.from(part.body.data, "base64").toString("utf-8");
32
+ } else if (part.mimeType === "text/html" && part.body?.data) {
33
+ html += Buffer.from(part.body.data, "base64").toString("utf-8");
34
+ }
35
+ if (part.parts) {
36
+ for (const child of part.parts) {
37
+ traverse(child);
38
+ }
39
+ }
40
+ }
41
+
42
+ if (parts) {
43
+ for (const part of parts) {
44
+ traverse(part);
45
+ }
46
+ }
47
+
48
+ return { text, html };
49
+ }
50
+
51
+ function extractAttachments(parts: MessagePart[] | null | undefined): AttachmentMeta[] {
52
+ const results: AttachmentMeta[] = [];
53
+
54
+ function traverse(part: MessagePart) {
55
+ const attachmentId = part.body?.attachmentId ?? null;
56
+ const filenameHeader = part.headers?.find(
57
+ (h) => h.name?.toLowerCase() === "content-disposition",
58
+ )?.value;
59
+
60
+ let filename = part.filename ?? null;
61
+ if (!filename && filenameHeader) {
62
+ const match = filenameHeader.match(/filename\*?=(?:UTF-8'')?["']?([^"';\r\n]+)["']?/i);
63
+ if (match) {
64
+ filename = decodeURIComponent(match[1].trim());
65
+ }
66
+ }
67
+
68
+ if (attachmentId && filename) {
69
+ results.push({
70
+ filename,
71
+ mimeType: part.mimeType ?? "application/octet-stream",
72
+ size: part.body?.size ?? 0,
73
+ attachmentId,
74
+ partId: part.partId ?? "",
75
+ });
76
+ }
77
+
78
+ if (part.parts) {
79
+ for (const child of part.parts) {
80
+ traverse(child);
81
+ }
82
+ }
83
+ }
84
+
85
+ if (parts) {
86
+ for (const part of parts) {
87
+ traverse(part);
88
+ }
89
+ }
90
+
91
+ return results;
92
+ }
93
+
94
+ export async function handleGet(
95
+ clientManager: OAuthClientManager,
96
+ account: string,
97
+ id: string,
98
+ ): Promise<void> {
99
+ if (!clientManager.listAccounts().includes(account)) {
100
+ outputAuthRequired();
101
+ }
102
+
103
+ const client = clientManager.getClient(account);
104
+ const gmail = google.gmail({ version: "v1", auth: client });
105
+
106
+ const res = await gmail.users.messages.get({
107
+ userId: "me",
108
+ id,
109
+ format: "full",
110
+ });
111
+
112
+ const payload = res.data.payload;
113
+ const headers = payload?.headers ?? [];
114
+ const get = (name: string) =>
115
+ headers.find((h) => h.name?.toLowerCase() === name.toLowerCase())?.value ?? "";
116
+
117
+ let bodyText = "";
118
+ let bodyHtml = "";
119
+
120
+ if (payload?.body?.data) {
121
+ const decoded = Buffer.from(payload.body.data, "base64").toString("utf-8");
122
+ if (payload.mimeType === "text/html") {
123
+ bodyHtml = decoded;
124
+ } else {
125
+ bodyText = decoded;
126
+ }
127
+ } else {
128
+ const { text, html } = extractBody(payload?.parts as MessagePart[] | undefined);
129
+ bodyText = text;
130
+ bodyHtml = html;
131
+ }
132
+
133
+ const attachments = extractAttachments(payload?.parts as MessagePart[] | undefined);
134
+
135
+ outputJson({
136
+ id: res.data.id ?? id,
137
+ threadId: res.data.threadId ?? "",
138
+ subject: get("Subject"),
139
+ from: get("From"),
140
+ to: get("To"),
141
+ cc: get("Cc"),
142
+ bcc: get("Bcc"),
143
+ reply_to: get("Reply-To"),
144
+ date: get("Date"),
145
+ snippet: res.data.snippet ?? "",
146
+ labelIds: res.data.labelIds ?? [],
147
+ body_text: bodyText,
148
+ body_html: bodyHtml,
149
+ attachments,
150
+ });
151
+ }
152
+
153
+ export async function handleDownloadAttachment(
154
+ clientManager: OAuthClientManager,
155
+ account: string,
156
+ id: string,
157
+ attachmentId: string,
158
+ filename: string,
159
+ saveDir: string,
160
+ ): Promise<void> {
161
+ if (!clientManager.listAccounts().includes(account)) {
162
+ outputAuthRequired();
163
+ }
164
+
165
+ const client = clientManager.getClient(account);
166
+ const gmail = google.gmail({ version: "v1", auth: client });
167
+
168
+ const res = await gmail.users.messages.attachments.get({
169
+ userId: "me",
170
+ messageId: id,
171
+ id: attachmentId,
172
+ });
173
+
174
+ const data = res.data.data;
175
+ if (!data) {
176
+ outputError("no_data", "Attachment returned no data.");
177
+ }
178
+
179
+ const buffer = Buffer.from(data!, "base64url");
180
+
181
+ ensureDir(saveDir);
182
+ const safeFilename = sanitizeFilename(filename);
183
+ const filePath = path.join(saveDir, safeFilename);
184
+ writeFileSync(filePath, buffer);
185
+
186
+ const ext = path.extname(safeFilename);
187
+ const mimeType = extToMime(ext) || "application/octet-stream";
188
+
189
+ outputJson({
190
+ path: filePath,
191
+ filename: safeFilename,
192
+ mimeType,
193
+ size: buffer.length,
194
+ });
195
+ }
@@ -0,0 +1,78 @@
1
+ import { google } from "googleapis";
2
+ import type { OAuthClientManager } from "@digitalpresence/cliclaw-auth";
3
+ import { outputJson, outputAuthRequired } from "../lib/output.js";
4
+
5
+ interface MessageSummary {
6
+ id: string;
7
+ subject: string;
8
+ from: string;
9
+ date: string;
10
+ snippet: string;
11
+ }
12
+
13
+ async function fetchMessages(
14
+ clientManager: OAuthClientManager,
15
+ account: string,
16
+ maxResults: number,
17
+ query?: string,
18
+ labelIds?: string[],
19
+ ): Promise<MessageSummary[]> {
20
+ if (!clientManager.listAccounts().includes(account)) {
21
+ outputAuthRequired();
22
+ }
23
+
24
+ const client = clientManager.getClient(account);
25
+ const gmail = google.gmail({ version: "v1", auth: client });
26
+
27
+ const listRes = await gmail.users.messages.list({
28
+ userId: "me",
29
+ ...(labelIds ? { labelIds } : {}),
30
+ maxResults,
31
+ ...(query ? { q: query } : {}),
32
+ });
33
+
34
+ const messages = listRes.data.messages ?? [];
35
+ if (messages.length === 0) return [];
36
+
37
+ const summaries = await Promise.all(
38
+ messages.map(async (msg) => {
39
+ const detail = await gmail.users.messages.get({
40
+ userId: "me",
41
+ id: msg.id!,
42
+ format: "metadata",
43
+ metadataHeaders: ["Subject", "From", "Date"],
44
+ });
45
+ const headers = detail.data.payload?.headers ?? [];
46
+ const get = (name: string) =>
47
+ headers.find((h) => h.name?.toLowerCase() === name.toLowerCase())?.value ?? "";
48
+ return {
49
+ id: msg.id ?? "",
50
+ subject: get("Subject"),
51
+ from: get("From"),
52
+ date: get("Date"),
53
+ snippet: detail.data.snippet ?? "",
54
+ };
55
+ }),
56
+ );
57
+
58
+ return summaries;
59
+ }
60
+
61
+ export async function handleInbox(
62
+ clientManager: OAuthClientManager,
63
+ account: string,
64
+ maxResults: number,
65
+ ): Promise<void> {
66
+ const result = await fetchMessages(clientManager, account, maxResults, undefined, ["INBOX"]);
67
+ outputJson(result);
68
+ }
69
+
70
+ export async function handleSearch(
71
+ clientManager: OAuthClientManager,
72
+ account: string,
73
+ query: string,
74
+ maxResults: number,
75
+ ): Promise<void> {
76
+ const result = await fetchMessages(clientManager, account, maxResults, query);
77
+ outputJson(result);
78
+ }
@@ -0,0 +1,69 @@
1
+ import { google } from "googleapis";
2
+ import type { OAuthClientManager } from "@digitalpresence/cliclaw-auth";
3
+ import { outputJson, outputAuthRequired } from "../lib/output.js";
4
+
5
+ export async function handleLabelsList(
6
+ clientManager: OAuthClientManager,
7
+ account: string,
8
+ ): Promise<void> {
9
+ if (!clientManager.listAccounts().includes(account)) {
10
+ outputAuthRequired();
11
+ }
12
+
13
+ const client = clientManager.getClient(account);
14
+ const gmail = google.gmail({ version: "v1", auth: client });
15
+
16
+ const res = await gmail.users.labels.list({ userId: "me" });
17
+ const labels = res.data.labels ?? [];
18
+
19
+ const result = labels.map((label) => ({
20
+ id: label.id ?? "",
21
+ name: label.name ?? "",
22
+ type: label.type ?? "",
23
+ messagesTotal: label.messagesTotal ?? 0,
24
+ messagesUnread: label.messagesUnread ?? 0,
25
+ }));
26
+
27
+ outputJson(result);
28
+ }
29
+
30
+ export async function handleLabelCreate(
31
+ clientManager: OAuthClientManager,
32
+ account: string,
33
+ name: string,
34
+ ): Promise<void> {
35
+ if (!clientManager.listAccounts().includes(account)) {
36
+ outputAuthRequired();
37
+ }
38
+
39
+ const client = clientManager.getClient(account);
40
+ const gmail = google.gmail({ version: "v1", auth: client });
41
+
42
+ const res = await gmail.users.labels.create({
43
+ userId: "me",
44
+ requestBody: {
45
+ name,
46
+ labelListVisibility: "labelShow",
47
+ messageListVisibility: "show",
48
+ },
49
+ });
50
+
51
+ outputJson({ id: res.data.id, name: res.data.name, success: true });
52
+ }
53
+
54
+ export async function handleLabelDelete(
55
+ clientManager: OAuthClientManager,
56
+ account: string,
57
+ labelId: string,
58
+ ): Promise<void> {
59
+ if (!clientManager.listAccounts().includes(account)) {
60
+ outputAuthRequired();
61
+ }
62
+
63
+ const client = clientManager.getClient(account);
64
+ const gmail = google.gmail({ version: "v1", auth: client });
65
+
66
+ await gmail.users.labels.delete({ userId: "me", id: labelId });
67
+
68
+ outputJson({ label_id: labelId, success: true });
69
+ }
@@ -0,0 +1,89 @@
1
+ import { google } from "googleapis";
2
+ import type { OAuthClientManager } from "@digitalpresence/cliclaw-auth";
3
+ import { outputJson, outputAuthRequired } from "../lib/output.js";
4
+
5
+ type ModifyAction =
6
+ | "mark_read"
7
+ | "mark_unread"
8
+ | "archive"
9
+ | "trash"
10
+ | "untrash"
11
+ | "star"
12
+ | "unstar"
13
+ | "add_labels"
14
+ | "remove_labels";
15
+
16
+ export async function handleModify(
17
+ clientManager: OAuthClientManager,
18
+ account: string,
19
+ id: string,
20
+ action: ModifyAction,
21
+ labelIds?: string[],
22
+ ): Promise<void> {
23
+ if (!clientManager.listAccounts().includes(account)) {
24
+ outputAuthRequired();
25
+ }
26
+
27
+ const client = clientManager.getClient(account);
28
+ const gmail = google.gmail({ version: "v1", auth: client });
29
+
30
+ switch (action) {
31
+ case "mark_read":
32
+ await gmail.users.messages.modify({
33
+ userId: "me",
34
+ id,
35
+ requestBody: { removeLabelIds: ["UNREAD"] },
36
+ });
37
+ break;
38
+ case "mark_unread":
39
+ await gmail.users.messages.modify({
40
+ userId: "me",
41
+ id,
42
+ requestBody: { addLabelIds: ["UNREAD"] },
43
+ });
44
+ break;
45
+ case "archive":
46
+ await gmail.users.messages.modify({
47
+ userId: "me",
48
+ id,
49
+ requestBody: { removeLabelIds: ["INBOX"] },
50
+ });
51
+ break;
52
+ case "trash":
53
+ await gmail.users.messages.trash({ userId: "me", id });
54
+ break;
55
+ case "untrash":
56
+ await gmail.users.messages.untrash({ userId: "me", id });
57
+ break;
58
+ case "star":
59
+ await gmail.users.messages.modify({
60
+ userId: "me",
61
+ id,
62
+ requestBody: { addLabelIds: ["STARRED"] },
63
+ });
64
+ break;
65
+ case "unstar":
66
+ await gmail.users.messages.modify({
67
+ userId: "me",
68
+ id,
69
+ requestBody: { removeLabelIds: ["STARRED"] },
70
+ });
71
+ break;
72
+ case "add_labels":
73
+ await gmail.users.messages.modify({
74
+ userId: "me",
75
+ id,
76
+ requestBody: { addLabelIds: labelIds ?? [] },
77
+ });
78
+ break;
79
+ case "remove_labels":
80
+ await gmail.users.messages.modify({
81
+ userId: "me",
82
+ id,
83
+ requestBody: { removeLabelIds: labelIds ?? [] },
84
+ });
85
+ break;
86
+ }
87
+
88
+ outputJson({ id, action, success: true });
89
+ }