@lhremote/mcp 0.7.0 → 0.9.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 (236) hide show
  1. package/README.md +6 -3
  2. package/dist/helpers.d.ts +2 -15
  3. package/dist/helpers.d.ts.map +1 -1
  4. package/dist/helpers.js +15 -16
  5. package/dist/helpers.js.map +1 -1
  6. package/dist/server.test.js +22 -1
  7. package/dist/server.test.js.map +1 -1
  8. package/dist/tools/add-people-to-collection.test.d.ts +2 -0
  9. package/dist/tools/add-people-to-collection.test.d.ts.map +1 -0
  10. package/dist/tools/add-people-to-collection.test.js +61 -0
  11. package/dist/tools/add-people-to-collection.test.js.map +1 -0
  12. package/dist/tools/campaign-delete.d.ts.map +1 -1
  13. package/dist/tools/campaign-delete.js +7 -3
  14. package/dist/tools/campaign-delete.js.map +1 -1
  15. package/dist/tools/campaign-erase.d.ts +4 -0
  16. package/dist/tools/campaign-erase.d.ts.map +1 -0
  17. package/dist/tools/campaign-erase.js +28 -0
  18. package/dist/tools/campaign-erase.js.map +1 -0
  19. package/dist/tools/campaign-erase.test.d.ts +2 -0
  20. package/dist/tools/campaign-erase.test.d.ts.map +1 -0
  21. package/dist/tools/campaign-erase.test.js +87 -0
  22. package/dist/tools/campaign-erase.test.js.map +1 -0
  23. package/dist/tools/check-replies.d.ts.map +1 -1
  24. package/dist/tools/check-replies.js +11 -3
  25. package/dist/tools/check-replies.js.map +1 -1
  26. package/dist/tools/check-replies.test.js +10 -10
  27. package/dist/tools/check-replies.test.js.map +1 -1
  28. package/dist/tools/comment-on-post.d.ts +4 -0
  29. package/dist/tools/comment-on-post.d.ts.map +1 -0
  30. package/dist/tools/comment-on-post.js +26 -0
  31. package/dist/tools/comment-on-post.js.map +1 -0
  32. package/dist/tools/comment-on-post.test.d.ts +2 -0
  33. package/dist/tools/comment-on-post.test.d.ts.map +1 -0
  34. package/dist/tools/comment-on-post.test.js +64 -0
  35. package/dist/tools/comment-on-post.test.js.map +1 -0
  36. package/dist/tools/create-collection.test.d.ts +2 -0
  37. package/dist/tools/create-collection.test.d.ts.map +1 -0
  38. package/dist/tools/create-collection.test.js +55 -0
  39. package/dist/tools/create-collection.test.js.map +1 -0
  40. package/dist/tools/delete-collection.test.d.ts +2 -0
  41. package/dist/tools/delete-collection.test.d.ts.map +1 -0
  42. package/dist/tools/delete-collection.test.js +55 -0
  43. package/dist/tools/delete-collection.test.js.map +1 -0
  44. package/dist/tools/dismiss-errors.d.ts +4 -0
  45. package/dist/tools/dismiss-errors.d.ts.map +1 -0
  46. package/dist/tools/dismiss-errors.js +19 -0
  47. package/dist/tools/dismiss-errors.js.map +1 -0
  48. package/dist/tools/dismiss-errors.test.d.ts +2 -0
  49. package/dist/tools/dismiss-errors.test.d.ts.map +1 -0
  50. package/dist/tools/dismiss-errors.test.js +74 -0
  51. package/dist/tools/dismiss-errors.test.js.map +1 -0
  52. package/dist/tools/endorse-skills.d.ts +4 -0
  53. package/dist/tools/endorse-skills.d.ts.map +1 -0
  54. package/dist/tools/endorse-skills.js +53 -0
  55. package/dist/tools/endorse-skills.js.map +1 -0
  56. package/dist/tools/endorse-skills.test.d.ts +2 -0
  57. package/dist/tools/endorse-skills.test.d.ts.map +1 -0
  58. package/dist/tools/endorse-skills.test.js +58 -0
  59. package/dist/tools/endorse-skills.test.js.map +1 -0
  60. package/dist/tools/enrich-profile.d.ts +4 -0
  61. package/dist/tools/enrich-profile.d.ts.map +1 -0
  62. package/dist/tools/enrich-profile.js +60 -0
  63. package/dist/tools/enrich-profile.js.map +1 -0
  64. package/dist/tools/enrich-profile.test.d.ts +2 -0
  65. package/dist/tools/enrich-profile.test.d.ts.map +1 -0
  66. package/dist/tools/enrich-profile.test.js +66 -0
  67. package/dist/tools/enrich-profile.test.js.map +1 -0
  68. package/dist/tools/find-app.test.js +1 -1
  69. package/dist/tools/find-app.test.js.map +1 -1
  70. package/dist/tools/follow-person.d.ts +4 -0
  71. package/dist/tools/follow-person.d.ts.map +1 -0
  72. package/dist/tools/follow-person.js +47 -0
  73. package/dist/tools/follow-person.js.map +1 -0
  74. package/dist/tools/follow-person.test.d.ts +2 -0
  75. package/dist/tools/follow-person.test.d.ts.map +1 -0
  76. package/dist/tools/follow-person.test.js +66 -0
  77. package/dist/tools/follow-person.test.js.map +1 -0
  78. package/dist/tools/get-action-budget.d.ts +4 -0
  79. package/dist/tools/get-action-budget.d.ts.map +1 -0
  80. package/dist/tools/get-action-budget.js +19 -0
  81. package/dist/tools/get-action-budget.js.map +1 -0
  82. package/dist/tools/get-action-budget.test.d.ts +2 -0
  83. package/dist/tools/get-action-budget.test.d.ts.map +1 -0
  84. package/dist/tools/get-action-budget.test.js +59 -0
  85. package/dist/tools/get-action-budget.test.js.map +1 -0
  86. package/dist/tools/get-errors.js +1 -1
  87. package/dist/tools/get-errors.js.map +1 -1
  88. package/dist/tools/get-errors.test.js +3 -0
  89. package/dist/tools/get-errors.test.js.map +1 -1
  90. package/dist/tools/get-feed.d.ts +4 -0
  91. package/dist/tools/get-feed.d.ts.map +1 -0
  92. package/dist/tools/get-feed.js +37 -0
  93. package/dist/tools/get-feed.js.map +1 -0
  94. package/dist/tools/get-feed.test.d.ts +2 -0
  95. package/dist/tools/get-feed.test.d.ts.map +1 -0
  96. package/dist/tools/get-feed.test.js +72 -0
  97. package/dist/tools/get-feed.test.js.map +1 -0
  98. package/dist/tools/get-post-engagers.d.ts +4 -0
  99. package/dist/tools/get-post-engagers.d.ts.map +1 -0
  100. package/dist/tools/get-post-engagers.js +44 -0
  101. package/dist/tools/get-post-engagers.js.map +1 -0
  102. package/dist/tools/get-post-engagers.test.d.ts +2 -0
  103. package/dist/tools/get-post-engagers.test.d.ts.map +1 -0
  104. package/dist/tools/get-post-engagers.test.js +86 -0
  105. package/dist/tools/get-post-engagers.test.js.map +1 -0
  106. package/dist/tools/get-post-stats.d.ts +4 -0
  107. package/dist/tools/get-post-stats.d.ts.map +1 -0
  108. package/dist/tools/get-post-stats.js +23 -0
  109. package/dist/tools/get-post-stats.js.map +1 -0
  110. package/dist/tools/get-post-stats.test.d.ts +2 -0
  111. package/dist/tools/get-post-stats.test.d.ts.map +1 -0
  112. package/dist/tools/get-post-stats.test.js +64 -0
  113. package/dist/tools/get-post-stats.test.js.map +1 -0
  114. package/dist/tools/get-post.d.ts +4 -0
  115. package/dist/tools/get-post.d.ts.map +1 -0
  116. package/dist/tools/get-post.js +36 -0
  117. package/dist/tools/get-post.js.map +1 -0
  118. package/dist/tools/get-post.test.d.ts +2 -0
  119. package/dist/tools/get-post.test.d.ts.map +1 -0
  120. package/dist/tools/get-post.test.js +90 -0
  121. package/dist/tools/get-post.test.js.map +1 -0
  122. package/dist/tools/get-profile-activity.d.ts +4 -0
  123. package/dist/tools/get-profile-activity.d.ts.map +1 -0
  124. package/dist/tools/get-profile-activity.js +41 -0
  125. package/dist/tools/get-profile-activity.js.map +1 -0
  126. package/dist/tools/get-profile-activity.test.d.ts +2 -0
  127. package/dist/tools/get-profile-activity.test.d.ts.map +1 -0
  128. package/dist/tools/get-profile-activity.test.js +73 -0
  129. package/dist/tools/get-profile-activity.test.js.map +1 -0
  130. package/dist/tools/get-throttle-status.d.ts +4 -0
  131. package/dist/tools/get-throttle-status.d.ts.map +1 -0
  132. package/dist/tools/get-throttle-status.js +19 -0
  133. package/dist/tools/get-throttle-status.js.map +1 -0
  134. package/dist/tools/get-throttle-status.test.d.ts +2 -0
  135. package/dist/tools/get-throttle-status.test.d.ts.map +1 -0
  136. package/dist/tools/get-throttle-status.test.js +56 -0
  137. package/dist/tools/get-throttle-status.test.js.map +1 -0
  138. package/dist/tools/import-people-from-collection.test.d.ts +2 -0
  139. package/dist/tools/import-people-from-collection.test.d.ts.map +1 -0
  140. package/dist/tools/import-people-from-collection.test.js +86 -0
  141. package/dist/tools/import-people-from-collection.test.js.map +1 -0
  142. package/dist/tools/index.d.ts +22 -1
  143. package/dist/tools/index.d.ts.map +1 -1
  144. package/dist/tools/index.js +43 -1
  145. package/dist/tools/index.js.map +1 -1
  146. package/dist/tools/like-person-posts.d.ts +4 -0
  147. package/dist/tools/like-person-posts.d.ts.map +1 -0
  148. package/dist/tools/like-person-posts.js +86 -0
  149. package/dist/tools/like-person-posts.js.map +1 -0
  150. package/dist/tools/like-person-posts.test.d.ts +2 -0
  151. package/dist/tools/like-person-posts.test.d.ts.map +1 -0
  152. package/dist/tools/like-person-posts.test.js +72 -0
  153. package/dist/tools/like-person-posts.test.js.map +1 -0
  154. package/dist/tools/list-accounts.d.ts.map +1 -1
  155. package/dist/tools/list-accounts.js +19 -13
  156. package/dist/tools/list-accounts.js.map +1 -1
  157. package/dist/tools/list-collections.test.d.ts +2 -0
  158. package/dist/tools/list-collections.test.d.ts.map +1 -0
  159. package/dist/tools/list-collections.test.js +56 -0
  160. package/dist/tools/list-collections.test.js.map +1 -0
  161. package/dist/tools/message-person.d.ts +4 -0
  162. package/dist/tools/message-person.d.ts.map +1 -0
  163. package/dist/tools/message-person.js +71 -0
  164. package/dist/tools/message-person.js.map +1 -0
  165. package/dist/tools/message-person.test.d.ts +2 -0
  166. package/dist/tools/message-person.test.d.ts.map +1 -0
  167. package/dist/tools/message-person.test.js +101 -0
  168. package/dist/tools/message-person.test.js.map +1 -0
  169. package/dist/tools/query-messages.integration.test.js +24 -3
  170. package/dist/tools/query-messages.integration.test.js.map +1 -1
  171. package/dist/tools/react-to-post.d.ts +4 -0
  172. package/dist/tools/react-to-post.d.ts.map +1 -0
  173. package/dist/tools/react-to-post.js +34 -0
  174. package/dist/tools/react-to-post.js.map +1 -0
  175. package/dist/tools/react-to-post.test.d.ts +2 -0
  176. package/dist/tools/react-to-post.test.d.ts.map +1 -0
  177. package/dist/tools/react-to-post.test.js +64 -0
  178. package/dist/tools/react-to-post.test.js.map +1 -0
  179. package/dist/tools/remove-connection.d.ts +4 -0
  180. package/dist/tools/remove-connection.d.ts.map +1 -0
  181. package/dist/tools/remove-connection.js +39 -0
  182. package/dist/tools/remove-connection.js.map +1 -0
  183. package/dist/tools/remove-connection.test.d.ts +2 -0
  184. package/dist/tools/remove-connection.test.d.ts.map +1 -0
  185. package/dist/tools/remove-connection.test.js +58 -0
  186. package/dist/tools/remove-connection.test.js.map +1 -0
  187. package/dist/tools/remove-people-from-collection.test.d.ts +2 -0
  188. package/dist/tools/remove-people-from-collection.test.d.ts.map +1 -0
  189. package/dist/tools/remove-people-from-collection.test.js +60 -0
  190. package/dist/tools/remove-people-from-collection.test.js.map +1 -0
  191. package/dist/tools/scrape-messaging-history.d.ts.map +1 -1
  192. package/dist/tools/scrape-messaging-history.js +6 -2
  193. package/dist/tools/scrape-messaging-history.js.map +1 -1
  194. package/dist/tools/search-posts.d.ts +4 -0
  195. package/dist/tools/search-posts.d.ts.map +1 -0
  196. package/dist/tools/search-posts.js +43 -0
  197. package/dist/tools/search-posts.js.map +1 -0
  198. package/dist/tools/search-posts.test.d.ts +2 -0
  199. package/dist/tools/search-posts.test.d.ts.map +1 -0
  200. package/dist/tools/search-posts.test.js +86 -0
  201. package/dist/tools/search-posts.test.js.map +1 -0
  202. package/dist/tools/send-inmail.d.ts +4 -0
  203. package/dist/tools/send-inmail.d.ts.map +1 -0
  204. package/dist/tools/send-inmail.js +71 -0
  205. package/dist/tools/send-inmail.js.map +1 -0
  206. package/dist/tools/send-inmail.test.d.ts +2 -0
  207. package/dist/tools/send-inmail.test.d.ts.map +1 -0
  208. package/dist/tools/send-inmail.test.js +82 -0
  209. package/dist/tools/send-inmail.test.js.map +1 -0
  210. package/dist/tools/send-invite.d.ts +4 -0
  211. package/dist/tools/send-invite.d.ts.map +1 -0
  212. package/dist/tools/send-invite.js +57 -0
  213. package/dist/tools/send-invite.js.map +1 -0
  214. package/dist/tools/send-invite.test.d.ts +2 -0
  215. package/dist/tools/send-invite.test.d.ts.map +1 -0
  216. package/dist/tools/send-invite.test.js +80 -0
  217. package/dist/tools/send-invite.test.js.map +1 -0
  218. package/dist/tools/start-instance.d.ts.map +1 -1
  219. package/dist/tools/start-instance.js +32 -26
  220. package/dist/tools/start-instance.js.map +1 -1
  221. package/dist/tools/stop-instance.d.ts.map +1 -1
  222. package/dist/tools/stop-instance.js +28 -22
  223. package/dist/tools/stop-instance.js.map +1 -1
  224. package/dist/tools/testing/ephemeral-action-errors.d.ts +9 -0
  225. package/dist/tools/testing/ephemeral-action-errors.d.ts.map +1 -0
  226. package/dist/tools/testing/ephemeral-action-errors.js +48 -0
  227. package/dist/tools/testing/ephemeral-action-errors.js.map +1 -0
  228. package/dist/tools/visit-profile.d.ts +4 -0
  229. package/dist/tools/visit-profile.d.ts.map +1 -0
  230. package/dist/tools/visit-profile.js +37 -0
  231. package/dist/tools/visit-profile.js.map +1 -0
  232. package/dist/tools/visit-profile.test.d.ts +2 -0
  233. package/dist/tools/visit-profile.test.d.ts.map +1 -0
  234. package/dist/tools/visit-profile.test.js +164 -0
  235. package/dist/tools/visit-profile.test.js.map +1 -0
  236. package/package.json +2 -2
