@lhremote/mcp 0.0.0 → 0.2.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 (299) hide show
  1. package/LICENSE +661 -0
  2. package/README.md +58 -0
  3. package/dist/helpers.d.ts +59 -0
  4. package/dist/helpers.d.ts.map +1 -0
  5. package/dist/helpers.js +90 -0
  6. package/dist/helpers.js.map +1 -0
  7. package/dist/index.d.ts +1 -0
  8. package/dist/index.d.ts.map +1 -1
  9. package/dist/index.js +5 -1
  10. package/dist/index.js.map +1 -1
  11. package/dist/server.d.ts +6 -0
  12. package/dist/server.d.ts.map +1 -0
  13. package/dist/server.js +16 -0
  14. package/dist/server.js.map +1 -0
  15. package/dist/server.test.d.ts +2 -0
  16. package/dist/server.test.d.ts.map +1 -0
  17. package/dist/server.test.js +85 -0
  18. package/dist/server.test.js.map +1 -0
  19. package/dist/stdio.d.ts +7 -0
  20. package/dist/stdio.d.ts.map +1 -0
  21. package/dist/stdio.js +37 -0
  22. package/dist/stdio.js.map +1 -0
  23. package/dist/tools/campaign-add-action.d.ts +4 -0
  24. package/dist/tools/campaign-add-action.d.ts.map +1 -0
  25. package/dist/tools/campaign-add-action.js +63 -0
  26. package/dist/tools/campaign-add-action.js.map +1 -0
  27. package/dist/tools/campaign-add-action.test.d.ts +2 -0
  28. package/dist/tools/campaign-add-action.test.d.ts.map +1 -0
  29. package/dist/tools/campaign-add-action.test.js +107 -0
  30. package/dist/tools/campaign-add-action.test.js.map +1 -0
  31. package/dist/tools/campaign-create.d.ts +4 -0
  32. package/dist/tools/campaign-create.d.ts.map +1 -0
  33. package/dist/tools/campaign-create.js +44 -0
  34. package/dist/tools/campaign-create.js.map +1 -0
  35. package/dist/tools/campaign-create.test.d.ts +2 -0
  36. package/dist/tools/campaign-create.test.d.ts.map +1 -0
  37. package/dist/tools/campaign-create.test.js +163 -0
  38. package/dist/tools/campaign-create.test.js.map +1 -0
  39. package/dist/tools/campaign-delete.d.ts +4 -0
  40. package/dist/tools/campaign-delete.d.ts.map +1 -0
  41. package/dist/tools/campaign-delete.js +28 -0
  42. package/dist/tools/campaign-delete.js.map +1 -0
  43. package/dist/tools/campaign-delete.test.d.ts +2 -0
  44. package/dist/tools/campaign-delete.test.d.ts.map +1 -0
  45. package/dist/tools/campaign-delete.test.js +87 -0
  46. package/dist/tools/campaign-delete.test.js.map +1 -0
  47. package/dist/tools/campaign-exclude-add.d.ts +4 -0
  48. package/dist/tools/campaign-exclude-add.d.ts.map +1 -0
  49. package/dist/tools/campaign-exclude-add.js +44 -0
  50. package/dist/tools/campaign-exclude-add.js.map +1 -0
  51. package/dist/tools/campaign-exclude-add.test.d.ts +2 -0
  52. package/dist/tools/campaign-exclude-add.test.d.ts.map +1 -0
  53. package/dist/tools/campaign-exclude-add.test.js +161 -0
  54. package/dist/tools/campaign-exclude-add.test.js.map +1 -0
  55. package/dist/tools/campaign-exclude-list.d.ts +4 -0
  56. package/dist/tools/campaign-exclude-list.d.ts.map +1 -0
  57. package/dist/tools/campaign-exclude-list.js +40 -0
  58. package/dist/tools/campaign-exclude-list.js.map +1 -0
  59. package/dist/tools/campaign-exclude-list.test.d.ts +2 -0
  60. package/dist/tools/campaign-exclude-list.test.d.ts.map +1 -0
  61. package/dist/tools/campaign-exclude-list.test.js +183 -0
  62. package/dist/tools/campaign-exclude-list.test.js.map +1 -0
  63. package/dist/tools/campaign-exclude-remove.d.ts +4 -0
  64. package/dist/tools/campaign-exclude-remove.d.ts.map +1 -0
  65. package/dist/tools/campaign-exclude-remove.js +44 -0
  66. package/dist/tools/campaign-exclude-remove.js.map +1 -0
  67. package/dist/tools/campaign-exclude-remove.test.d.ts +2 -0
  68. package/dist/tools/campaign-exclude-remove.test.d.ts.map +1 -0
  69. package/dist/tools/campaign-exclude-remove.test.js +161 -0
  70. package/dist/tools/campaign-exclude-remove.test.js.map +1 -0
  71. package/dist/tools/campaign-export.d.ts +4 -0
  72. package/dist/tools/campaign-export.d.ts.map +1 -0
  73. package/dist/tools/campaign-export.js +30 -0
  74. package/dist/tools/campaign-export.js.map +1 -0
  75. package/dist/tools/campaign-export.test.d.ts +2 -0
  76. package/dist/tools/campaign-export.test.d.ts.map +1 -0
  77. package/dist/tools/campaign-export.test.js +80 -0
  78. package/dist/tools/campaign-export.test.js.map +1 -0
  79. package/dist/tools/campaign-get.d.ts +4 -0
  80. package/dist/tools/campaign-get.d.ts.map +1 -0
  81. package/dist/tools/campaign-get.js +25 -0
  82. package/dist/tools/campaign-get.js.map +1 -0
  83. package/dist/tools/campaign-get.test.d.ts +2 -0
  84. package/dist/tools/campaign-get.test.d.ts.map +1 -0
  85. package/dist/tools/campaign-get.test.js +90 -0
  86. package/dist/tools/campaign-get.test.js.map +1 -0
  87. package/dist/tools/campaign-list.d.ts +4 -0
  88. package/dist/tools/campaign-list.d.ts.map +1 -0
  89. package/dist/tools/campaign-list.js +25 -0
  90. package/dist/tools/campaign-list.js.map +1 -0
  91. package/dist/tools/campaign-list.test.d.ts +2 -0
  92. package/dist/tools/campaign-list.test.d.ts.map +1 -0
  93. package/dist/tools/campaign-list.test.js +90 -0
  94. package/dist/tools/campaign-list.test.js.map +1 -0
  95. package/dist/tools/campaign-move-next.d.ts +4 -0
  96. package/dist/tools/campaign-move-next.d.ts.map +1 -0
  97. package/dist/tools/campaign-move-next.js +40 -0
  98. package/dist/tools/campaign-move-next.js.map +1 -0
  99. package/dist/tools/campaign-move-next.test.d.ts +2 -0
  100. package/dist/tools/campaign-move-next.test.d.ts.map +1 -0
  101. package/dist/tools/campaign-move-next.test.js +138 -0
  102. package/dist/tools/campaign-move-next.test.js.map +1 -0
  103. package/dist/tools/campaign-remove-action.d.ts +4 -0
  104. package/dist/tools/campaign-remove-action.d.ts.map +1 -0
  105. package/dist/tools/campaign-remove-action.js +36 -0
  106. package/dist/tools/campaign-remove-action.js.map +1 -0
  107. package/dist/tools/campaign-remove-action.test.d.ts +2 -0
  108. package/dist/tools/campaign-remove-action.test.d.ts.map +1 -0
  109. package/dist/tools/campaign-remove-action.test.js +130 -0
  110. package/dist/tools/campaign-remove-action.test.js.map +1 -0
  111. package/dist/tools/campaign-reorder-actions.d.ts +4 -0
  112. package/dist/tools/campaign-reorder-actions.d.ts.map +1 -0
  113. package/dist/tools/campaign-reorder-actions.js +35 -0
  114. package/dist/tools/campaign-reorder-actions.js.map +1 -0
  115. package/dist/tools/campaign-reorder-actions.test.d.ts +2 -0
  116. package/dist/tools/campaign-reorder-actions.test.d.ts.map +1 -0
  117. package/dist/tools/campaign-reorder-actions.test.js +146 -0
  118. package/dist/tools/campaign-reorder-actions.test.js.map +1 -0
  119. package/dist/tools/campaign-retry.d.ts +4 -0
  120. package/dist/tools/campaign-retry.d.ts.map +1 -0
  121. package/dist/tools/campaign-retry.js +29 -0
  122. package/dist/tools/campaign-retry.js.map +1 -0
  123. package/dist/tools/campaign-retry.test.d.ts +2 -0
  124. package/dist/tools/campaign-retry.test.d.ts.map +1 -0
  125. package/dist/tools/campaign-retry.test.js +91 -0
  126. package/dist/tools/campaign-retry.test.js.map +1 -0
  127. package/dist/tools/campaign-start.d.ts +4 -0
  128. package/dist/tools/campaign-start.d.ts.map +1 -0
  129. package/dist/tools/campaign-start.js +35 -0
  130. package/dist/tools/campaign-start.js.map +1 -0
  131. package/dist/tools/campaign-start.test.d.ts +2 -0
  132. package/dist/tools/campaign-start.test.d.ts.map +1 -0
  133. package/dist/tools/campaign-start.test.js +115 -0
  134. package/dist/tools/campaign-start.test.js.map +1 -0
  135. package/dist/tools/campaign-statistics.d.ts +4 -0
  136. package/dist/tools/campaign-statistics.d.ts.map +1 -0
  137. package/dist/tools/campaign-statistics.js +41 -0
  138. package/dist/tools/campaign-statistics.js.map +1 -0
  139. package/dist/tools/campaign-statistics.test.d.ts +2 -0
  140. package/dist/tools/campaign-statistics.test.d.ts.map +1 -0
  141. package/dist/tools/campaign-statistics.test.js +177 -0
  142. package/dist/tools/campaign-statistics.test.js.map +1 -0
  143. package/dist/tools/campaign-status.d.ts +4 -0
  144. package/dist/tools/campaign-status.d.ts.map +1 -0
  145. package/dist/tools/campaign-status.js +47 -0
  146. package/dist/tools/campaign-status.js.map +1 -0
  147. package/dist/tools/campaign-status.test.d.ts +2 -0
  148. package/dist/tools/campaign-status.test.d.ts.map +1 -0
  149. package/dist/tools/campaign-status.test.js +275 -0
  150. package/dist/tools/campaign-status.test.js.map +1 -0
  151. package/dist/tools/campaign-stop.d.ts +4 -0
  152. package/dist/tools/campaign-stop.d.ts.map +1 -0
  153. package/dist/tools/campaign-stop.js +28 -0
  154. package/dist/tools/campaign-stop.js.map +1 -0
  155. package/dist/tools/campaign-stop.test.d.ts +2 -0
  156. package/dist/tools/campaign-stop.test.d.ts.map +1 -0
  157. package/dist/tools/campaign-stop.test.js +91 -0
  158. package/dist/tools/campaign-stop.test.js.map +1 -0
  159. package/dist/tools/campaign-update.d.ts +4 -0
  160. package/dist/tools/campaign-update.d.ts.map +1 -0
  161. package/dist/tools/campaign-update.js +43 -0
  162. package/dist/tools/campaign-update.js.map +1 -0
  163. package/dist/tools/campaign-update.test.d.ts +2 -0
  164. package/dist/tools/campaign-update.test.d.ts.map +1 -0
  165. package/dist/tools/campaign-update.test.js +117 -0
  166. package/dist/tools/campaign-update.test.js.map +1 -0
  167. package/dist/tools/check-replies.d.ts +4 -0
  168. package/dist/tools/check-replies.d.ts.map +1 -0
  169. package/dist/tools/check-replies.js +24 -0
  170. package/dist/tools/check-replies.js.map +1 -0
  171. package/dist/tools/check-replies.test.d.ts +2 -0
  172. package/dist/tools/check-replies.test.d.ts.map +1 -0
  173. package/dist/tools/check-replies.test.js +192 -0
  174. package/dist/tools/check-replies.test.js.map +1 -0
  175. package/dist/tools/check-status.d.ts +4 -0
  176. package/dist/tools/check-status.d.ts.map +1 -0
  177. package/dist/tools/check-status.js +19 -0
  178. package/dist/tools/check-status.js.map +1 -0
  179. package/dist/tools/check-status.test.d.ts +2 -0
  180. package/dist/tools/check-status.test.d.ts.map +1 -0
  181. package/dist/tools/check-status.test.js +88 -0
  182. package/dist/tools/check-status.test.js.map +1 -0
  183. package/dist/tools/describe-actions.d.ts +4 -0
  184. package/dist/tools/describe-actions.d.ts.map +1 -0
  185. package/dist/tools/describe-actions.js +30 -0
  186. package/dist/tools/describe-actions.js.map +1 -0
  187. package/dist/tools/describe-actions.test.d.ts +2 -0
  188. package/dist/tools/describe-actions.test.d.ts.map +1 -0
  189. package/dist/tools/describe-actions.test.js +120 -0
  190. package/dist/tools/describe-actions.test.js.map +1 -0
  191. package/dist/tools/find-app.d.ts +4 -0
  192. package/dist/tools/find-app.d.ts.map +1 -0
  193. package/dist/tools/find-app.js +20 -0
  194. package/dist/tools/find-app.js.map +1 -0
  195. package/dist/tools/find-app.test.d.ts +2 -0
  196. package/dist/tools/find-app.test.d.ts.map +1 -0
  197. package/dist/tools/find-app.test.js +65 -0
  198. package/dist/tools/find-app.test.js.map +1 -0
  199. package/dist/tools/import-people-from-urls.d.ts +4 -0
  200. package/dist/tools/import-people-from-urls.d.ts.map +1 -0
  201. package/dist/tools/import-people-from-urls.js +32 -0
  202. package/dist/tools/import-people-from-urls.js.map +1 -0
  203. package/dist/tools/import-people-from-urls.test.d.ts +2 -0
  204. package/dist/tools/import-people-from-urls.test.d.ts.map +1 -0
  205. package/dist/tools/import-people-from-urls.test.js +136 -0
  206. package/dist/tools/import-people-from-urls.test.js.map +1 -0
  207. package/dist/tools/index.d.ts +36 -0
  208. package/dist/tools/index.d.ts.map +1 -0
  209. package/dist/tools/index.js +70 -0
  210. package/dist/tools/index.js.map +1 -0
  211. package/dist/tools/index.test.d.ts +2 -0
  212. package/dist/tools/index.test.d.ts.map +1 -0
  213. package/dist/tools/index.test.js +22 -0
  214. package/dist/tools/index.test.js.map +1 -0
  215. package/dist/tools/launch-app.d.ts +4 -0
  216. package/dist/tools/launch-app.d.ts.map +1 -0
  217. package/dist/tools/launch-app.js +30 -0
  218. package/dist/tools/launch-app.js.map +1 -0
  219. package/dist/tools/launch-app.test.d.ts +2 -0
  220. package/dist/tools/launch-app.test.d.ts.map +1 -0
  221. package/dist/tools/launch-app.test.js +104 -0
  222. package/dist/tools/launch-app.test.js.map +1 -0
  223. package/dist/tools/list-accounts.d.ts +4 -0
  224. package/dist/tools/list-accounts.d.ts.map +1 -0
  225. package/dist/tools/list-accounts.js +29 -0
  226. package/dist/tools/list-accounts.js.map +1 -0
  227. package/dist/tools/list-accounts.test.d.ts +2 -0
  228. package/dist/tools/list-accounts.test.d.ts.map +1 -0
  229. package/dist/tools/list-accounts.test.js +101 -0
  230. package/dist/tools/list-accounts.test.js.map +1 -0
  231. package/dist/tools/query-messages.d.ts +4 -0
  232. package/dist/tools/query-messages.d.ts.map +1 -0
  233. package/dist/tools/query-messages.integration.test.d.ts +2 -0
  234. package/dist/tools/query-messages.integration.test.d.ts.map +1 -0
  235. package/dist/tools/query-messages.integration.test.js +119 -0
  236. package/dist/tools/query-messages.integration.test.js.map +1 -0
  237. package/dist/tools/query-messages.js +55 -0
  238. package/dist/tools/query-messages.js.map +1 -0
  239. package/dist/tools/query-messages.test.d.ts +2 -0
  240. package/dist/tools/query-messages.test.d.ts.map +1 -0
  241. package/dist/tools/query-messages.test.js +211 -0
  242. package/dist/tools/query-messages.test.js.map +1 -0
  243. package/dist/tools/query-profile.d.ts +4 -0
  244. package/dist/tools/query-profile.d.ts.map +1 -0
  245. package/dist/tools/query-profile.js +49 -0
  246. package/dist/tools/query-profile.js.map +1 -0
  247. package/dist/tools/query-profile.test.d.ts +2 -0
  248. package/dist/tools/query-profile.test.d.ts.map +1 -0
  249. package/dist/tools/query-profile.test.js +275 -0
  250. package/dist/tools/query-profile.test.js.map +1 -0
  251. package/dist/tools/query-profiles.d.ts +4 -0
  252. package/dist/tools/query-profiles.d.ts.map +1 -0
  253. package/dist/tools/query-profiles.js +67 -0
  254. package/dist/tools/query-profiles.js.map +1 -0
  255. package/dist/tools/query-profiles.test.d.ts +2 -0
  256. package/dist/tools/query-profiles.test.d.ts.map +1 -0
  257. package/dist/tools/query-profiles.test.js +239 -0
  258. package/dist/tools/query-profiles.test.js.map +1 -0
  259. package/dist/tools/quit-app.d.ts +4 -0
  260. package/dist/tools/quit-app.d.ts.map +1 -0
  261. package/dist/tools/quit-app.js +27 -0
  262. package/dist/tools/quit-app.js.map +1 -0
  263. package/dist/tools/quit-app.test.d.ts +2 -0
  264. package/dist/tools/quit-app.test.d.ts.map +1 -0
  265. package/dist/tools/quit-app.test.js +67 -0
  266. package/dist/tools/quit-app.test.js.map +1 -0
  267. package/dist/tools/scrape-messaging-history.d.ts +4 -0
  268. package/dist/tools/scrape-messaging-history.d.ts.map +1 -0
  269. package/dist/tools/scrape-messaging-history.js +19 -0
  270. package/dist/tools/scrape-messaging-history.js.map +1 -0
  271. package/dist/tools/scrape-messaging-history.test.d.ts +2 -0
  272. package/dist/tools/scrape-messaging-history.test.d.ts.map +1 -0
  273. package/dist/tools/scrape-messaging-history.test.js +114 -0
  274. package/dist/tools/scrape-messaging-history.test.js.map +1 -0
  275. package/dist/tools/start-instance.d.ts +4 -0
  276. package/dist/tools/start-instance.d.ts.map +1 -0
  277. package/dist/tools/start-instance.js +53 -0
  278. package/dist/tools/start-instance.js.map +1 -0
  279. package/dist/tools/start-instance.test.d.ts +2 -0
  280. package/dist/tools/start-instance.test.d.ts.map +1 -0
  281. package/dist/tools/start-instance.test.js +234 -0
  282. package/dist/tools/start-instance.test.js.map +1 -0
  283. package/dist/tools/stop-instance.d.ts +4 -0
  284. package/dist/tools/stop-instance.d.ts.map +1 -0
  285. package/dist/tools/stop-instance.js +47 -0
  286. package/dist/tools/stop-instance.js.map +1 -0
  287. package/dist/tools/stop-instance.test.d.ts +2 -0
  288. package/dist/tools/stop-instance.test.d.ts.map +1 -0
  289. package/dist/tools/stop-instance.test.js +179 -0
  290. package/dist/tools/stop-instance.test.js.map +1 -0
  291. package/dist/tools/testing/infrastructure-errors.d.ts +18 -0
  292. package/dist/tools/testing/infrastructure-errors.d.ts.map +1 -0
  293. package/dist/tools/testing/infrastructure-errors.js +59 -0
  294. package/dist/tools/testing/infrastructure-errors.js.map +1 -0
  295. package/dist/tools/testing/mock-server.d.ts +10 -0
  296. package/dist/tools/testing/mock-server.d.ts.map +1 -0
  297. package/dist/tools/testing/mock-server.js +33 -0
  298. package/dist/tools/testing/mock-server.js.map +1 -0
  299. package/package.json +38 -14
