@lhremote/mcp 0.7.0 → 0.8.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.
- package/dist/helpers.d.ts.map +1 -1
- package/dist/helpers.js +5 -2
- package/dist/helpers.js.map +1 -1
- package/dist/server.test.js +22 -1
- package/dist/server.test.js.map +1 -1
- package/dist/tools/campaign-delete.d.ts.map +1 -1
- package/dist/tools/campaign-delete.js +7 -3
- package/dist/tools/campaign-delete.js.map +1 -1
- package/dist/tools/campaign-erase.d.ts +4 -0
- package/dist/tools/campaign-erase.d.ts.map +1 -0
- package/dist/tools/campaign-erase.js +28 -0
- package/dist/tools/campaign-erase.js.map +1 -0
- package/dist/tools/campaign-erase.test.d.ts +2 -0
- package/dist/tools/campaign-erase.test.d.ts.map +1 -0
- package/dist/tools/campaign-erase.test.js +87 -0
- package/dist/tools/campaign-erase.test.js.map +1 -0
- package/dist/tools/comment-on-post.d.ts +4 -0
- package/dist/tools/comment-on-post.d.ts.map +1 -0
- package/dist/tools/comment-on-post.js +26 -0
- package/dist/tools/comment-on-post.js.map +1 -0
- package/dist/tools/dismiss-errors.d.ts +4 -0
- package/dist/tools/dismiss-errors.d.ts.map +1 -0
- package/dist/tools/dismiss-errors.js +19 -0
- package/dist/tools/dismiss-errors.js.map +1 -0
- package/dist/tools/dismiss-errors.test.d.ts +2 -0
- package/dist/tools/dismiss-errors.test.d.ts.map +1 -0
- package/dist/tools/dismiss-errors.test.js +74 -0
- package/dist/tools/dismiss-errors.test.js.map +1 -0
- package/dist/tools/endorse-skills.d.ts +4 -0
- package/dist/tools/endorse-skills.d.ts.map +1 -0
- package/dist/tools/endorse-skills.js +53 -0
- package/dist/tools/endorse-skills.js.map +1 -0
- package/dist/tools/endorse-skills.test.d.ts +2 -0
- package/dist/tools/endorse-skills.test.d.ts.map +1 -0
- package/dist/tools/endorse-skills.test.js +58 -0
- package/dist/tools/endorse-skills.test.js.map +1 -0
- package/dist/tools/enrich-profile.d.ts +4 -0
- package/dist/tools/enrich-profile.d.ts.map +1 -0
- package/dist/tools/enrich-profile.js +60 -0
- package/dist/tools/enrich-profile.js.map +1 -0
- package/dist/tools/enrich-profile.test.d.ts +2 -0
- package/dist/tools/enrich-profile.test.d.ts.map +1 -0
- package/dist/tools/enrich-profile.test.js +66 -0
- package/dist/tools/enrich-profile.test.js.map +1 -0
- package/dist/tools/follow-person.d.ts +4 -0
- package/dist/tools/follow-person.d.ts.map +1 -0
- package/dist/tools/follow-person.js +47 -0
- package/dist/tools/follow-person.js.map +1 -0
- package/dist/tools/follow-person.test.d.ts +2 -0
- package/dist/tools/follow-person.test.d.ts.map +1 -0
- package/dist/tools/follow-person.test.js +66 -0
- package/dist/tools/follow-person.test.js.map +1 -0
- package/dist/tools/get-action-budget.d.ts +4 -0
- package/dist/tools/get-action-budget.d.ts.map +1 -0
- package/dist/tools/get-action-budget.js +19 -0
- package/dist/tools/get-action-budget.js.map +1 -0
- package/dist/tools/get-action-budget.test.d.ts +2 -0
- package/dist/tools/get-action-budget.test.d.ts.map +1 -0
- package/dist/tools/get-action-budget.test.js +59 -0
- package/dist/tools/get-action-budget.test.js.map +1 -0
- package/dist/tools/get-errors.js +1 -1
- package/dist/tools/get-errors.js.map +1 -1
- package/dist/tools/get-errors.test.js +3 -0
- package/dist/tools/get-errors.test.js.map +1 -1
- package/dist/tools/get-feed.d.ts +4 -0
- package/dist/tools/get-feed.d.ts.map +1 -0
- package/dist/tools/get-feed.js +37 -0
- package/dist/tools/get-feed.js.map +1 -0
- package/dist/tools/get-feed.test.d.ts +2 -0
- package/dist/tools/get-feed.test.d.ts.map +1 -0
- package/dist/tools/get-feed.test.js +73 -0
- package/dist/tools/get-feed.test.js.map +1 -0
- package/dist/tools/get-post-engagers.d.ts +4 -0
- package/dist/tools/get-post-engagers.d.ts.map +1 -0
- package/dist/tools/get-post-engagers.js +44 -0
- package/dist/tools/get-post-engagers.js.map +1 -0
- package/dist/tools/get-post-engagers.test.d.ts +2 -0
- package/dist/tools/get-post-engagers.test.d.ts.map +1 -0
- package/dist/tools/get-post-engagers.test.js +86 -0
- package/dist/tools/get-post-engagers.test.js.map +1 -0
- package/dist/tools/get-post-stats.d.ts +4 -0
- package/dist/tools/get-post-stats.d.ts.map +1 -0
- package/dist/tools/get-post-stats.js +23 -0
- package/dist/tools/get-post-stats.js.map +1 -0
- package/dist/tools/get-post-stats.test.d.ts +2 -0
- package/dist/tools/get-post-stats.test.d.ts.map +1 -0
- package/dist/tools/get-post-stats.test.js +64 -0
- package/dist/tools/get-post-stats.test.js.map +1 -0
- package/dist/tools/get-post.d.ts +4 -0
- package/dist/tools/get-post.d.ts.map +1 -0
- package/dist/tools/get-post.js +44 -0
- package/dist/tools/get-post.js.map +1 -0
- package/dist/tools/get-profile-activity.d.ts +4 -0
- package/dist/tools/get-profile-activity.d.ts.map +1 -0
- package/dist/tools/get-profile-activity.js +44 -0
- package/dist/tools/get-profile-activity.js.map +1 -0
- package/dist/tools/get-profile-activity.test.d.ts +2 -0
- package/dist/tools/get-profile-activity.test.d.ts.map +1 -0
- package/dist/tools/get-profile-activity.test.js +74 -0
- package/dist/tools/get-profile-activity.test.js.map +1 -0
- package/dist/tools/get-throttle-status.d.ts +4 -0
- package/dist/tools/get-throttle-status.d.ts.map +1 -0
- package/dist/tools/get-throttle-status.js +19 -0
- package/dist/tools/get-throttle-status.js.map +1 -0
- package/dist/tools/get-throttle-status.test.d.ts +2 -0
- package/dist/tools/get-throttle-status.test.d.ts.map +1 -0
- package/dist/tools/get-throttle-status.test.js +56 -0
- package/dist/tools/get-throttle-status.test.js.map +1 -0
- package/dist/tools/index.d.ts +22 -1
- package/dist/tools/index.d.ts.map +1 -1
- package/dist/tools/index.js +43 -1
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/like-person-posts.d.ts +4 -0
- package/dist/tools/like-person-posts.d.ts.map +1 -0
- package/dist/tools/like-person-posts.js +86 -0
- package/dist/tools/like-person-posts.js.map +1 -0
- package/dist/tools/like-person-posts.test.d.ts +2 -0
- package/dist/tools/like-person-posts.test.d.ts.map +1 -0
- package/dist/tools/like-person-posts.test.js +72 -0
- package/dist/tools/like-person-posts.test.js.map +1 -0
- package/dist/tools/message-person.d.ts +4 -0
- package/dist/tools/message-person.d.ts.map +1 -0
- package/dist/tools/message-person.js +71 -0
- package/dist/tools/message-person.js.map +1 -0
- package/dist/tools/message-person.test.d.ts +2 -0
- package/dist/tools/message-person.test.d.ts.map +1 -0
- package/dist/tools/message-person.test.js +101 -0
- package/dist/tools/message-person.test.js.map +1 -0
- package/dist/tools/query-messages.integration.test.js +24 -3
- package/dist/tools/query-messages.integration.test.js.map +1 -1
- package/dist/tools/react-to-post.d.ts +4 -0
- package/dist/tools/react-to-post.d.ts.map +1 -0
- package/dist/tools/react-to-post.js +34 -0
- package/dist/tools/react-to-post.js.map +1 -0
- package/dist/tools/remove-connection.d.ts +4 -0
- package/dist/tools/remove-connection.d.ts.map +1 -0
- package/dist/tools/remove-connection.js +39 -0
- package/dist/tools/remove-connection.js.map +1 -0
- package/dist/tools/remove-connection.test.d.ts +2 -0
- package/dist/tools/remove-connection.test.d.ts.map +1 -0
- package/dist/tools/remove-connection.test.js +58 -0
- package/dist/tools/remove-connection.test.js.map +1 -0
- package/dist/tools/search-posts.d.ts +4 -0
- package/dist/tools/search-posts.d.ts.map +1 -0
- package/dist/tools/search-posts.js +44 -0
- package/dist/tools/search-posts.js.map +1 -0
- package/dist/tools/search-posts.test.d.ts +2 -0
- package/dist/tools/search-posts.test.d.ts.map +1 -0
- package/dist/tools/search-posts.test.js +82 -0
- package/dist/tools/search-posts.test.js.map +1 -0
- package/dist/tools/send-inmail.d.ts +4 -0
- package/dist/tools/send-inmail.d.ts.map +1 -0
- package/dist/tools/send-inmail.js +71 -0
- package/dist/tools/send-inmail.js.map +1 -0
- package/dist/tools/send-inmail.test.d.ts +2 -0
- package/dist/tools/send-inmail.test.d.ts.map +1 -0
- package/dist/tools/send-inmail.test.js +82 -0
- package/dist/tools/send-inmail.test.js.map +1 -0
- package/dist/tools/send-invite.d.ts +4 -0
- package/dist/tools/send-invite.d.ts.map +1 -0
- package/dist/tools/send-invite.js +57 -0
- package/dist/tools/send-invite.js.map +1 -0
- package/dist/tools/send-invite.test.d.ts +2 -0
- package/dist/tools/send-invite.test.d.ts.map +1 -0
- package/dist/tools/send-invite.test.js +80 -0
- package/dist/tools/send-invite.test.js.map +1 -0
- package/dist/tools/testing/ephemeral-action-errors.d.ts +9 -0
- package/dist/tools/testing/ephemeral-action-errors.d.ts.map +1 -0
- package/dist/tools/testing/ephemeral-action-errors.js +48 -0
- package/dist/tools/testing/ephemeral-action-errors.js.map +1 -0
- package/dist/tools/visit-profile.d.ts +4 -0
- package/dist/tools/visit-profile.d.ts.map +1 -0
- package/dist/tools/visit-profile.js +37 -0
- package/dist/tools/visit-profile.js.map +1 -0
- package/dist/tools/visit-profile.test.d.ts +2 -0
- package/dist/tools/visit-profile.test.d.ts.map +1 -0
- package/dist/tools/visit-profile.test.js +164 -0
- package/dist/tools/visit-profile.test.js.map +1 -0
- package/package.json +2 -2
|
@@ -1,10 +1,19 @@
|
|
|
1
1
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
2
|
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { copyFileSync, unlinkSync } from "node:fs";
|
|
4
|
+
import { randomUUID } from "node:crypto";
|
|
5
|
+
import { tmpdir } from "node:os";
|
|
3
6
|
import { dirname, join } from "node:path";
|
|
4
7
|
import { fileURLToPath } from "node:url";
|
|
5
|
-
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
8
|
+
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi, } from "vitest";
|
|
6
9
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
7
|
-
const
|
|
10
|
+
const FIXTURE_ORIGIN = join(__dirname, "../../../core/src/db/testing/fixture.db");
|
|
11
|
+
/**
|
|
12
|
+
* Per-suite copy of the fixture database.
|
|
13
|
+
* Avoids SQLite file-locking contention when multiple vitest
|
|
14
|
+
* workers open the same DB file in parallel.
|
|
15
|
+
*/
|
|
16
|
+
let fixturePath;
|
|
8
17
|
vi.mock("@lhremote/core", async (importOriginal) => {
|
|
9
18
|
const actual = await importOriginal();
|
|
10
19
|
return {
|
|
@@ -20,7 +29,7 @@ import { createMockServer } from "./testing/mock-server.js";
|
|
|
20
29
|
* This replaces the mock so the MCP tool handler exercises real DB queries.
|
|
21
30
|
*/
|
|
22
31
|
function fixtureQueryMessages(input) {
|
|
23
|
-
const client = new DatabaseClient(
|
|
32
|
+
const client = new DatabaseClient(fixturePath);
|
|
24
33
|
try {
|
|
25
34
|
const repo = new MessageRepository(client);
|
|
26
35
|
const limit = input.limit ?? 20;
|
|
@@ -45,6 +54,18 @@ function fixtureQueryMessages(input) {
|
|
|
45
54
|
}
|
|
46
55
|
}
|
|
47
56
|
describe("registerQueryMessages (integration)", () => {
|
|
57
|
+
beforeAll(() => {
|
|
58
|
+
fixturePath = join(tmpdir(), `lhremote-fixture-${randomUUID()}.db`);
|
|
59
|
+
copyFileSync(FIXTURE_ORIGIN, fixturePath);
|
|
60
|
+
});
|
|
61
|
+
afterAll(() => {
|
|
62
|
+
try {
|
|
63
|
+
unlinkSync(fixturePath);
|
|
64
|
+
}
|
|
65
|
+
catch {
|
|
66
|
+
// Ignore cleanup errors
|
|
67
|
+
}
|
|
68
|
+
});
|
|
48
69
|
beforeEach(() => {
|
|
49
70
|
vi.mocked(queryMessages).mockImplementation(async (input) => fixtureQueryMessages(input));
|
|
50
71
|
});
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"query-messages.integration.test.js","sourceRoot":"","sources":["../../src/tools/query-messages.integration.test.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAEzC,OAAO,
|
|
1
|
+
{"version":3,"file":"query-messages.integration.test.js","sourceRoot":"","sources":["../../src/tools/query-messages.integration.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#react-to-post | react-to-post} MCP tool. */
|
|
3
|
+
export declare function registerReactToPost(server: McpServer): void;
|
|
4
|
+
//# sourceMappingURL=react-to-post.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"react-to-post.d.ts","sourceRoot":"","sources":["../../src/tools/react-to-post.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE,6GAA6G;AAC7G,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAkC3D"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { reactToPost, REACTION_TYPES } 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#react-to-post | react-to-post} MCP tool. */
|
|
7
|
+
export function registerReactToPost(server) {
|
|
8
|
+
server.tool("react-to-post", "React to a LinkedIn post with a specific reaction type (like, celebrate, support, love, insightful, funny). Navigates to the post and clicks the reaction button.", {
|
|
9
|
+
postUrl: z
|
|
10
|
+
.string()
|
|
11
|
+
.describe("LinkedIn post URL (e.g. https://www.linkedin.com/feed/update/urn:li:activity:1234567890/)"),
|
|
12
|
+
reactionType: z
|
|
13
|
+
.enum(REACTION_TYPES)
|
|
14
|
+
.optional()
|
|
15
|
+
.default("like")
|
|
16
|
+
.describe("Reaction type to apply (default: like). Options: like, celebrate, support, love, insightful, funny"),
|
|
17
|
+
...cdpConnectionSchema,
|
|
18
|
+
}, async ({ postUrl, reactionType, cdpPort, cdpHost, allowRemote }) => {
|
|
19
|
+
try {
|
|
20
|
+
const result = await reactToPost({
|
|
21
|
+
postUrl,
|
|
22
|
+
reactionType: reactionType,
|
|
23
|
+
cdpPort,
|
|
24
|
+
cdpHost,
|
|
25
|
+
allowRemote,
|
|
26
|
+
});
|
|
27
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
28
|
+
}
|
|
29
|
+
catch (error) {
|
|
30
|
+
return mcpCatchAll(error, "Failed to react to post");
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
//# sourceMappingURL=react-to-post.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"react-to-post.js","sourceRoot":"","sources":["../../src/tools/react-to-post.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAC7D,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE7E,6GAA6G;AAC7G,MAAM,UAAU,mBAAmB,CAAC,MAAiB;IACnD,MAAM,CAAC,IAAI,CACT,eAAe,EACf,mKAAmK,EACnK;QACE,OAAO,EAAE,CAAC;aACP,MAAM,EAAE;aACR,QAAQ,CACP,2FAA2F,CAC5F;QACH,YAAY,EAAE,CAAC;aACZ,IAAI,CAAC,cAAkD,CAAC;aACxD,QAAQ,EAAE;aACV,OAAO,CAAC,MAAM,CAAC;aACf,QAAQ,CACP,oGAAoG,CACrG;QACH,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QACjE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC;gBAC/B,OAAO;gBACP,YAAY,EAAE,YAAiE;gBAC/E,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,yBAAyB,CAAC,CAAC;QACvD,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#remove-connection | remove-connection} MCP tool. */
|
|
3
|
+
export declare function registerRemoveConnection(server: McpServer): void;
|
|
4
|
+
//# sourceMappingURL=remove-connection.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.d.ts","sourceRoot":"","sources":["../../src/tools/remove-connection.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE,qHAAqH;AACrH,wBAAgB,wBAAwB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAoChE"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { removeConnection } 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#remove-connection | remove-connection} MCP tool. */
|
|
7
|
+
export function registerRemoveConnection(server) {
|
|
8
|
+
server.tool("remove-connection", "Remove a person from 1st-degree LinkedIn connections (unfriend) 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
|
+
keepCampaign: z
|
|
20
|
+
.boolean()
|
|
21
|
+
.optional()
|
|
22
|
+
.describe("Archive the ephemeral campaign instead of deleting it"),
|
|
23
|
+
...cdpConnectionSchema,
|
|
24
|
+
}, async ({ personId, url, keepCampaign, cdpPort, cdpHost, allowRemote }) => {
|
|
25
|
+
if ((personId == null) === (url == null)) {
|
|
26
|
+
return mcpError("Exactly one of personId or url must be provided.");
|
|
27
|
+
}
|
|
28
|
+
try {
|
|
29
|
+
const result = await removeConnection({
|
|
30
|
+
personId, url, keepCampaign, cdpPort, cdpHost, allowRemote,
|
|
31
|
+
});
|
|
32
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
33
|
+
}
|
|
34
|
+
catch (error) {
|
|
35
|
+
return mcpCatchAll(error, "Failed to remove connection");
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=remove-connection.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.js","sourceRoot":"","sources":["../../src/tools/remove-connection.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAClD,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEvF,qHAAqH;AACrH,MAAM,UAAU,wBAAwB,CAAC,MAAiB;IACxD,MAAM,CAAC,IAAI,CACT,mBAAmB,EACnB,+KAA+K,EAC/K;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,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,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QACvE,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,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,gBAAgB,CAAC;gBACpC,QAAQ,EAAE,GAAG,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW;aAC3D,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,6BAA6B,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.test.d.ts","sourceRoot":"","sources":["../../src/tools/remove-connection.test.ts"],"names":[],"mappings":""}
|
|
@@ -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,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,CA0C3D"}
|
|
@@ -0,0 +1,44 @@
|
|
|
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 with author info and engagement counts. Supports pagination.", {
|
|
9
|
+
query: z
|
|
10
|
+
.string()
|
|
11
|
+
.describe('Search query — keywords (e.g. "AI agents") or hashtag (e.g. "#AIAgents")'),
|
|
12
|
+
start: z
|
|
13
|
+
.number()
|
|
14
|
+
.int()
|
|
15
|
+
.nonnegative()
|
|
16
|
+
.optional()
|
|
17
|
+
.default(0)
|
|
18
|
+
.describe("Pagination offset (default: 0)"),
|
|
19
|
+
count: z
|
|
20
|
+
.number()
|
|
21
|
+
.int()
|
|
22
|
+
.positive()
|
|
23
|
+
.optional()
|
|
24
|
+
.default(10)
|
|
25
|
+
.describe("Number of results per page (default: 10)"),
|
|
26
|
+
...cdpConnectionSchema,
|
|
27
|
+
}, async ({ query, start, count, cdpPort, cdpHost, allowRemote }) => {
|
|
28
|
+
try {
|
|
29
|
+
const result = await searchPosts({
|
|
30
|
+
query,
|
|
31
|
+
start,
|
|
32
|
+
count,
|
|
33
|
+
cdpPort,
|
|
34
|
+
cdpHost,
|
|
35
|
+
allowRemote,
|
|
36
|
+
});
|
|
37
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
38
|
+
}
|
|
39
|
+
catch (error) {
|
|
40
|
+
return mcpCatchAll(error, "Failed to search posts");
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
//# 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,4IAA4I,EAC5I;QACE,KAAK,EAAE,CAAC;aACL,MAAM,EAAE;aACR,QAAQ,CACP,0EAA0E,CAC3E;QACH,KAAK,EAAE,CAAC;aACL,MAAM,EAAE;aACR,GAAG,EAAE;aACL,WAAW,EAAE;aACb,QAAQ,EAAE;aACV,OAAO,CAAC,CAAC,CAAC;aACV,QAAQ,CAAC,gCAAgC,CAAC;QAC7C,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,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAC/D,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC;gBAC/B,KAAK;gBACL,KAAK;gBACL,KAAK;gBACL,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 @@
|
|
|
1
|
+
{"version":3,"file":"search-posts.test.d.ts","sourceRoot":"","sources":["../../src/tools/search-posts.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 { ...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
|
+
postUrn: "urn:li:activity:7123456789012345678",
|
|
16
|
+
text: "Excited about AI agents!",
|
|
17
|
+
authorFirstName: "Jane",
|
|
18
|
+
authorLastName: "Smith",
|
|
19
|
+
authorPublicId: "janesmith",
|
|
20
|
+
authorHeadline: "CEO at Acme Corp",
|
|
21
|
+
reactionCount: 42,
|
|
22
|
+
commentCount: 7,
|
|
23
|
+
},
|
|
24
|
+
],
|
|
25
|
+
paging: { start: 0, count: 10, total: 1 },
|
|
26
|
+
};
|
|
27
|
+
describe("registerSearchPosts", () => {
|
|
28
|
+
beforeEach(() => {
|
|
29
|
+
vi.clearAllMocks();
|
|
30
|
+
});
|
|
31
|
+
afterEach(() => {
|
|
32
|
+
vi.restoreAllMocks();
|
|
33
|
+
});
|
|
34
|
+
it("registers a tool named search-posts", () => {
|
|
35
|
+
const { server } = createMockServer();
|
|
36
|
+
registerSearchPosts(server);
|
|
37
|
+
expect(server.tool).toHaveBeenCalledOnce();
|
|
38
|
+
expect(server.tool).toHaveBeenCalledWith("search-posts", expect.any(String), expect.any(Object), expect.any(Function));
|
|
39
|
+
});
|
|
40
|
+
it("returns search results as JSON on success", async () => {
|
|
41
|
+
const { server, getHandler } = createMockServer();
|
|
42
|
+
registerSearchPosts(server);
|
|
43
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
44
|
+
const handler = getHandler("search-posts");
|
|
45
|
+
const result = await handler({
|
|
46
|
+
query: "AI agents",
|
|
47
|
+
cdpPort: 9222,
|
|
48
|
+
});
|
|
49
|
+
expect(result).toEqual({
|
|
50
|
+
content: [
|
|
51
|
+
{ type: "text", text: JSON.stringify(MOCK_RESULTS, null, 2) },
|
|
52
|
+
],
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
it("passes pagination parameters to operation", async () => {
|
|
56
|
+
const { server, getHandler } = createMockServer();
|
|
57
|
+
registerSearchPosts(server);
|
|
58
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
59
|
+
const handler = getHandler("search-posts");
|
|
60
|
+
await handler({
|
|
61
|
+
query: "AI agents",
|
|
62
|
+
start: 10,
|
|
63
|
+
count: 5,
|
|
64
|
+
cdpPort: 9222,
|
|
65
|
+
});
|
|
66
|
+
expect(searchPosts).toHaveBeenCalledWith(expect.objectContaining({ query: "AI agents", start: 10, count: 5 }));
|
|
67
|
+
});
|
|
68
|
+
it("returns error on failure", async () => {
|
|
69
|
+
const { server, getHandler } = createMockServer();
|
|
70
|
+
registerSearchPosts(server);
|
|
71
|
+
vi.mocked(searchPosts).mockRejectedValue(new Error("connection refused"));
|
|
72
|
+
const handler = getHandler("search-posts");
|
|
73
|
+
const result = (await handler({
|
|
74
|
+
query: "AI agents",
|
|
75
|
+
cdpPort: 9222,
|
|
76
|
+
}));
|
|
77
|
+
expect(result.isError).toBe(true);
|
|
78
|
+
const text = result.content[0].text;
|
|
79
|
+
expect(text).toContain("Failed to search posts");
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
//# 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":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAEzE,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE;IACjD,MAAM,MAAM,GAAG,MAAM,cAAc,EAAmC,CAAC;IACvE,OAAO,EAAE,GAAG,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC;AAC7C,CAAC,CAAC,CAAC;AAEH,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACxD,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAE5D,MAAM,YAAY,GAAG;IACnB,KAAK,EAAE,WAAW;IAClB,KAAK,EAAE;QACL;YACE,OAAO,EAAE,qCAAqC;YAC9C,IAAI,EAAE,0BAA0B;YAChC,eAAe,EAAE,MAAM;YACvB,cAAc,EAAE,OAAO;YACvB,cAAc,EAAE,WAAW;YAC3B,cAAc,EAAE,kBAAkB;YAClC,aAAa,EAAE,EAAE;YACjB,YAAY,EAAE,CAAC;SAChB;KACF;IACD,MAAM,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE;CAC1C,CAAC;AAEF,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;IACnC,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,eAAe,EAAE,CAAC;IACvB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,EAAE,MAAM,EAAE,GAAG,gBAAgB,EAAE,CAAC;QACtC,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAE5B,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,oBAAoB,EAAE,CAAC;QAC3C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,oBAAoB,CACtC,cAAc,EACd,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,EAClB,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,EAClB,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC,CACrB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,KAAK,IAAI,EAAE;QACzD,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,gBAAgB,EAAE,CAAC;QAClD,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAC5B,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;QAEvD,MAAM,OAAO,GAAG,UAAU,CAAC,cAAc,CAAC,CAAC;QAC3C,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC;YAC3B,KAAK,EAAE,WAAW;YAClB,OAAO,EAAE,IAAI;SACd,CAAC,CAAC;QAEH,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;YACrB,OAAO,EAAE;gBACP,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE;aAC9D;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,KAAK,IAAI,EAAE;QACzD,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,gBAAgB,EAAE,CAAC;QAClD,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAC5B,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;QAEvD,MAAM,OAAO,GAAG,UAAU,CAAC,cAAc,CAAC,CAAC;QAC3C,MAAM,OAAO,CAAC;YACZ,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE,EAAE;YACT,KAAK,EAAE,CAAC;YACR,OAAO,EAAE,IAAI;SACd,CAAC,CAAC;QAEH,MAAM,CAAC,WAAW,CAAC,CAAC,oBAAoB,CACtC,MAAM,CAAC,gBAAgB,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CACrE,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0BAA0B,EAAE,KAAK,IAAI,EAAE;QACxC,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,gBAAgB,EAAE,CAAC;QAClD,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAC5B,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,iBAAiB,CACtC,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAChC,CAAC;QAEF,MAAM,OAAO,GAAG,UAAU,CAAC,cAAc,CAAC,CAAC;QAC3C,MAAM,MAAM,GAAG,CAAC,MAAM,OAAO,CAAC;YAC5B,KAAK,EAAE,WAAW;YAClB,OAAO,EAAE,IAAI;SACd,CAAC,CAAuD,CAAC;QAE1D,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClC,MAAM,IAAI,GAAI,MAAM,CAAC,OAAO,CAAC,CAAC,CAAsB,CAAC,IAAI,CAAC;QAC1D,MAAM,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,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#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 @@
|
|
|
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"}
|