@@ -0,0 +1,58 @@
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
+ removeConnection: vi.fn(),
9
+ };
10
+ });
11
+ import { removeConnection, } from "@lhremote/core";
12
+ import { registerRemoveConnection } from "./remove-connection.js";
13
+ import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
14
+ import { describeEphemeralActionErrors } from "./testing/ephemeral-action-errors.js";
15
+ import { createMockServer } from "./testing/mock-server.js";
16
+ const MOCK_RESULT = {
17
+ success: true,
18
+ personId: 100,
19
+ results: [{ id: 1, actionVersionId: 1, personId: 100, result: 1, platform: null, createdAt: "2026-01-01T00:00:00Z", profile: null }],
20
+ };
21
+ describe("registerRemoveConnection", () => {
22
+ beforeEach(() => {
23
+ vi.clearAllMocks();
24
+ });
25
+ afterEach(() => {
26
+ vi.restoreAllMocks();
27
+ });
28
+ it("registers a tool named remove-connection", () => {
29
+ const { server } = createMockServer();
30
+ registerRemoveConnection(server);
31
+ expect(server.tool).toHaveBeenCalledOnce();
32
+ expect(server.tool).toHaveBeenCalledWith("remove-connection", expect.any(String), expect.any(Object), expect.any(Function));
33
+ });
34
+ it("removes connection on success", async () => {
35
+ const { server, getHandler } = createMockServer();
36
+ registerRemoveConnection(server);
37
+ vi.mocked(removeConnection).mockResolvedValue(MOCK_RESULT);
38
+ const handler = getHandler("remove-connection");
39
+ const result = await handler({ personId: 100, cdpPort: 9222 });
40
+ expect(removeConnection).toHaveBeenCalledWith(expect.objectContaining({ personId: 100, cdpPort: 9222 }));
41
+ expect(result).toEqual({
42
+ content: [{ type: "text", text: JSON.stringify(MOCK_RESULT, null, 2) }],
43
+ });
44
+ });
45
+ it("returns error when neither personId nor url provided", async () => {
46
+ const { server, getHandler } = createMockServer();
47
+ registerRemoveConnection(server);
48
+ const handler = getHandler("remove-connection");
49
+ const result = await handler({ cdpPort: 9222 });
50
+ expect(result).toEqual({
51
+ isError: true,
52
+ content: [{ type: "text", text: "Exactly one of personId or url must be provided." }],
53
+ });
54
+ });
55
+ describeInfrastructureErrors(registerRemoveConnection, "remove-connection", () => ({ personId: 100, cdpPort: 9222 }), (error) => vi.mocked(removeConnection).mockRejectedValue(error), "Failed to remove connection");
56
+ describeEphemeralActionErrors(registerRemoveConnection, "remove-connection", () => ({ personId: 100, cdpPort: 9222 }), (error) => vi.mocked(removeConnection).mockRejectedValue(error), "Failed to remove connection");
57
+ });
58
+ //# sourceMappingURL=remove-connection.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"remove-connection.test.js","sourceRoot":"","sources":["../../src/tools/remove-connection.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=remove-people-from-collection.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"remove-people-from-collection.test.d.ts","sourceRoot":"","sources":["../../src/tools/remove-people-from-collection.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,60 @@
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 { ...actual, removePeopleFromCollection: vi.fn() };
7
+ });
8
+ import { removePeopleFromCollection } from "@lhremote/core";
9
+ import { registerRemovePeopleFromCollection } from "./remove-people-from-collection.js";
10
+ import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
11
+ import { createMockServer } from "./testing/mock-server.js";
12
+ const MOCK_RESULT = {
13
+ success: true,
14
+ collectionId: 1,
15
+ removed: 2,
16
+ };
17
+ describe("registerRemovePeopleFromCollection", () => {
18
+ beforeEach(() => {
19
+ vi.clearAllMocks();
20
+ });
21
+ afterEach(() => {
22
+ vi.restoreAllMocks();
23
+ });
24
+ it("registers a tool named remove-people-from-collection", () => {
25
+ const { server } = createMockServer();
26
+ registerRemovePeopleFromCollection(server);
27
+ expect(server.tool).toHaveBeenCalledOnce();
28
+ expect(server.tool).toHaveBeenCalledWith("remove-people-from-collection", expect.any(String), expect.any(Object), expect.any(Function));
29
+ });
30
+ it("returns result as JSON on success", async () => {
31
+ const { server, getHandler } = createMockServer();
32
+ registerRemovePeopleFromCollection(server);
33
+ vi.mocked(removePeopleFromCollection).mockResolvedValue(MOCK_RESULT);
34
+ const handler = getHandler("remove-people-from-collection");
35
+ const result = await handler({
36
+ collectionId: 1,
37
+ personIds: [100, 200],
38
+ cdpPort: 9222,
39
+ });
40
+ expect(result).toEqual({
41
+ content: [{ type: "text", text: JSON.stringify(MOCK_RESULT, null, 2) }],
42
+ });
43
+ });
44
+ it("returns error on failure", async () => {
45
+ const { server, getHandler } = createMockServer();
46
+ registerRemovePeopleFromCollection(server);
47
+ vi.mocked(removePeopleFromCollection).mockRejectedValue(new Error("connection refused"));
48
+ const handler = getHandler("remove-people-from-collection");
49
+ const result = (await handler({
50
+ collectionId: 1,
51
+ personIds: [100, 200],
52
+ cdpPort: 9222,
53
+ }));
54
+ expect(result.isError).toBe(true);
55
+ const text = result.content[0].text;
56
+ expect(text).toContain("Failed to remove people from collection");
57
+ });
58
+ describeInfrastructureErrors(registerRemovePeopleFromCollection, "remove-people-from-collection", () => ({ collectionId: 1, personIds: [100, 200], cdpPort: 9222 }), (error) => vi.mocked(removePeopleFromCollection).mockRejectedValue(error), "Failed to remove people from collection");
59
+ });
60
+ //# sourceMappingURL=remove-people-from-collection.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"remove-people-from-collection.test.js","sourceRoot":"","sources":["../../src/tools/remove-people-from-collection.test.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"scrape-messaging-history.d.ts","sourceRoot":"","sources":["../../src/tools/scrape-messaging-history.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAOzE,mIAAmI;AACnI,wBAAgB,8BAA8B,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAoBtE"}
1
+ {"version":3,"file":"scrape-messaging-history.d.ts","sourceRoot":"","sources":["../../src/tools/scrape-messaging-history.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAOzE,mIAAmI;AACnI,wBAAgB,8BAA8B,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CA0BtE"}
@@ -10,10 +10,14 @@ export function registerScrapeMessagingHistory(server) {
10
10
  .array(z.number().int().positive())
11
11
  .nonempty()
12
12
  .describe("Person IDs whose messaging history should be scraped"),
13
+ pauseOthers: z
14
+ .boolean()
15
+ .optional()
16
+ .describe("Pause all other campaigns during execution to ensure the runner is available, then restore them"),
13
17
  ...cdpConnectionSchema,
14
- }, async ({ personIds, cdpPort, cdpHost, allowRemote }) => {
18
+ }, async ({ personIds, pauseOthers, cdpPort, cdpHost, allowRemote }) => {
15
19
  try {
16
- const result = await scrapeMessagingHistory({ personIds, cdpPort, cdpHost, allowRemote });
20
+ const result = await scrapeMessagingHistory({ personIds, pauseOthers, cdpPort, cdpHost, allowRemote });
17
21
  return mcpSuccess(JSON.stringify(result, null, 2));
18
22
  }
19
23
  catch (error) {
@@ -1 +1 @@
1
- {"version":3,"file":"scrape-messaging-history.js","sourceRoot":"","sources":["../../src/tools/scrape-messaging-history.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,sBAAsB,GACvB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE7E,mIAAmI;AACnI,MAAM,UAAU,8BAA8B,CAAC,MAAiB;IAC9D,MAAM,CAAC,IAAI,CACT,0BAA0B,EAC1B,+MAA+M,EAC/M;QACE,SAAS,EAAE,CAAC;aACT,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,CAAC;aAClC,QAAQ,EAAE;aACV,QAAQ,CAAC,sDAAsD,CAAC;QACnE,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QACrD,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,sBAAsB,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;YAC1F,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,oCAAoC,CAAC,CAAC;QAClE,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"scrape-messaging-history.js","sourceRoot":"","sources":["../../src/tools/scrape-messaging-history.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,sBAAsB,GACvB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE7E,mIAAmI;AACnI,MAAM,UAAU,8BAA8B,CAAC,MAAiB;IAC9D,MAAM,CAAC,IAAI,CACT,0BAA0B,EAC1B,+MAA+M,EAC/M;QACE,SAAS,EAAE,CAAC;aACT,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,CAAC;aAClC,QAAQ,EAAE;aACV,QAAQ,CAAC,sDAAsD,CAAC;QACnE,WAAW,EAAE,CAAC;aACX,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CACP,iGAAiG,CAClG;QACH,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAClE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,sBAAsB,CAAC,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;YACvG,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,oCAAoC,CAAC,CAAC;QAClE,CAAC;IACH,CAAC,CACF,CAAC;AACJ,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#search-posts | search-posts} MCP tool. */
3
+ export declare function registerSearchPosts(server: McpServer): void;
4
+ //# sourceMappingURL=search-posts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-posts.d.ts","sourceRoot":"","sources":["../../src/tools/search-posts.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE,2GAA2G;AAC3G,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CA2C3D"}
@@ -0,0 +1,43 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { searchPosts } from "@lhremote/core";
4
+ import { z } from "zod";
5
+ import { cdpConnectionSchema, mcpCatchAll, mcpSuccess } from "../helpers.js";
6
+ /** Register the {@link https://github.com/alexey-pelykh/lhremote#search-posts | search-posts} MCP tool. */
7
+ export function registerSearchPosts(server) {
8
+ server.tool("search-posts", "Search LinkedIn for posts by keyword or hashtag. Returns structured post data (URL, author, text, media type, engagement counts, timestamp) with cursor-based pagination.", {
9
+ query: z
10
+ .string()
11
+ .describe('Search query — keywords (e.g. "AI agents") or hashtag (e.g. "#AIAgents")'),
12
+ count: z
13
+ .number()
14
+ .int()
15
+ .positive()
16
+ .optional()
17
+ .default(10)
18
+ .describe("Number of results per page (default: 10)"),
19
+ cursor: z
20
+ .number()
21
+ .int()
22
+ .nonnegative()
23
+ .optional()
24
+ .describe("Index-based cursor from a previous search-posts call for the next page"),
25
+ ...cdpConnectionSchema,
26
+ }, async ({ query, count, cursor, cdpPort, cdpHost, allowRemote }) => {
27
+ try {
28
+ const result = await searchPosts({
29
+ query,
30
+ count,
31
+ cursor,
32
+ cdpPort,
33
+ cdpHost,
34
+ allowRemote,
35
+ });
36
+ return mcpSuccess(JSON.stringify(result, null, 2));
37
+ }
38
+ catch (error) {
39
+ return mcpCatchAll(error, "Failed to search posts");
40
+ }
41
+ });
42
+ }
43
+ //# sourceMappingURL=search-posts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-posts.js","sourceRoot":"","sources":["../../src/tools/search-posts.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE7E,2GAA2G;AAC3G,MAAM,UAAU,mBAAmB,CAAC,MAAiB;IACnD,MAAM,CAAC,IAAI,CACT,cAAc,EACd,2KAA2K,EAC3K;QACE,KAAK,EAAE,CAAC;aACL,MAAM,EAAE;aACR,QAAQ,CACP,0EAA0E,CAC3E;QACH,KAAK,EAAE,CAAC;aACL,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,EAAE;aACV,OAAO,CAAC,EAAE,CAAC;aACX,QAAQ,CAAC,0CAA0C,CAAC;QACvD,MAAM,EAAE,CAAC;aACN,MAAM,EAAE;aACR,GAAG,EAAE;aACL,WAAW,EAAE;aACb,QAAQ,EAAE;aACV,QAAQ,CACP,wEAAwE,CACzE;QACH,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAChE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC;gBAC/B,KAAK;gBACL,KAAK;gBACL,MAAM;gBACN,OAAO;gBACP,OAAO;gBACP,WAAW;aACZ,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,wBAAwB,CAAC,CAAC;QACtD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=search-posts.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-posts.test.d.ts","sourceRoot":"","sources":["../../src/tools/search-posts.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,86 @@
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 { ...actual, searchPosts: vi.fn() };
7
+ });
8
+ import { searchPosts } from "@lhremote/core";
9
+ import { registerSearchPosts } from "./search-posts.js";
10
+ import { createMockServer } from "./testing/mock-server.js";
11
+ const MOCK_RESULTS = {
12
+ query: "AI agents",
13
+ posts: [
14
+ {
15
+ url: "https://www.linkedin.com/feed/update/urn:li:activity:7123456789012345678/",
16
+ authorName: "Jane Smith",
17
+ authorHeadline: "CEO at Acme Corp",
18
+ authorProfileUrl: "https://www.linkedin.com/in/janesmith",
19
+ authorPublicId: null,
20
+ text: "Excited about AI agents!",
21
+ mediaType: null,
22
+ reactionCount: 42,
23
+ commentCount: 7,
24
+ shareCount: 3,
25
+ timestamp: null,
26
+ hashtags: [],
27
+ },
28
+ ],
29
+ nextCursor: null,
30
+ };
31
+ describe("registerSearchPosts", () => {
32
+ beforeEach(() => {
33
+ vi.clearAllMocks();
34
+ });
35
+ afterEach(() => {
36
+ vi.restoreAllMocks();
37
+ });
38
+ it("registers a tool named search-posts", () => {
39
+ const { server } = createMockServer();
40
+ registerSearchPosts(server);
41
+ expect(server.tool).toHaveBeenCalledOnce();
42
+ expect(server.tool).toHaveBeenCalledWith("search-posts", expect.any(String), expect.any(Object), expect.any(Function));
43
+ });
44
+ it("returns search results as JSON on success", async () => {
45
+ const { server, getHandler } = createMockServer();
46
+ registerSearchPosts(server);
47
+ vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
48
+ const handler = getHandler("search-posts");
49
+ const result = await handler({
50
+ query: "AI agents",
51
+ cdpPort: 9222,
52
+ });
53
+ expect(result).toEqual({
54
+ content: [
55
+ { type: "text", text: JSON.stringify(MOCK_RESULTS, null, 2) },
56
+ ],
57
+ });
58
+ });
59
+ it("passes pagination parameters to operation", async () => {
60
+ const { server, getHandler } = createMockServer();
61
+ registerSearchPosts(server);
62
+ vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
63
+ const handler = getHandler("search-posts");
64
+ await handler({
65
+ query: "AI agents",
66
+ cursor: 10,
67
+ count: 5,
68
+ cdpPort: 9222,
69
+ });
70
+ expect(searchPosts).toHaveBeenCalledWith(expect.objectContaining({ query: "AI agents", cursor: 10, count: 5 }));
71
+ });
72
+ it("returns error on failure", async () => {
73
+ const { server, getHandler } = createMockServer();
74
+ registerSearchPosts(server);
75
+ vi.mocked(searchPosts).mockRejectedValue(new Error("connection refused"));
76
+ const handler = getHandler("search-posts");
77
+ const result = (await handler({
78
+ query: "AI agents",
79
+ cdpPort: 9222,
80
+ }));
81
+ expect(result.isError).toBe(true);
82
+ const text = result.content[0].text;
83
+ expect(text).toContain("Failed to search posts");
84
+ });
85
+ });
86
+ //# sourceMappingURL=search-posts.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-posts.test.js","sourceRoot":"","sources":["../../src/tools/search-posts.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
2
+ /** Register the {@link https://github.com/alexey-pelykh/lhremote#send-inmail | send-inmail} MCP tool. */
3
+ export declare function registerSendInmail(server: McpServer): void;
4
+ //# sourceMappingURL=send-inmail.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-inmail.d.ts","sourceRoot":"","sources":["../../src/tools/send-inmail.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE,yGAAyG;AACzG,wBAAgB,kBAAkB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAoE1D"}
@@ -0,0 +1,71 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { sendInmail } 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#send-inmail | send-inmail} MCP tool. */
7
+ export function registerSendInmail(server) {
8
+ server.tool("send-inmail", "Send an InMail message to a LinkedIn member (no connection required) via an ephemeral campaign. Accepts a person ID or LinkedIn profile URL. Deducts from the daily action budget.", {
9
+ personId: z
10
+ .number()
11
+ .int()
12
+ .positive()
13
+ .optional()
14
+ .describe("Internal person ID"),
15
+ url: z
16
+ .string()
17
+ .optional()
18
+ .describe("LinkedIn profile URL (e.g. https://www.linkedin.com/in/jane-doe)"),
19
+ messageTemplate: z
20
+ .string()
21
+ .describe("InMail body template as JSON string (required)"),
22
+ subjectTemplate: z
23
+ .string()
24
+ .optional()
25
+ .describe("InMail subject line template as JSON string"),
26
+ rejectIfReplied: z
27
+ .boolean()
28
+ .optional()
29
+ .describe("Skip if person already replied"),
30
+ proceedOnOutOfCredits: z
31
+ .boolean()
32
+ .optional()
33
+ .describe("Continue even when InMail credits are exhausted"),
34
+ keepCampaign: z
35
+ .boolean()
36
+ .optional()
37
+ .describe("Archive the ephemeral campaign instead of deleting it"),
38
+ ...cdpConnectionSchema,
39
+ }, async ({ personId, url, messageTemplate, subjectTemplate, rejectIfReplied, proceedOnOutOfCredits, keepCampaign, cdpPort, cdpHost, allowRemote }) => {
40
+ if ((personId == null) === (url == null)) {
41
+ return mcpError("Exactly one of personId or url must be provided.");
42
+ }
43
+ let parsedMessageTemplate;
44
+ try {
45
+ parsedMessageTemplate = JSON.parse(messageTemplate);
46
+ }
47
+ catch {
48
+ return mcpError("Invalid JSON in messageTemplate.");
49
+ }
50
+ let parsedSubjectTemplate;
51
+ if (subjectTemplate) {
52
+ try {
53
+ parsedSubjectTemplate = JSON.parse(subjectTemplate);
54
+ }
55
+ catch {
56
+ return mcpError("Invalid JSON in subjectTemplate.");
57
+ }
58
+ }
59
+ try {
60
+ const result = await sendInmail({
61
+ personId, url, messageTemplate: parsedMessageTemplate, subjectTemplate: parsedSubjectTemplate,
62
+ rejectIfReplied, proceedOnOutOfCredits, keepCampaign, cdpPort, cdpHost, allowRemote,
63
+ });
64
+ return mcpSuccess(JSON.stringify(result, null, 2));
65
+ }
66
+ catch (error) {
67
+ return mcpCatchAll(error, "Failed to send InMail");
68
+ }
69
+ });
70
+ }
71
+ //# sourceMappingURL=send-inmail.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-inmail.js","sourceRoot":"","sources":["../../src/tools/send-inmail.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEvF,yGAAyG;AACzG,MAAM,UAAU,kBAAkB,CAAC,MAAiB;IAClD,MAAM,CAAC,IAAI,CACT,aAAa,EACb,oLAAoL,EACpL;QACE,QAAQ,EAAE,CAAC;aACR,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,EAAE;aACV,QAAQ,CAAC,oBAAoB,CAAC;QACjC,GAAG,EAAE,CAAC;aACH,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,kEAAkE,CAAC;QAC/E,eAAe,EAAE,CAAC;aACf,MAAM,EAAE;aACR,QAAQ,CAAC,gDAAgD,CAAC;QAC7D,eAAe,EAAE,CAAC;aACf,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,6CAA6C,CAAC;QAC1D,eAAe,EAAE,CAAC;aACf,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CAAC,gCAAgC,CAAC;QAC7C,qBAAqB,EAAE,CAAC;aACrB,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CAAC,iDAAiD,CAAC;QAC9D,YAAY,EAAE,CAAC;aACZ,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CAAC,uDAAuD,CAAC;QACpE,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,QAAQ,EAAE,GAAG,EAAE,eAAe,EAAE,eAAe,EAAE,eAAe,EAAE,qBAAqB,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QACjJ,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;YACzC,OAAO,QAAQ,CAAC,kDAAkD,CAAC,CAAC;QACtE,CAAC;QAED,IAAI,qBAA8C,CAAC;QACnD,IAAI,CAAC;YACH,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAA4B,CAAC;QACjF,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,QAAQ,CAAC,kCAAkC,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,qBAA0D,CAAC;QAC/D,IAAI,eAAe,EAAE,CAAC;YACpB,IAAI,CAAC;gBACH,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAA4B,CAAC;YACjF,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,QAAQ,CAAC,kCAAkC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QAED,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,UAAU,CAAC;gBAC9B,QAAQ,EAAE,GAAG,EAAE,eAAe,EAAE,qBAAqB,EAAE,eAAe,EAAE,qBAAqB;gBAC7F,eAAe,EAAE,qBAAqB,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW;aACpF,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,uBAAuB,CAAC,CAAC;QACrD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=send-inmail.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-inmail.test.d.ts","sourceRoot":"","sources":["../../src/tools/send-inmail.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,82 @@
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
+ sendInmail: vi.fn(),
9
+ };
10
+ });
11
+ import { sendInmail, } from "@lhremote/core";
12
+ import { registerSendInmail } from "./send-inmail.js";
13
+ import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
14
+ import { describeEphemeralActionErrors } from "./testing/ephemeral-action-errors.js";
15
+ import { createMockServer } from "./testing/mock-server.js";
16
+ const MOCK_RESULT = {
17
+ success: true,
18
+ personId: 100,
19
+ results: [{ id: 1, actionVersionId: 1, personId: 100, result: 1, platform: null, createdAt: "2026-01-01T00:00:00Z", profile: null }],
20
+ };
21
+ describe("registerSendInmail", () => {
22
+ beforeEach(() => {
23
+ vi.clearAllMocks();
24
+ });
25
+ afterEach(() => {
26
+ vi.restoreAllMocks();
27
+ });
28
+ it("registers a tool named send-inmail", () => {
29
+ const { server } = createMockServer();
30
+ registerSendInmail(server);
31
+ expect(server.tool).toHaveBeenCalledOnce();
32
+ expect(server.tool).toHaveBeenCalledWith("send-inmail", expect.any(String), expect.any(Object), expect.any(Function));
33
+ });
34
+ it("sends InMail on success with personId", async () => {
35
+ const { server, getHandler } = createMockServer();
36
+ registerSendInmail(server);
37
+ vi.mocked(sendInmail).mockResolvedValue(MOCK_RESULT);
38
+ const handler = getHandler("send-inmail");
39
+ const result = await handler({
40
+ personId: 100,
41
+ messageTemplate: '{"type":"text","value":"Hello"}',
42
+ cdpPort: 9222,
43
+ });
44
+ expect(sendInmail).toHaveBeenCalledWith(expect.objectContaining({
45
+ personId: 100,
46
+ messageTemplate: { type: "text", value: "Hello" },
47
+ }));
48
+ expect(result).toEqual({
49
+ content: [{ type: "text", text: JSON.stringify(MOCK_RESULT, null, 2) }],
50
+ });
51
+ });
52
+ it("returns error when neither personId nor url provided", async () => {
53
+ const { server, getHandler } = createMockServer();
54
+ registerSendInmail(server);
55
+ const handler = getHandler("send-inmail");
56
+ const result = await handler({
57
+ messageTemplate: '{"type":"text","value":"Hi"}',
58
+ cdpPort: 9222,
59
+ });
60
+ expect(result).toEqual({
61
+ isError: true,
62
+ content: [{ type: "text", text: "Exactly one of personId or url must be provided." }],
63
+ });
64
+ });
65
+ it("returns error on invalid messageTemplate JSON", async () => {
66
+ const { server, getHandler } = createMockServer();
67
+ registerSendInmail(server);
68
+ const handler = getHandler("send-inmail");
69
+ const result = await handler({
70
+ personId: 100,
71
+ messageTemplate: "not-json",
72
+ cdpPort: 9222,
73
+ });
74
+ expect(result).toEqual({
75
+ isError: true,
76
+ content: [{ type: "text", text: "Invalid JSON in messageTemplate." }],
77
+ });
78
+ });
79
+ describeInfrastructureErrors(registerSendInmail, "send-inmail", () => ({ personId: 100, messageTemplate: '{"type":"text","value":"Hi"}', cdpPort: 9222 }), (error) => vi.mocked(sendInmail).mockRejectedValue(error), "Failed to send InMail");
80
+ describeEphemeralActionErrors(registerSendInmail, "send-inmail", () => ({ personId: 100, messageTemplate: '{"type":"text","value":"Hi"}', cdpPort: 9222 }), (error) => vi.mocked(sendInmail).mockRejectedValue(error), "Failed to send InMail");
81
+ });
82
+ //# sourceMappingURL=send-inmail.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-inmail.test.js","sourceRoot":"","sources":["../../src/tools/send-inmail.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
2
+ /** Register the {@link https://github.com/alexey-pelykh/lhremote#send-invite | send-invite} MCP tool. */
3
+ export declare function registerSendInvite(server: McpServer): void;
4
+ //# sourceMappingURL=send-invite.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-invite.d.ts","sourceRoot":"","sources":["../../src/tools/send-invite.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE,yGAAyG;AACzG,wBAAgB,kBAAkB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAsD1D"}
@@ -0,0 +1,57 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { sendInvite } 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#send-invite | send-invite} MCP tool. */
7
+ export function registerSendInvite(server) {
8
+ server.tool("send-invite", "Send a LinkedIn connection request via an ephemeral campaign. Accepts a person ID or LinkedIn profile URL. Deducts from the daily action budget.", {
9
+ personId: z
10
+ .number()
11
+ .int()
12
+ .positive()
13
+ .optional()
14
+ .describe("Internal person ID"),
15
+ url: z
16
+ .string()
17
+ .optional()
18
+ .describe("LinkedIn profile URL (e.g. https://www.linkedin.com/in/jane-doe)"),
19
+ messageTemplate: z
20
+ .string()
21
+ .optional()
22
+ .describe("Invitation message template as JSON string (empty for no message)"),
23
+ saveAsLeadSN: z
24
+ .boolean()
25
+ .optional()
26
+ .describe("Save as lead in Sales Navigator (default: false)"),
27
+ keepCampaign: z
28
+ .boolean()
29
+ .optional()
30
+ .describe("Archive the ephemeral campaign instead of deleting it"),
31
+ ...cdpConnectionSchema,
32
+ }, async ({ personId, url, messageTemplate, saveAsLeadSN, keepCampaign, cdpPort, cdpHost, allowRemote }) => {
33
+ if ((personId == null) === (url == null)) {
34
+ return mcpError("Exactly one of personId or url must be provided.");
35
+ }
36
+ let parsedMessageTemplate;
37
+ if (messageTemplate) {
38
+ try {
39
+ parsedMessageTemplate = JSON.parse(messageTemplate);
40
+ }
41
+ catch {
42
+ return mcpError("Invalid JSON in messageTemplate.");
43
+ }
44
+ }
45
+ try {
46
+ const result = await sendInvite({
47
+ personId, url, messageTemplate: parsedMessageTemplate, saveAsLeadSN,
48
+ keepCampaign, cdpPort, cdpHost, allowRemote,
49
+ });
50
+ return mcpSuccess(JSON.stringify(result, null, 2));
51
+ }
52
+ catch (error) {
53
+ return mcpCatchAll(error, "Failed to send invite");
54
+ }
55
+ });
56
+ }
57
+ //# sourceMappingURL=send-invite.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-invite.js","sourceRoot":"","sources":["../../src/tools/send-invite.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEvF,yGAAyG;AACzG,MAAM,UAAU,kBAAkB,CAAC,MAAiB;IAClD,MAAM,CAAC,IAAI,CACT,aAAa,EACb,kJAAkJ,EAClJ;QACE,QAAQ,EAAE,CAAC;aACR,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,EAAE;aACV,QAAQ,CAAC,oBAAoB,CAAC;QACjC,GAAG,EAAE,CAAC;aACH,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,kEAAkE,CAAC;QAC/E,eAAe,EAAE,CAAC;aACf,MAAM,EAAE;aACR,QAAQ,EAAE;aACV,QAAQ,CAAC,mEAAmE,CAAC;QAChF,YAAY,EAAE,CAAC;aACZ,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CAAC,kDAAkD,CAAC;QAC/D,YAAY,EAAE,CAAC;aACZ,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,QAAQ,CAAC,uDAAuD,CAAC;QACpE,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,QAAQ,EAAE,GAAG,EAAE,eAAe,EAAE,YAAY,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QACtG,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;YACzC,OAAO,QAAQ,CAAC,kDAAkD,CAAC,CAAC;QACtE,CAAC;QAED,IAAI,qBAA0D,CAAC;QAC/D,IAAI,eAAe,EAAE,CAAC;YACpB,IAAI,CAAC;gBACH,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAA4B,CAAC;YACjF,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,QAAQ,CAAC,kCAAkC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QAED,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,UAAU,CAAC;gBAC9B,QAAQ,EAAE,GAAG,EAAE,eAAe,EAAE,qBAAqB,EAAE,YAAY;gBACnE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW;aAC5C,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,uBAAuB,CAAC,CAAC;QACrD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=send-invite.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-invite.test.d.ts","sourceRoot":"","sources":["../../src/tools/send-invite.test.ts"],"names":[],"mappings":""}