package/README.md CHANGED
@@ -2,12 +2,70 @@
2
2
 
3
3
  MCP server for [lhremote](https://github.com/alexey-pelykh/lhremote) — LinkedHelper automation toolkit.
4
4
 
5
+ This package exposes the full LinkedHelper automation surface as a [Model Context Protocol](https://modelcontextprotocol.io) server. AI assistants (Claude, etc.) connect over stdio and use the 32 registered tools to control LinkedHelper.
6
+
7
+ Built on [`@lhremote/core`](../core).
8
+
5
9
  ## Installation
6
10
 
7
11
  ```bash
8
12
  npm install @lhremote/mcp
9
13
  ```
10
14
 
15
+ ## Usage with Claude Desktop
16
+
17
+ Add to `claude_desktop_config.json`:
18
+
19
+ ```json
20
+ {
21
+ "mcpServers": {
22
+ "lhremote": {
23
+ "command": "npx",
24
+ "args": ["lhremote", "mcp"]
25
+ }
26
+ }
27
+ }
28
+ ```
29
+
30
+ ## Programmatic Usage
31
+
32
+ ```typescript
33
+ import { createServer } from "@lhremote/mcp";
34
+
35
+ const server = createServer();
36
+ // server is a fully configured McpServer with all tools registered
37
+ ```
38
+
39
+ Or start the stdio transport directly:
40
+
41
+ ```typescript
42
+ import { runStdioServer } from "@lhremote/mcp/stdio";
43
+
44
+ await runStdioServer();
45
+ ```
46
+
47
+ ## Registered Tools
48
+
49
+ | Category | Tools |
50
+ |----------|-------|
51
+ | App Management | `find-app`, `launch-app`, `quit-app` |
52
+ | Account & Instance | `list-accounts`, `start-instance`, `stop-instance`, `check-status` |
53
+ | Campaigns | `campaign-list`, `campaign-create`, `campaign-get`, `campaign-export`, `campaign-update`, `campaign-delete`, `campaign-start`, `campaign-stop` |
54
+ | Campaign Status | `campaign-status`, `campaign-statistics`, `campaign-retry` |
55
+ | Campaign Actions | `campaign-add-action`, `campaign-remove-action`, `campaign-reorder-actions`, `campaign-move-next` |
56
+ | Campaign Targeting | `campaign-exclude-list`, `campaign-exclude-add`, `campaign-exclude-remove`, `import-people-from-urls` |
57
+ | Profiles & Messaging | `query-profile`, `query-profiles`, `query-messages`, `check-replies`, `scrape-messaging-history` |
58
+ | Utilities | `describe-actions` |
59
+
60
+ See the [root README](https://github.com/alexey-pelykh/lhremote#mcp-tools) for parameter details on each tool.
61
+
62
+ ## Exports
63
+
64
+ | Export | Description |
65
+ |--------|-------------|
66
+ | `createServer()` | Create a configured `McpServer` with all tools registered |
67
+ | `runStdioServer()` | Start the MCP server on stdio (from `@lhremote/mcp/stdio`) |
68
+
11
69
  ## License
12
70
 
13
71
  [AGPL-3.0-only](https://github.com/alexey-pelykh/lhremote/blob/main/LICENSE)
@@ -0,0 +1,59 @@
1
+ import { z } from "zod";
2
+ type TextContent = {
3
+ type: "text";
4
+ text: string;
5
+ };
6
+ type McpResult = {
7
+ isError?: boolean;
8
+ content: TextContent[];
9
+ };
10
+ /**
11
+ * Shared Zod schema fields for CDP connection parameters.
12
+ *
13
+ * Spread into every tool that connects to a LinkedHelper instance:
14
+ * ```ts
15
+ * { campaignId: z.number(), ...cdpConnectionSchema }
16
+ * ```
17
+ */
18
+ export declare const cdpConnectionSchema: {
19
+ cdpPort: z.ZodDefault<z.ZodOptional<z.ZodNumber>>;
20
+ cdpHost: z.ZodOptional<z.ZodString>;
21
+ allowRemote: z.ZodOptional<z.ZodBoolean>;
22
+ };
23
+ /**
24
+ * Build the CDP connection options object from parsed tool arguments.
25
+ *
26
+ * Replaces the inline conditional-spread pattern used by tools that
27
+ * call `resolveAccount` or construct a `LauncherService`.
28
+ */
29
+ export declare function buildCdpOptions(args: {
30
+ cdpHost?: string | undefined;
31
+ allowRemote?: boolean | undefined;
32
+ }): {
33
+ host?: string;
34
+ allowRemote?: boolean;
35
+ };
36
+ /**
37
+ * Build an MCP error response from a plain message string.
38
+ */
39
+ export declare function mcpError(text: string): McpResult;
40
+ /**
41
+ * Build an MCP success response from a plain text or JSON payload.
42
+ */
43
+ export declare function mcpSuccess(text: string): McpResult;
44
+ /**
45
+ * Map common infrastructure errors (launcher not running, account
46
+ * resolution failures) and domain errors (campaign not found) to an
47
+ * MCP error response.
48
+ *
49
+ * Returns `undefined` if the error is not a recognised error so the
50
+ * caller can fall through to domain-specific handling.
51
+ */
52
+ export declare function mapErrorToMcpResponse(error: unknown): McpResult | undefined;
53
+ /**
54
+ * Map an arbitrary caught error to an MCP error response with a
55
+ * contextual prefix (e.g. "Failed to create campaign").
56
+ */
57
+ export declare function mcpCatchAll(error: unknown, prefix: string): McpResult;
58
+ export {};
59
+ //# sourceMappingURL=helpers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"helpers.d.ts","sourceRoot":"","sources":["../src/helpers.ts"],"names":[],"mappings":"AAUA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,KAAK,WAAW,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,CAAC;AAClD,KAAK,SAAS,GAAG;IAAE,OAAO,CAAC,EAAE,OAAO,CAAC;IAAC,OAAO,EAAE,WAAW,EAAE,CAAA;CAAE,CAAC;AAE/D;;;;;;;GAOG;AACH,eAAO,MAAM,mBAAmB;;;;CAgB/B,CAAC;AAEF;;;;;GAKG;AACH,wBAAgB,eAAe,CAAC,IAAI,EAAE;IACpC,OAAO,CAAC,EAAE,MAAM,GAAG,SAAS,CAAC;IAC7B,WAAW,CAAC,EAAE,OAAO,GAAG,SAAS,CAAC;CACnC,GAAG;IAAE,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,WAAW,CAAC,EAAE,OAAO,CAAA;CAAE,CAK3C;AAED;;GAEG;AACH,wBAAgB,QAAQ,CAAC,IAAI,EAAE,MAAM,GAAG,SAAS,CAKhD;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,SAAS,CAIlD;AAED;;;;;;;GAOG;AACH,wBAAgB,qBAAqB,CAAC,KAAK,EAAE,OAAO,GAAG,SAAS,GAAG,SAAS,CAa3E;AAED;;;GAGG;AACH,wBAAgB,WAAW,CAAC,KAAK,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,GAAG,SAAS,CAMrE"}
@@ -0,0 +1,90 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { AccountResolutionError, CampaignNotFoundError, DEFAULT_CDP_PORT, errorMessage, LinkedHelperNotRunningError, } from "@lhremote/core";
4
+ import { z } from "zod";
5
+ /**
6
+ * Shared Zod schema fields for CDP connection parameters.
7
+ *
8
+ * Spread into every tool that connects to a LinkedHelper instance:
9
+ * ```ts
10
+ * { campaignId: z.number(), ...cdpConnectionSchema }
11
+ * ```
12
+ */
13
+ export const cdpConnectionSchema = {
14
+ cdpPort: z
15
+ .number()
16
+ .int()
17
+ .positive()
18
+ .optional()
19
+ .default(DEFAULT_CDP_PORT)
20
+ .describe("CDP port"),
21
+ cdpHost: z
22
+ .string()
23
+ .optional()
24
+ .describe("CDP host (default: 127.0.0.1)"),
25
+ allowRemote: z
26
+ .boolean()
27
+ .optional()
28
+ .describe("SECURITY: Allow non-loopback CDP connections. Enables remote code execution on target host. Only use if network path is secured."),
29
+ };
30
+ /**
31
+ * Build the CDP connection options object from parsed tool arguments.
32
+ *
33
+ * Replaces the inline conditional-spread pattern used by tools that
34
+ * call `resolveAccount` or construct a `LauncherService`.
35
+ */
36
+ export function buildCdpOptions(args) {
37
+ return {
38
+ ...(args.cdpHost !== undefined && { host: args.cdpHost }),
39
+ ...(args.allowRemote !== undefined && { allowRemote: args.allowRemote }),
40
+ };
41
+ }
42
+ /**
43
+ * Build an MCP error response from a plain message string.
44
+ */
45
+ export function mcpError(text) {
46
+ return {
47
+ isError: true,
48
+ content: [{ type: "text", text }],
49
+ };
50
+ }
51
+ /**
52
+ * Build an MCP success response from a plain text or JSON payload.
53
+ */
54
+ export function mcpSuccess(text) {
55
+ return {
56
+ content: [{ type: "text", text }],
57
+ };
58
+ }
59
+ /**
60
+ * Map common infrastructure errors (launcher not running, account
61
+ * resolution failures) and domain errors (campaign not found) to an
62
+ * MCP error response.
63
+ *
64
+ * Returns `undefined` if the error is not a recognised error so the
65
+ * caller can fall through to domain-specific handling.
66
+ */
67
+ export function mapErrorToMcpResponse(error) {
68
+ if (error instanceof LinkedHelperNotRunningError) {
69
+ return mcpError("LinkedHelper is not running. Use launch-app first.");
70
+ }
71
+ if (error instanceof AccountResolutionError) {
72
+ return mcpError(error.message);
73
+ }
74
+ if (error instanceof CampaignNotFoundError) {
75
+ return mcpError(`Campaign ${String(error.campaignId)} not found.`);
76
+ }
77
+ return undefined;
78
+ }
79
+ /**
80
+ * Map an arbitrary caught error to an MCP error response with a
81
+ * contextual prefix (e.g. "Failed to create campaign").
82
+ */
83
+ export function mcpCatchAll(error, prefix) {
84
+ const mapped = mapErrorToMcpResponse(error);
85
+ if (mapped)
86
+ return mapped;
87
+ const message = errorMessage(error);
88
+ return mcpError(`${prefix}: ${message}`);
89
+ }
90
+ //# sourceMappingURL=helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../src/helpers.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,sBAAsB,EACtB,qBAAqB,EACrB,gBAAgB,EAChB,YAAY,EACZ,2BAA2B,GAC5B,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAKxB;;;;;;;GAOG;AACH,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,OAAO,EAAE,CAAC;SACP,MAAM,EAAE;SACR,GAAG,EAAE;SACL,QAAQ,EAAE;SACV,QAAQ,EAAE;SACV,OAAO,CAAC,gBAAgB,CAAC;SACzB,QAAQ,CAAC,UAAU,CAAC;IACvB,OAAO,EAAE,CAAC;SACP,MAAM,EAAE;SACR,QAAQ,EAAE;SACV,QAAQ,CAAC,+BAA+B,CAAC;IAC5C,WAAW,EAAE,CAAC;SACX,OAAO,EAAE;SACT,QAAQ,EAAE;SACV,QAAQ,CAAC,kIAAkI,CAAC;CAChJ,CAAC;AAEF;;;;;GAKG;AACH,MAAM,UAAU,eAAe,CAAC,IAG/B;IACC,OAAO;QACL,GAAG,CAAC,IAAI,CAAC,OAAO,KAAK,SAAS,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,OAAO,EAAE,CAAC;QACzD,GAAG,CAAC,IAAI,CAAC,WAAW,KAAK,SAAS,IAAI,EAAE,WAAW,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC;KACzE,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,QAAQ,CAAC,IAAY;IACnC,OAAO;QACL,OAAO,EAAE,IAAI;QACb,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,CAAC;KAC3C,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,UAAU,CAAC,IAAY;IACrC,OAAO;QACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,CAAC;KAC3C,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,qBAAqB,CAAC,KAAc;IAClD,IAAI,KAAK,YAAY,2BAA2B,EAAE,CAAC;QACjD,OAAO,QAAQ,CAAC,oDAAoD,CAAC,CAAC;IACxE,CAAC;IACD,IAAI,KAAK,YAAY,sBAAsB,EAAE,CAAC;QAC5C,OAAO,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IACD,IAAI,KAAK,YAAY,qBAAqB,EAAE,CAAC;QAC3C,OAAO,QAAQ,CACb,YAAY,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,aAAa,CAClD,CAAC;IACJ,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,WAAW,CAAC,KAAc,EAAE,MAAc;IACxD,MAAM,MAAM,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;IAC5C,IAAI,MAAM;QAAE,OAAO,MAAM,CAAC;IAE1B,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;IACpC,OAAO,QAAQ,CAAC,GAAG,MAAM,KAAK,OAAO,EAAE,CAAC,CAAC;AAC3C,CAAC"}
package/dist/index.d.ts CHANGED
@@ -1,2 +1,3 @@
1
+ #!/usr/bin/env node
1
2
  export {};
2
3
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
package/dist/index.js CHANGED
@@ -1,2 +1,6 @@
1
- export {};
1
+ #!/usr/bin/env node
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+ // Copyright (C) 2026 Oleksii PELYKH
4
+ import { runStdioServer } from "./stdio.js";
5
+ await runStdioServer();
2
6
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAE5C,MAAM,cAAc,EAAE,CAAC"}
@@ -0,0 +1,6 @@
1
+ import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
2
+ /**
3
+ * Create a configured MCP server instance with all tools registered.
4
+ */
5
+ export declare function createServer(): McpServer;
6
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAOpE;;GAEG;AACH,wBAAgB,YAAY,IAAI,SAAS,CAIxC"}
package/dist/server.js ADDED
@@ -0,0 +1,16 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { createRequire } from "node:module";
4
+ import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
5
+ import { registerAllTools } from "./tools/index.js";
6
+ const require = createRequire(import.meta.url);
7
+ const { name, version } = require("../package.json");
8
+ /**
9
+ * Create a configured MCP server instance with all tools registered.
10
+ */
11
+ export function createServer() {
12
+ const server = new McpServer({ name, version });
13
+ registerAllTools(server);
14
+ return server;
15
+ }
16
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAEpE,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAEpD,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAC/C,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,iBAAiB,CAAsC,CAAC;AAE1F;;GAEG;AACH,MAAM,UAAU,YAAY;IAC1B,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;IAChD,gBAAgB,CAAC,MAAM,CAAC,CAAC;IACzB,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=server.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.d.ts","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,85 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { createRequire } from "node:module";
4
+ import { afterEach, describe, expect, it } from "vitest";
5
+ import { createServer } from "./server.js";
6
+ const require = createRequire(import.meta.url);
7
+ const { version } = require("../package.json");
8
+ let server;
9
+ let client;
10
+ afterEach(async () => {
11
+ await client?.close();
12
+ await server?.close();
13
+ client = undefined;
14
+ server = undefined;
15
+ });
16
+ async function connectPair() {
17
+ const { Client: ClientCtor } = await import("@modelcontextprotocol/sdk/client/index.js");
18
+ const { InMemoryTransport } = await import("@modelcontextprotocol/sdk/inMemory.js");
19
+ server = createServer();
20
+ const [clientTransport, serverTransport] = InMemoryTransport.createLinkedPair();
21
+ client = new ClientCtor({ name: "test-client", version: "1.0.0" });
22
+ await Promise.all([
23
+ server.connect(serverTransport),
24
+ client.connect(clientTransport),
25
+ ]);
26
+ return { server, client };
27
+ }
28
+ describe("createServer", () => {
29
+ it("returns an McpServer instance", () => {
30
+ server = createServer();
31
+ expect(server).toBeDefined();
32
+ expect(server).toHaveProperty("connect");
33
+ expect(server).toHaveProperty("close");
34
+ expect(server).toHaveProperty("tool");
35
+ });
36
+ it("can connect and close with an in-memory transport", async () => {
37
+ const { client: c } = await connectPair();
38
+ const info = c.getServerVersion();
39
+ expect(info).toEqual(expect.objectContaining({ name: "@lhremote/mcp", version }));
40
+ });
41
+ it("advertises tools capability", async () => {
42
+ const { client: c } = await connectPair();
43
+ const capabilities = c.getServerCapabilities();
44
+ expect(capabilities?.tools).toBeDefined();
45
+ });
46
+ it("lists registered tools", async () => {
47
+ const { client: c } = await connectPair();
48
+ const { tools } = await c.listTools();
49
+ const names = tools.map((t) => t.name);
50
+ expect(names).toContain("find-app");
51
+ expect(names).toContain("launch-app");
52
+ expect(names).toContain("quit-app");
53
+ expect(names).toContain("list-accounts");
54
+ expect(names).toContain("start-instance");
55
+ expect(names).toContain("stop-instance");
56
+ expect(names).toContain("query-profile");
57
+ expect(names).toContain("query-profiles");
58
+ expect(names).toContain("query-messages");
59
+ expect(names).toContain("scrape-messaging-history");
60
+ expect(names).toContain("campaign-create");
61
+ expect(names).toContain("campaign-delete");
62
+ expect(names).toContain("campaign-export");
63
+ expect(names).toContain("campaign-get");
64
+ expect(names).toContain("campaign-list");
65
+ expect(names).toContain("campaign-retry");
66
+ expect(names).toContain("campaign-start");
67
+ expect(names).toContain("campaign-statistics");
68
+ expect(names).toContain("campaign-status");
69
+ expect(names).toContain("campaign-stop");
70
+ expect(names).toContain("campaign-update");
71
+ expect(names).toContain("check-replies");
72
+ expect(names).toContain("check-status");
73
+ expect(names).toContain("describe-actions");
74
+ expect(names).toContain("import-people-from-urls");
75
+ expect(names).toContain("campaign-move-next");
76
+ expect(names).toContain("campaign-add-action");
77
+ expect(names).toContain("campaign-remove-action");
78
+ expect(names).toContain("campaign-reorder-actions");
79
+ expect(names).toContain("campaign-exclude-list");
80
+ expect(names).toContain("campaign-exclude-add");
81
+ expect(names).toContain("campaign-exclude-remove");
82
+ expect(names).toHaveLength(32);
83
+ });
84
+ });
85
+ //# sourceMappingURL=server.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.js","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAI5C,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAEzD,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAC/C,MAAM,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,iBAAiB,CAAwB,CAAC;AAEtE,IAAI,MAA6B,CAAC;AAClC,IAAI,MAA0B,CAAC;AAE/B,SAAS,CAAC,KAAK,IAAI,EAAE;IACnB,MAAM,MAAM,EAAE,KAAK,EAAE,CAAC;IACtB,MAAM,MAAM,EAAE,KAAK,EAAE,CAAC;IACtB,MAAM,GAAG,SAAS,CAAC;IACnB,MAAM,GAAG,SAAS,CAAC;AACrB,CAAC,CAAC,CAAC;AAEH,KAAK,UAAU,WAAW;IACxB,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,MAAM,CACzC,2CAA2C,CAC5C,CAAC;IACF,MAAM,EAAE,iBAAiB,EAAE,GAAG,MAAM,MAAM,CACxC,uCAAuC,CACxC,CAAC;IAEF,MAAM,GAAG,YAAY,EAAE,CAAC;IACxB,MAAM,CAAC,eAAe,EAAE,eAAe,CAAC,GACtC,iBAAiB,CAAC,gBAAgB,EAAE,CAAC;IACvC,MAAM,GAAG,IAAI,UAAU,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,CAAC;IAEnE,MAAM,OAAO,CAAC,GAAG,CAAC;QAChB,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC;QAC/B,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC;KAChC,CAAC,CAAC;IAEH,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC;AAC5B,CAAC;AAED,QAAQ,CAAC,cAAc,EAAE,GAAG,EAAE;IAC5B,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;QACvC,MAAM,GAAG,YAAY,EAAE,CAAC;QAExB,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAC;QAC7B,MAAM,CAAC,MAAM,CAAC,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;QACzC,MAAM,CAAC,MAAM,CAAC,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;QACvC,MAAM,CAAC,MAAM,CAAC,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;QACjE,MAAM,EAAE,MAAM,EAAE,CAAC,EAAE,GAAG,MAAM,WAAW,EAAE,CAAC;QAE1C,MAAM,IAAI,GAAG,CAAC,CAAC,gBAAgB,EAAE,CAAC;QAClC,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAClB,MAAM,CAAC,gBAAgB,CAAC,EAAE,IAAI,EAAE,eAAe,EAAE,OAAO,EAAE,CAAC,CAC5D,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE,KAAK,IAAI,EAAE;QAC3C,MAAM,EAAE,MAAM,EAAE,CAAC,EAAE,GAAG,MAAM,WAAW,EAAE,CAAC;QAE1C,MAAM,YAAY,GAAG,CAAC,CAAC,qBAAqB,EAAE,CAAC;QAC/C,MAAM,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC,WAAW,EAAE,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,KAAK,IAAI,EAAE;QACtC,MAAM,EAAE,MAAM,EAAE,CAAC,EAAE,GAAG,MAAM,WAAW,EAAE,CAAC;QAE1C,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC,CAAC,SAAS,EAAE,CAAC;QACtC,MAAM,KAAK,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAEvC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QACpC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;QACtC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QACpC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,0BAA0B,CAAC,CAAC;QACpD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QAC/C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAC5C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,yBAAyB,CAAC,CAAC;QACnD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;QAC9C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QAC/C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;QAClD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,0BAA0B,CAAC,CAAC;QACpD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;QACjD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,sBAAsB,CAAC,CAAC;QAChD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,yBAAyB,CAAC,CAAC;QACnD,MAAM,CAAC,KAAK,CAAC,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Start the MCP server on stdio and register signal handlers for
3
+ * graceful shutdown. This function does not return under normal
4
+ * operation — the process stays alive until SIGINT/SIGTERM.
5
+ */
6
+ export declare function runStdioServer(): Promise<void>;
7
+ //# sourceMappingURL=stdio.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stdio.d.ts","sourceRoot":"","sources":["../src/stdio.ts"],"names":[],"mappings":"AAQA;;;;GAIG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,IAAI,CAAC,CA4BpD"}
package/dist/stdio.js ADDED
@@ -0,0 +1,37 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js";
4
+ import { errorMessage } from "@lhremote/core";
5
+ import { createServer } from "./server.js";
6
+ /**
7
+ * Start the MCP server on stdio and register signal handlers for
8
+ * graceful shutdown. This function does not return under normal
9
+ * operation — the process stays alive until SIGINT/SIGTERM.
10
+ */
11
+ export async function runStdioServer() {
12
+ const server = createServer();
13
+ const transport = new StdioServerTransport();
14
+ try {
15
+ await server.connect(transport);
16
+ }
17
+ catch (error) {
18
+ const message = errorMessage(error);
19
+ process.stderr.write(`Failed to start MCP server: ${message}\n`);
20
+ process.exit(1);
21
+ }
22
+ process.stderr.write("lhremote MCP server running on stdio\n");
23
+ function shutdown() {
24
+ server
25
+ .close()
26
+ .catch((error) => {
27
+ const message = errorMessage(error);
28
+ process.stderr.write(`Error during shutdown: ${message}\n`);
29
+ })
30
+ .finally(() => {
31
+ process.exit(0);
32
+ });
33
+ }
34
+ process.on("SIGINT", shutdown);
35
+ process.on("SIGTERM", shutdown);
36
+ }
37
+ //# sourceMappingURL=stdio.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stdio.js","sourceRoot":"","sources":["../src/stdio.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAC;AACjF,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAE9C,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc;IAClC,MAAM,MAAM,GAAG,YAAY,EAAE,CAAC;IAC9B,MAAM,SAAS,GAAG,IAAI,oBAAoB,EAAE,CAAC;IAE7C,IAAI,CAAC;QACH,MAAM,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAAC,OAAO,KAAc,EAAE,CAAC;QACxB,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,+BAA+B,OAAO,IAAI,CAAC,CAAC;QACjE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC;IAE/D,SAAS,QAAQ;QACf,MAAM;aACH,KAAK,EAAE;aACP,KAAK,CAAC,CAAC,KAAc,EAAE,EAAE;YACxB,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;YACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,OAAO,IAAI,CAAC,CAAC;QAC9D,CAAC,CAAC;aACD,OAAO,CAAC,GAAG,EAAE;YACZ,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC,CAAC,CAAC;IACP,CAAC;IAED,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC/B,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;AAClC,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
2
+ /** Register the {@link https://github.com/alexey-pelykh/lhremote#campaign-add-action | campaign-add-action} MCP tool. */
3
+ export declare function registerCampaignAddAction(server: McpServer): void;
4
+ //# sourceMappingURL=campaign-add-action.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"campaign-add-action.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-add-action.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAOzE,yHAAyH;AACzH,wBAAgB,yBAAyB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAsEjE"}
@@ -0,0 +1,63 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { campaignAddAction, } from "@lhremote/core";
4
+ import { z } from "zod";
5
+ import { cdpConnectionSchema, mcpCatchAll, mcpError, mcpSuccess } from "../helpers.js";
6
+ /** Register the {@link https://github.com/alexey-pelykh/lhremote#campaign-add-action | campaign-add-action} MCP tool. */
7
+ export function registerCampaignAddAction(server) {
8
+ server.tool("campaign-add-action", "Add a new action to an existing campaign's action chain", {
9
+ campaignId: z
10
+ .number()
11
+ .int()
12
+ .positive()
13
+ .describe("Campaign ID"),
14
+ name: z
15
+ .string()
16
+ .describe("Display name for the action"),
17
+ actionType: z
18
+ .string()
19
+ .describe("Action type identifier (e.g., 'VisitAndExtract', 'MessageToPerson')"),
20
+ description: z
21
+ .string()
22
+ .optional()
23
+ .describe("Optional action description"),
24
+ coolDown: z
25
+ .number()
26
+ .int()
27
+ .optional()
28
+ .describe("Milliseconds between action executions (default: 60000)"),
29
+ maxActionResultsPerIteration: z
30
+ .number()
31
+ .int()
32
+ .optional()
33
+ .describe("Maximum results per iteration (default: 10, -1 for unlimited)"),
34
+ actionSettings: z
35
+ .string()
36
+ .optional()
37
+ .describe("Action-specific settings as a JSON string"),
38
+ ...cdpConnectionSchema,
39
+ }, async ({ campaignId, name, actionType, description, coolDown, maxActionResultsPerIteration, actionSettings, cdpPort, cdpHost, allowRemote, }) => {
40
+ // Parse action settings JSON if provided
41
+ let parsedSettings = {};
42
+ if (actionSettings !== undefined) {
43
+ try {
44
+ parsedSettings = JSON.parse(actionSettings);
45
+ }
46
+ catch {
47
+ return mcpError("Invalid JSON in actionSettings.");
48
+ }
49
+ }
50
+ try {
51
+ const result = await campaignAddAction({
52
+ campaignId, name, actionType, description, coolDown,
53
+ maxActionResultsPerIteration, actionSettings: parsedSettings,
54
+ cdpPort, cdpHost, allowRemote,
55
+ });
56
+ return mcpSuccess(JSON.stringify(result, null, 2));
57
+ }
58
+ catch (error) {
59
+ return mcpCatchAll(error, "Failed to add action to campaign");
60
+ }
61
+ });
62
+ }
63
+ //# sourceMappingURL=campaign-add-action.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"campaign-add-action.js","sourceRoot":"","sources":["../../src/tools/campaign-add-action.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,iBAAiB,GAClB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEvF,yHAAyH;AACzH,MAAM,UAAU,yBAAyB,CAAC,MAAiB;IACzD,MAAM,CAAC,IAAI,CACT,qBAAqB,EACrB,yDAAyD,EACzD;QACE,UAAU,EAAE,CAAC;aACV,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,aAAa,CAAC;QAC1B,IAAI,EAAE,CAAC;aACJ,MAAM,EAAE;aACR,QAAQ,CAAC,6BAA6B,CAAC;QAC1C,UAAU,EAAE,CAAC;aACV,MAAM,EAAE;aACR,QAAQ,CAAC,qEAAqE,CAAC;QAClF,WAAW,EAAE,CAAC;aACX,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,6BAA6B,CAAC;QAC1C,QAAQ,EAAE,CAAC;aACR,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,yDAAyD,CAAC;QACtE,4BAA4B,EAAE,CAAC;aAC5B,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,+DAA+D,CAAC;QAC5E,cAAc,EAAE,CAAC;aACd,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,2CAA2C,CAAC;QACxD,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EACL,UAAU,EACV,IAAI,EACJ,UAAU,EACV,WAAW,EACX,QAAQ,EACR,4BAA4B,EAC5B,cAAc,EACd,OAAO,EACP,OAAO,EACP,WAAW,GACZ,EAAE,EAAE;QACH,yCAAyC;QACzC,IAAI,cAAc,GAA4B,EAAE,CAAC;QACjD,IAAI,cAAc,KAAK,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC;gBACH,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAA4B,CAAC;YACzE,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,QAAQ,CAAC,iCAAiC,CAAC,CAAC;YACrD,CAAC;QACH,CAAC;QAED,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,iBAAiB,CAAC;gBACrC,UAAU,EAAE,IAAI,EAAE,UAAU,EAAE,WAAW,EAAE,QAAQ;gBACnD,4BAA4B,EAAE,cAAc,EAAE,cAAc;gBAC5D,OAAO,EAAE,OAAO,EAAE,WAAW;aAC9B,CAAC,CAAC;YACH,OAAO,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,WAAW,CAAC,KAAK,EAAE,kCAAkC,CAAC,CAAC;QAChE,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=campaign-add-action.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"campaign-add-action.test.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-add-action.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,107 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
4
+ vi.mock("@lhremote/core", async (importOriginal) => {
5
+ const actual = await importOriginal();
6
+ return {
7
+ ...actual,
8
+ campaignAddAction: vi.fn(),
9
+ };
10
+ });
11
+ import { CampaignNotFoundError, campaignAddAction, } from "@lhremote/core";
12
+ import { registerCampaignAddAction } from "./campaign-add-action.js";
13
+ import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
14
+ import { createMockServer } from "./testing/mock-server.js";
15
+ const MOCK_ACTION = {
16
+ id: 50,
17
+ campaignId: 15,
18
+ name: "Visit & Extract",
19
+ description: null,
20
+ config: {
21
+ id: 500,
22
+ actionType: "VisitAndExtract",
23
+ actionSettings: {},
24
+ coolDown: 60000,
25
+ maxActionResultsPerIteration: 10,
26
+ isDraft: false,
27
+ },
28
+ versionId: 5000,
29
+ };
30
+ describe("registerCampaignAddAction", () => {
31
+ beforeEach(() => {
32
+ vi.clearAllMocks();
33
+ });
34
+ afterEach(() => {
35
+ vi.restoreAllMocks();
36
+ });
37
+ it("registers a tool named campaign-add-action", () => {
38
+ const { server } = createMockServer();
39
+ registerCampaignAddAction(server);
40
+ expect(server.tool).toHaveBeenCalledOnce();
41
+ expect(server.tool).toHaveBeenCalledWith("campaign-add-action", expect.any(String), expect.any(Object), expect.any(Function));
42
+ });
43
+ it("successfully adds action with required params", async () => {
44
+ const { server, getHandler } = createMockServer();
45
+ registerCampaignAddAction(server);
46
+ vi.mocked(campaignAddAction).mockResolvedValue(MOCK_ACTION);
47
+ const handler = getHandler("campaign-add-action");
48
+ const result = await handler({
49
+ campaignId: 15,
50
+ name: "Visit & Extract",
51
+ actionType: "VisitAndExtract",
52
+ cdpPort: 9222,
53
+ });
54
+ expect(result).toEqual({
55
+ content: [
56
+ {
57
+ type: "text",
58
+ text: JSON.stringify(MOCK_ACTION, null, 2),
59
+ },
60
+ ],
61
+ });
62
+ });
63
+ it("returns error for invalid actionSettings JSON", async () => {
64
+ const { server, getHandler } = createMockServer();
65
+ registerCampaignAddAction(server);
66
+ const handler = getHandler("campaign-add-action");
67
+ const result = await handler({
68
+ campaignId: 15,
69
+ name: "Visit & Extract",
70
+ actionType: "VisitAndExtract",
71
+ actionSettings: "{not-valid-json",
72
+ cdpPort: 9222,
73
+ });
74
+ expect(result).toEqual({
75
+ isError: true,
76
+ content: [
77
+ {
78
+ type: "text",
79
+ text: "Invalid JSON in actionSettings.",
80
+ },
81
+ ],
82
+ });
83
+ });
84
+ it("returns error for non-existent campaign", async () => {
85
+ const { server, getHandler } = createMockServer();
86
+ registerCampaignAddAction(server);
87
+ vi.mocked(campaignAddAction).mockRejectedValue(new CampaignNotFoundError(999));
88
+ const handler = getHandler("campaign-add-action");
89
+ const result = await handler({
90
+ campaignId: 999,
91
+ name: "Visit",
92
+ actionType: "VisitAndExtract",
93
+ cdpPort: 9222,
94
+ });
95
+ expect(result).toEqual({
96
+ isError: true,
97
+ content: [
98
+ {
99
+ type: "text",
100
+ text: "Campaign 999 not found.",
101
+ },
102
+ ],
103
+ });
104
+ });
105
+ describeInfrastructureErrors(registerCampaignAddAction, "campaign-add-action", () => ({ campaignId: 15, name: "Visit", actionType: "VisitAndExtract", cdpPort: 9222 }), (error) => vi.mocked(campaignAddAction).mockRejectedValue(error));
106
+ });
107
+ //# sourceMappingURL=campaign-add-action.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"campaign-add-action.test.js","sourceRoot":"","sources":["../../src/tools/campaign-add-action.test.ts"],"names":[],"mappings":"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"}