@lhremote/mcp 0.1.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.
- package/README.md +58 -0
- package/dist/helpers.d.ts +59 -0
- package/dist/helpers.d.ts.map +1 -0
- package/dist/helpers.js +90 -0
- package/dist/helpers.js.map +1 -0
- package/dist/index.js +2 -0
- package/dist/index.js.map +1 -1
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +2 -0
- package/dist/server.js.map +1 -1
- package/dist/server.test.js +29 -2
- package/dist/server.test.js.map +1 -1
- package/dist/stdio.d.ts.map +1 -1
- package/dist/stdio.js +5 -2
- package/dist/stdio.js.map +1 -1
- package/dist/tools/campaign-add-action.d.ts +4 -0
- package/dist/tools/campaign-add-action.d.ts.map +1 -0
- package/dist/tools/campaign-add-action.js +63 -0
- package/dist/tools/campaign-add-action.js.map +1 -0
- package/dist/tools/campaign-add-action.test.d.ts +2 -0
- package/dist/tools/campaign-add-action.test.d.ts.map +1 -0
- package/dist/tools/campaign-add-action.test.js +107 -0
- package/dist/tools/campaign-add-action.test.js.map +1 -0
- package/dist/tools/campaign-create.d.ts +4 -0
- package/dist/tools/campaign-create.d.ts.map +1 -0
- package/dist/tools/campaign-create.js +44 -0
- package/dist/tools/campaign-create.js.map +1 -0
- package/dist/tools/campaign-create.test.d.ts +2 -0
- package/dist/tools/campaign-create.test.d.ts.map +1 -0
- package/dist/tools/campaign-create.test.js +163 -0
- package/dist/tools/campaign-create.test.js.map +1 -0
- package/dist/tools/campaign-delete.d.ts +4 -0
- package/dist/tools/campaign-delete.d.ts.map +1 -0
- package/dist/tools/campaign-delete.js +28 -0
- package/dist/tools/campaign-delete.js.map +1 -0
- package/dist/tools/campaign-delete.test.d.ts +2 -0
- package/dist/tools/campaign-delete.test.d.ts.map +1 -0
- package/dist/tools/campaign-delete.test.js +87 -0
- package/dist/tools/campaign-delete.test.js.map +1 -0
- package/dist/tools/campaign-exclude-add.d.ts +4 -0
- package/dist/tools/campaign-exclude-add.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-add.js +44 -0
- package/dist/tools/campaign-exclude-add.js.map +1 -0
- package/dist/tools/campaign-exclude-add.test.d.ts +2 -0
- package/dist/tools/campaign-exclude-add.test.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-add.test.js +161 -0
- package/dist/tools/campaign-exclude-add.test.js.map +1 -0
- package/dist/tools/campaign-exclude-list.d.ts +4 -0
- package/dist/tools/campaign-exclude-list.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-list.js +40 -0
- package/dist/tools/campaign-exclude-list.js.map +1 -0
- package/dist/tools/campaign-exclude-list.test.d.ts +2 -0
- package/dist/tools/campaign-exclude-list.test.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-list.test.js +183 -0
- package/dist/tools/campaign-exclude-list.test.js.map +1 -0
- package/dist/tools/campaign-exclude-remove.d.ts +4 -0
- package/dist/tools/campaign-exclude-remove.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-remove.js +44 -0
- package/dist/tools/campaign-exclude-remove.js.map +1 -0
- package/dist/tools/campaign-exclude-remove.test.d.ts +2 -0
- package/dist/tools/campaign-exclude-remove.test.d.ts.map +1 -0
- package/dist/tools/campaign-exclude-remove.test.js +161 -0
- package/dist/tools/campaign-exclude-remove.test.js.map +1 -0
- package/dist/tools/campaign-export.d.ts +4 -0
- package/dist/tools/campaign-export.d.ts.map +1 -0
- package/dist/tools/campaign-export.js +30 -0
- package/dist/tools/campaign-export.js.map +1 -0
- package/dist/tools/campaign-export.test.d.ts +2 -0
- package/dist/tools/campaign-export.test.d.ts.map +1 -0
- package/dist/tools/campaign-export.test.js +80 -0
- package/dist/tools/campaign-export.test.js.map +1 -0
- package/dist/tools/campaign-get.d.ts +4 -0
- package/dist/tools/campaign-get.d.ts.map +1 -0
- package/dist/tools/campaign-get.js +25 -0
- package/dist/tools/campaign-get.js.map +1 -0
- package/dist/tools/campaign-get.test.d.ts +2 -0
- package/dist/tools/campaign-get.test.d.ts.map +1 -0
- package/dist/tools/campaign-get.test.js +90 -0
- package/dist/tools/campaign-get.test.js.map +1 -0
- package/dist/tools/campaign-list.d.ts +4 -0
- package/dist/tools/campaign-list.d.ts.map +1 -0
- package/dist/tools/campaign-list.js +25 -0
- package/dist/tools/campaign-list.js.map +1 -0
- package/dist/tools/campaign-list.test.d.ts +2 -0
- package/dist/tools/campaign-list.test.d.ts.map +1 -0
- package/dist/tools/campaign-list.test.js +90 -0
- package/dist/tools/campaign-list.test.js.map +1 -0
- package/dist/tools/campaign-move-next.d.ts +4 -0
- package/dist/tools/campaign-move-next.d.ts.map +1 -0
- package/dist/tools/campaign-move-next.js +40 -0
- package/dist/tools/campaign-move-next.js.map +1 -0
- package/dist/tools/campaign-move-next.test.d.ts +2 -0
- package/dist/tools/campaign-move-next.test.d.ts.map +1 -0
- package/dist/tools/campaign-move-next.test.js +138 -0
- package/dist/tools/campaign-move-next.test.js.map +1 -0
- package/dist/tools/campaign-remove-action.d.ts +4 -0
- package/dist/tools/campaign-remove-action.d.ts.map +1 -0
- package/dist/tools/campaign-remove-action.js +36 -0
- package/dist/tools/campaign-remove-action.js.map +1 -0
- package/dist/tools/campaign-remove-action.test.d.ts +2 -0
- package/dist/tools/campaign-remove-action.test.d.ts.map +1 -0
- package/dist/tools/campaign-remove-action.test.js +130 -0
- package/dist/tools/campaign-remove-action.test.js.map +1 -0
- package/dist/tools/campaign-reorder-actions.d.ts +4 -0
- package/dist/tools/campaign-reorder-actions.d.ts.map +1 -0
- package/dist/tools/campaign-reorder-actions.js +35 -0
- package/dist/tools/campaign-reorder-actions.js.map +1 -0
- package/dist/tools/campaign-reorder-actions.test.d.ts +2 -0
- package/dist/tools/campaign-reorder-actions.test.d.ts.map +1 -0
- package/dist/tools/campaign-reorder-actions.test.js +146 -0
- package/dist/tools/campaign-reorder-actions.test.js.map +1 -0
- package/dist/tools/campaign-retry.d.ts +4 -0
- package/dist/tools/campaign-retry.d.ts.map +1 -0
- package/dist/tools/campaign-retry.js +29 -0
- package/dist/tools/campaign-retry.js.map +1 -0
- package/dist/tools/campaign-retry.test.d.ts +2 -0
- package/dist/tools/campaign-retry.test.d.ts.map +1 -0
- package/dist/tools/campaign-retry.test.js +91 -0
- package/dist/tools/campaign-retry.test.js.map +1 -0
- package/dist/tools/campaign-start.d.ts +4 -0
- package/dist/tools/campaign-start.d.ts.map +1 -0
- package/dist/tools/campaign-start.js +35 -0
- package/dist/tools/campaign-start.js.map +1 -0
- package/dist/tools/campaign-start.test.d.ts +2 -0
- package/dist/tools/campaign-start.test.d.ts.map +1 -0
- package/dist/tools/campaign-start.test.js +115 -0
- package/dist/tools/campaign-start.test.js.map +1 -0
- package/dist/tools/campaign-statistics.d.ts +4 -0
- package/dist/tools/campaign-statistics.d.ts.map +1 -0
- package/dist/tools/campaign-statistics.js +41 -0
- package/dist/tools/campaign-statistics.js.map +1 -0
- package/dist/tools/campaign-statistics.test.d.ts +2 -0
- package/dist/tools/campaign-statistics.test.d.ts.map +1 -0
- package/dist/tools/campaign-statistics.test.js +177 -0
- package/dist/tools/campaign-statistics.test.js.map +1 -0
- package/dist/tools/campaign-status.d.ts +4 -0
- package/dist/tools/campaign-status.d.ts.map +1 -0
- package/dist/tools/campaign-status.js +47 -0
- package/dist/tools/campaign-status.js.map +1 -0
- package/dist/tools/campaign-status.test.d.ts +2 -0
- package/dist/tools/campaign-status.test.d.ts.map +1 -0
- package/dist/tools/campaign-status.test.js +275 -0
- package/dist/tools/campaign-status.test.js.map +1 -0
- package/dist/tools/campaign-stop.d.ts +4 -0
- package/dist/tools/campaign-stop.d.ts.map +1 -0
- package/dist/tools/campaign-stop.js +28 -0
- package/dist/tools/campaign-stop.js.map +1 -0
- package/dist/tools/campaign-stop.test.d.ts +2 -0
- package/dist/tools/campaign-stop.test.d.ts.map +1 -0
- package/dist/tools/campaign-stop.test.js +91 -0
- package/dist/tools/campaign-stop.test.js.map +1 -0
- package/dist/tools/campaign-update.d.ts +4 -0
- package/dist/tools/campaign-update.d.ts.map +1 -0
- package/dist/tools/campaign-update.js +43 -0
- package/dist/tools/campaign-update.js.map +1 -0
- package/dist/tools/campaign-update.test.d.ts +2 -0
- package/dist/tools/campaign-update.test.d.ts.map +1 -0
- package/dist/tools/campaign-update.test.js +117 -0
- package/dist/tools/campaign-update.test.js.map +1 -0
- package/dist/tools/check-replies.d.ts +4 -0
- package/dist/tools/check-replies.d.ts.map +1 -0
- package/dist/tools/check-replies.js +24 -0
- package/dist/tools/check-replies.js.map +1 -0
- package/dist/tools/check-replies.test.d.ts +2 -0
- package/dist/tools/check-replies.test.d.ts.map +1 -0
- package/dist/tools/check-replies.test.js +192 -0
- package/dist/tools/check-replies.test.js.map +1 -0
- package/dist/tools/check-status.d.ts +1 -0
- package/dist/tools/check-status.d.ts.map +1 -1
- package/dist/tools/check-status.js +9 -25
- package/dist/tools/check-status.js.map +1 -1
- package/dist/tools/check-status.test.js +3 -1
- package/dist/tools/check-status.test.js.map +1 -1
- package/dist/tools/describe-actions.d.ts +4 -0
- package/dist/tools/describe-actions.d.ts.map +1 -0
- package/dist/tools/describe-actions.js +30 -0
- package/dist/tools/describe-actions.js.map +1 -0
- package/dist/tools/describe-actions.test.d.ts +2 -0
- package/dist/tools/describe-actions.test.d.ts.map +1 -0
- package/dist/tools/describe-actions.test.js +120 -0
- package/dist/tools/describe-actions.test.js.map +1 -0
- package/dist/tools/find-app.d.ts +4 -0
- package/dist/tools/find-app.d.ts.map +1 -0
- package/dist/tools/find-app.js +20 -0
- package/dist/tools/find-app.js.map +1 -0
- package/dist/tools/find-app.test.d.ts +2 -0
- package/dist/tools/find-app.test.d.ts.map +1 -0
- package/dist/tools/find-app.test.js +65 -0
- package/dist/tools/find-app.test.js.map +1 -0
- package/dist/tools/import-people-from-urls.d.ts +4 -0
- package/dist/tools/import-people-from-urls.d.ts.map +1 -0
- package/dist/tools/import-people-from-urls.js +32 -0
- package/dist/tools/import-people-from-urls.js.map +1 -0
- package/dist/tools/import-people-from-urls.test.d.ts +2 -0
- package/dist/tools/import-people-from-urls.test.d.ts.map +1 -0
- package/dist/tools/import-people-from-urls.test.js +136 -0
- package/dist/tools/import-people-from-urls.test.js.map +1 -0
- package/dist/tools/index.d.ts +33 -0
- package/dist/tools/index.d.ts.map +1 -1
- package/dist/tools/index.js +55 -2
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/index.test.d.ts +2 -0
- package/dist/tools/index.test.d.ts.map +1 -0
- package/dist/tools/index.test.js +22 -0
- package/dist/tools/index.test.js.map +1 -0
- package/dist/tools/launch-app.d.ts +1 -0
- package/dist/tools/launch-app.d.ts.map +1 -1
- package/dist/tools/launch-app.js +7 -19
- package/dist/tools/launch-app.js.map +1 -1
- package/dist/tools/launch-app.test.js +2 -0
- package/dist/tools/launch-app.test.js.map +1 -1
- package/dist/tools/list-accounts.d.ts +1 -0
- package/dist/tools/list-accounts.d.ts.map +1 -1
- package/dist/tools/list-accounts.js +11 -44
- package/dist/tools/list-accounts.js.map +1 -1
- package/dist/tools/list-accounts.test.js +3 -1
- package/dist/tools/list-accounts.test.js.map +1 -1
- package/dist/tools/query-messages.d.ts +4 -0
- package/dist/tools/query-messages.d.ts.map +1 -0
- package/dist/tools/query-messages.integration.test.d.ts +2 -0
- package/dist/tools/query-messages.integration.test.d.ts.map +1 -0
- package/dist/tools/query-messages.integration.test.js +119 -0
- package/dist/tools/query-messages.integration.test.js.map +1 -0
- package/dist/tools/query-messages.js +55 -0
- package/dist/tools/query-messages.js.map +1 -0
- package/dist/tools/query-messages.test.d.ts +2 -0
- package/dist/tools/query-messages.test.d.ts.map +1 -0
- package/dist/tools/query-messages.test.js +211 -0
- package/dist/tools/query-messages.test.js.map +1 -0
- package/dist/tools/query-profile.d.ts +4 -0
- package/dist/tools/query-profile.d.ts.map +1 -0
- package/dist/tools/query-profile.js +49 -0
- package/dist/tools/query-profile.js.map +1 -0
- package/dist/tools/query-profile.test.d.ts +2 -0
- package/dist/tools/query-profile.test.d.ts.map +1 -0
- package/dist/tools/query-profile.test.js +275 -0
- package/dist/tools/query-profile.test.js.map +1 -0
- package/dist/tools/query-profiles.d.ts +4 -0
- package/dist/tools/query-profiles.d.ts.map +1 -0
- package/dist/tools/query-profiles.js +67 -0
- package/dist/tools/query-profiles.js.map +1 -0
- package/dist/tools/query-profiles.test.d.ts +2 -0
- package/dist/tools/query-profiles.test.d.ts.map +1 -0
- package/dist/tools/query-profiles.test.js +239 -0
- package/dist/tools/query-profiles.test.js.map +1 -0
- package/dist/tools/quit-app.d.ts +1 -0
- package/dist/tools/quit-app.d.ts.map +1 -1
- package/dist/tools/quit-app.js +9 -13
- package/dist/tools/quit-app.js.map +1 -1
- package/dist/tools/quit-app.test.js +2 -0
- package/dist/tools/quit-app.test.js.map +1 -1
- package/dist/tools/scrape-messaging-history.d.ts +4 -0
- package/dist/tools/scrape-messaging-history.d.ts.map +1 -0
- package/dist/tools/scrape-messaging-history.js +19 -0
- package/dist/tools/scrape-messaging-history.js.map +1 -0
- package/dist/tools/scrape-messaging-history.test.d.ts +2 -0
- package/dist/tools/scrape-messaging-history.test.d.ts.map +1 -0
- package/dist/tools/scrape-messaging-history.test.js +114 -0
- package/dist/tools/scrape-messaging-history.test.js.map +1 -0
- package/dist/tools/start-instance.d.ts +1 -0
- package/dist/tools/start-instance.d.ts.map +1 -1
- package/dist/tools/start-instance.js +15 -77
- package/dist/tools/start-instance.js.map +1 -1
- package/dist/tools/start-instance.test.js +3 -1
- package/dist/tools/start-instance.test.js.map +1 -1
- package/dist/tools/stop-instance.d.ts +1 -0
- package/dist/tools/stop-instance.d.ts.map +1 -1
- package/dist/tools/stop-instance.js +13 -67
- package/dist/tools/stop-instance.js.map +1 -1
- package/dist/tools/stop-instance.test.js +3 -1
- package/dist/tools/stop-instance.test.js.map +1 -1
- package/dist/tools/testing/infrastructure-errors.d.ts +18 -0
- package/dist/tools/testing/infrastructure-errors.d.ts.map +1 -0
- package/dist/tools/testing/infrastructure-errors.js +59 -0
- package/dist/tools/testing/infrastructure-errors.js.map +1 -0
- package/dist/tools/testing/mock-server.d.ts +4 -0
- package/dist/tools/testing/mock-server.d.ts.map +1 -1
- package/dist/tools/testing/mock-server.js +19 -5
- package/dist/tools/testing/mock-server.js.map +1 -1
- package/package.json +14 -6
- package/dist/tools/visit-and-extract.d.ts +0 -3
- package/dist/tools/visit-and-extract.d.ts.map +0 -1
- package/dist/tools/visit-and-extract.js +0 -183
- package/dist/tools/visit-and-extract.js.map +0 -1
- package/dist/tools/visit-and-extract.test.d.ts +0 -2
- package/dist/tools/visit-and-extract.test.d.ts.map +0 -1
- package/dist/tools/visit-and-extract.test.js +0 -399
- package/dist/tools/visit-and-extract.test.js.map +0 -1
|
@@ -0,0 +1,90 @@
|
|
|
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
|
+
campaignGet: vi.fn(),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
import { CampaignNotFoundError, campaignGet, } from "@lhremote/core";
|
|
12
|
+
import { registerCampaignGet } from "./campaign-get.js";
|
|
13
|
+
import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
|
|
14
|
+
import { createMockServer } from "./testing/mock-server.js";
|
|
15
|
+
const MOCK_CAMPAIGN = {
|
|
16
|
+
id: 15,
|
|
17
|
+
name: "Outreach Campaign",
|
|
18
|
+
description: "Connect with engineering leaders",
|
|
19
|
+
state: "active",
|
|
20
|
+
liAccountId: 1,
|
|
21
|
+
isPaused: true,
|
|
22
|
+
isArchived: false,
|
|
23
|
+
isValid: true,
|
|
24
|
+
createdAt: "2026-02-07T10:00:00Z",
|
|
25
|
+
};
|
|
26
|
+
const MOCK_ACTIONS = [
|
|
27
|
+
{
|
|
28
|
+
id: 86,
|
|
29
|
+
campaignId: 15,
|
|
30
|
+
name: "Visit Profile",
|
|
31
|
+
description: null,
|
|
32
|
+
config: {
|
|
33
|
+
id: 100,
|
|
34
|
+
actionType: "VisitAndExtract",
|
|
35
|
+
actionSettings: { extractCurrentOrganizations: true },
|
|
36
|
+
coolDown: 60000,
|
|
37
|
+
maxActionResultsPerIteration: 10,
|
|
38
|
+
isDraft: false,
|
|
39
|
+
},
|
|
40
|
+
versionId: 1,
|
|
41
|
+
},
|
|
42
|
+
];
|
|
43
|
+
describe("registerCampaignGet", () => {
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
vi.clearAllMocks();
|
|
46
|
+
});
|
|
47
|
+
afterEach(() => {
|
|
48
|
+
vi.restoreAllMocks();
|
|
49
|
+
});
|
|
50
|
+
it("registers a tool named campaign-get", () => {
|
|
51
|
+
const { server } = createMockServer();
|
|
52
|
+
registerCampaignGet(server);
|
|
53
|
+
expect(server.tool).toHaveBeenCalledOnce();
|
|
54
|
+
expect(server.tool).toHaveBeenCalledWith("campaign-get", expect.any(String), expect.any(Object), expect.any(Function));
|
|
55
|
+
});
|
|
56
|
+
it("returns campaign details with actions", async () => {
|
|
57
|
+
const { server, getHandler } = createMockServer();
|
|
58
|
+
registerCampaignGet(server);
|
|
59
|
+
const resultData = { ...MOCK_CAMPAIGN, actions: MOCK_ACTIONS };
|
|
60
|
+
vi.mocked(campaignGet).mockResolvedValue(resultData);
|
|
61
|
+
const handler = getHandler("campaign-get");
|
|
62
|
+
const result = await handler({ campaignId: 15, cdpPort: 9222 });
|
|
63
|
+
expect(result).toEqual({
|
|
64
|
+
content: [
|
|
65
|
+
{
|
|
66
|
+
type: "text",
|
|
67
|
+
text: JSON.stringify(resultData, null, 2),
|
|
68
|
+
},
|
|
69
|
+
],
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
it("returns error for non-existent campaign", async () => {
|
|
73
|
+
const { server, getHandler } = createMockServer();
|
|
74
|
+
registerCampaignGet(server);
|
|
75
|
+
vi.mocked(campaignGet).mockRejectedValue(new CampaignNotFoundError(999));
|
|
76
|
+
const handler = getHandler("campaign-get");
|
|
77
|
+
const result = await handler({ campaignId: 999, cdpPort: 9222 });
|
|
78
|
+
expect(result).toEqual({
|
|
79
|
+
isError: true,
|
|
80
|
+
content: [
|
|
81
|
+
{
|
|
82
|
+
type: "text",
|
|
83
|
+
text: "Campaign 999 not found.",
|
|
84
|
+
},
|
|
85
|
+
],
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
describeInfrastructureErrors(registerCampaignGet, "campaign-get", () => ({ campaignId: 15, cdpPort: 9222 }), (error) => vi.mocked(campaignGet).mockRejectedValue(error), "Failed to get campaign");
|
|
89
|
+
});
|
|
90
|
+
//# sourceMappingURL=campaign-get.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-get.test.js","sourceRoot":"","sources":["../../src/tools/campaign-get.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#campaign-list | campaign-list} MCP tool. */
|
|
3
|
+
export declare function registerCampaignList(server: McpServer): void;
|
|
4
|
+
//# sourceMappingURL=campaign-list.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-list.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-list.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAOzE,6GAA6G;AAC7G,wBAAgB,oBAAoB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAqB5D"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { campaignList, } 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#campaign-list | campaign-list} MCP tool. */
|
|
7
|
+
export function registerCampaignList(server) {
|
|
8
|
+
server.tool("campaign-list", "List existing LinkedHelper campaigns with summary statistics", {
|
|
9
|
+
includeArchived: z
|
|
10
|
+
.boolean()
|
|
11
|
+
.optional()
|
|
12
|
+
.default(false)
|
|
13
|
+
.describe("Include archived campaigns"),
|
|
14
|
+
...cdpConnectionSchema,
|
|
15
|
+
}, async ({ includeArchived, cdpPort, cdpHost, allowRemote }) => {
|
|
16
|
+
try {
|
|
17
|
+
const result = await campaignList({ includeArchived, cdpPort, cdpHost, allowRemote });
|
|
18
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
19
|
+
}
|
|
20
|
+
catch (error) {
|
|
21
|
+
return mcpCatchAll(error, "Failed to list campaigns");
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=campaign-list.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-list.js","sourceRoot":"","sources":["../../src/tools/campaign-list.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,YAAY,GACb,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE7E,6GAA6G;AAC7G,MAAM,UAAU,oBAAoB,CAAC,MAAiB;IACpD,MAAM,CAAC,IAAI,CACT,eAAe,EACf,8DAA8D,EAC9D;QACE,eAAe,EAAE,CAAC;aACf,OAAO,EAAE;aACT,QAAQ,EAAE;aACV,OAAO,CAAC,KAAK,CAAC;aACd,QAAQ,CAAC,4BAA4B,CAAC;QACzC,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,eAAe,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAC3D,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,YAAY,CAAC,EAAE,eAAe,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;YACtF,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,0BAA0B,CAAC,CAAC;QACxD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-list.test.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-list.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,90 @@
|
|
|
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
|
+
campaignList: vi.fn(),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
import { campaignList, } from "@lhremote/core";
|
|
12
|
+
import { registerCampaignList } from "./campaign-list.js";
|
|
13
|
+
import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
|
|
14
|
+
import { createMockServer } from "./testing/mock-server.js";
|
|
15
|
+
const MOCK_CAMPAIGNS = [
|
|
16
|
+
{
|
|
17
|
+
id: 15,
|
|
18
|
+
name: "Outreach Campaign",
|
|
19
|
+
description: "Connect with engineering leaders",
|
|
20
|
+
state: "active",
|
|
21
|
+
liAccountId: 1,
|
|
22
|
+
actionCount: 2,
|
|
23
|
+
createdAt: "2026-02-07T10:00:00Z",
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
id: 16,
|
|
27
|
+
name: "Follow-up Campaign",
|
|
28
|
+
description: null,
|
|
29
|
+
state: "paused",
|
|
30
|
+
liAccountId: 1,
|
|
31
|
+
actionCount: 1,
|
|
32
|
+
createdAt: "2026-02-08T10:00:00Z",
|
|
33
|
+
},
|
|
34
|
+
];
|
|
35
|
+
describe("registerCampaignList", () => {
|
|
36
|
+
beforeEach(() => {
|
|
37
|
+
vi.clearAllMocks();
|
|
38
|
+
});
|
|
39
|
+
afterEach(() => {
|
|
40
|
+
vi.restoreAllMocks();
|
|
41
|
+
});
|
|
42
|
+
it("registers a tool named campaign-list", () => {
|
|
43
|
+
const { server } = createMockServer();
|
|
44
|
+
registerCampaignList(server);
|
|
45
|
+
expect(server.tool).toHaveBeenCalledOnce();
|
|
46
|
+
expect(server.tool).toHaveBeenCalledWith("campaign-list", expect.any(String), expect.any(Object), expect.any(Function));
|
|
47
|
+
});
|
|
48
|
+
it("returns list of campaigns", async () => {
|
|
49
|
+
const { server, getHandler } = createMockServer();
|
|
50
|
+
registerCampaignList(server);
|
|
51
|
+
const resultData = { campaigns: MOCK_CAMPAIGNS, total: 2 };
|
|
52
|
+
vi.mocked(campaignList).mockResolvedValue(resultData);
|
|
53
|
+
const handler = getHandler("campaign-list");
|
|
54
|
+
const result = await handler({ includeArchived: false, cdpPort: 9222 });
|
|
55
|
+
expect(result).toEqual({
|
|
56
|
+
content: [
|
|
57
|
+
{
|
|
58
|
+
type: "text",
|
|
59
|
+
text: JSON.stringify(resultData, null, 2),
|
|
60
|
+
},
|
|
61
|
+
],
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
it("returns empty list when no campaigns", async () => {
|
|
65
|
+
const { server, getHandler } = createMockServer();
|
|
66
|
+
registerCampaignList(server);
|
|
67
|
+
const resultData = { campaigns: [], total: 0 };
|
|
68
|
+
vi.mocked(campaignList).mockResolvedValue(resultData);
|
|
69
|
+
const handler = getHandler("campaign-list");
|
|
70
|
+
const result = await handler({ includeArchived: false, cdpPort: 9222 });
|
|
71
|
+
expect(result).toEqual({
|
|
72
|
+
content: [
|
|
73
|
+
{
|
|
74
|
+
type: "text",
|
|
75
|
+
text: JSON.stringify(resultData, null, 2),
|
|
76
|
+
},
|
|
77
|
+
],
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
it("passes includeArchived option to operation", async () => {
|
|
81
|
+
const { server, getHandler } = createMockServer();
|
|
82
|
+
registerCampaignList(server);
|
|
83
|
+
vi.mocked(campaignList).mockResolvedValue({ campaigns: [], total: 0 });
|
|
84
|
+
const handler = getHandler("campaign-list");
|
|
85
|
+
await handler({ includeArchived: true, cdpPort: 9222 });
|
|
86
|
+
expect(campaignList).toHaveBeenCalledWith(expect.objectContaining({ includeArchived: true }));
|
|
87
|
+
});
|
|
88
|
+
describeInfrastructureErrors(registerCampaignList, "campaign-list", () => ({ includeArchived: false, cdpPort: 9222 }), (error) => vi.mocked(campaignList).mockRejectedValue(error), "Failed to list campaigns");
|
|
89
|
+
});
|
|
90
|
+
//# sourceMappingURL=campaign-list.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-list.test.js","sourceRoot":"","sources":["../../src/tools/campaign-list.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#campaign-move-next | campaign-move-next} MCP tool. */
|
|
3
|
+
export declare function registerCampaignMoveNext(server: McpServer): void;
|
|
4
|
+
//# sourceMappingURL=campaign-move-next.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-move-next.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-move-next.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AASzE,uHAAuH;AACvH,wBAAgB,wBAAwB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAoChE"}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { ActionNotFoundError, NoNextActionError, campaignMoveNext, } 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-move-next | campaign-move-next} MCP tool. */
|
|
7
|
+
export function registerCampaignMoveNext(server) {
|
|
8
|
+
server.tool("campaign-move-next", "Move people from one action to the next action in a campaign chain (without executing the current action)", {
|
|
9
|
+
campaignId: z
|
|
10
|
+
.number()
|
|
11
|
+
.int()
|
|
12
|
+
.positive()
|
|
13
|
+
.describe("Campaign ID"),
|
|
14
|
+
actionId: z
|
|
15
|
+
.number()
|
|
16
|
+
.int()
|
|
17
|
+
.positive()
|
|
18
|
+
.describe("Action ID to move people from"),
|
|
19
|
+
personIds: z
|
|
20
|
+
.array(z.number().int().positive())
|
|
21
|
+
.nonempty()
|
|
22
|
+
.describe("Person IDs to advance to the next action"),
|
|
23
|
+
...cdpConnectionSchema,
|
|
24
|
+
}, async ({ campaignId, actionId, personIds, cdpPort, cdpHost, allowRemote }) => {
|
|
25
|
+
try {
|
|
26
|
+
const result = await campaignMoveNext({ campaignId, actionId, personIds, cdpPort, cdpHost, allowRemote });
|
|
27
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
28
|
+
}
|
|
29
|
+
catch (error) {
|
|
30
|
+
if (error instanceof ActionNotFoundError) {
|
|
31
|
+
return mcpError(`Action ${String(actionId)} not found in campaign ${String(campaignId)}.`);
|
|
32
|
+
}
|
|
33
|
+
if (error instanceof NoNextActionError) {
|
|
34
|
+
return mcpError(`Action ${String(actionId)} is the last action in campaign ${String(campaignId)}.`);
|
|
35
|
+
}
|
|
36
|
+
return mcpCatchAll(error, "Failed to move persons to next action");
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
//# sourceMappingURL=campaign-move-next.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-move-next.js","sourceRoot":"","sources":["../../src/tools/campaign-move-next.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,mBAAmB,EACnB,iBAAiB,EACjB,gBAAgB,GACjB,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,uHAAuH;AACvH,MAAM,UAAU,wBAAwB,CAAC,MAAiB;IACxD,MAAM,CAAC,IAAI,CACT,oBAAoB,EACpB,2GAA2G,EAC3G;QACE,UAAU,EAAE,CAAC;aACV,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,aAAa,CAAC;QAC1B,QAAQ,EAAE,CAAC;aACR,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,+BAA+B,CAAC;QAC5C,SAAS,EAAE,CAAC;aACT,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,CAAC;aAClC,QAAQ,EAAE;aACV,QAAQ,CAAC,0CAA0C,CAAC;QACvD,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,UAAU,EAAE,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAC3E,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,gBAAgB,CAAC,EAAE,UAAU,EAAE,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;YAC1G,OAAO,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,KAAK,YAAY,mBAAmB,EAAE,CAAC;gBACzC,OAAO,QAAQ,CAAC,UAAU,MAAM,CAAC,QAAQ,CAAC,0BAA0B,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YAC7F,CAAC;YACD,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACvC,OAAO,QAAQ,CAAC,UAAU,MAAM,CAAC,QAAQ,CAAC,mCAAmC,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YACtG,CAAC;YACD,OAAO,WAAW,CAAC,KAAK,EAAE,uCAAuC,CAAC,CAAC;QACrE,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-move-next.test.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-move-next.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,138 @@
|
|
|
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
|
+
campaignMoveNext: vi.fn(),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
import { ActionNotFoundError, CampaignNotFoundError, NoNextActionError, campaignMoveNext, } from "@lhremote/core";
|
|
12
|
+
import { registerCampaignMoveNext } from "./campaign-move-next.js";
|
|
13
|
+
import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
|
|
14
|
+
import { createMockServer } from "./testing/mock-server.js";
|
|
15
|
+
const MOVE_RESULT = {
|
|
16
|
+
success: true,
|
|
17
|
+
campaignId: 10,
|
|
18
|
+
fromActionId: 5,
|
|
19
|
+
toActionId: 6,
|
|
20
|
+
personsMoved: 2,
|
|
21
|
+
};
|
|
22
|
+
describe("registerCampaignMoveNext", () => {
|
|
23
|
+
beforeEach(() => {
|
|
24
|
+
vi.clearAllMocks();
|
|
25
|
+
});
|
|
26
|
+
afterEach(() => {
|
|
27
|
+
vi.restoreAllMocks();
|
|
28
|
+
});
|
|
29
|
+
it("registers a tool named campaign-move-next", () => {
|
|
30
|
+
const { server } = createMockServer();
|
|
31
|
+
registerCampaignMoveNext(server);
|
|
32
|
+
expect(server.tool).toHaveBeenCalledOnce();
|
|
33
|
+
expect(server.tool).toHaveBeenCalledWith("campaign-move-next", expect.any(String), expect.any(Object), expect.any(Function));
|
|
34
|
+
});
|
|
35
|
+
it("successfully moves persons to next action", async () => {
|
|
36
|
+
const { server, getHandler } = createMockServer();
|
|
37
|
+
registerCampaignMoveNext(server);
|
|
38
|
+
vi.mocked(campaignMoveNext).mockResolvedValue(MOVE_RESULT);
|
|
39
|
+
const handler = getHandler("campaign-move-next");
|
|
40
|
+
const result = await handler({
|
|
41
|
+
campaignId: 10,
|
|
42
|
+
actionId: 5,
|
|
43
|
+
personIds: [100, 200],
|
|
44
|
+
cdpPort: 9222,
|
|
45
|
+
});
|
|
46
|
+
expect(result).toEqual({
|
|
47
|
+
content: [
|
|
48
|
+
{
|
|
49
|
+
type: "text",
|
|
50
|
+
text: JSON.stringify(MOVE_RESULT, null, 2),
|
|
51
|
+
},
|
|
52
|
+
],
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
it("calls campaignMoveNext with correct arguments", async () => {
|
|
56
|
+
const { server, getHandler } = createMockServer();
|
|
57
|
+
registerCampaignMoveNext(server);
|
|
58
|
+
vi.mocked(campaignMoveNext).mockResolvedValue(MOVE_RESULT);
|
|
59
|
+
const handler = getHandler("campaign-move-next");
|
|
60
|
+
await handler({
|
|
61
|
+
campaignId: 10,
|
|
62
|
+
actionId: 5,
|
|
63
|
+
personIds: [100, 200],
|
|
64
|
+
cdpPort: 9222,
|
|
65
|
+
});
|
|
66
|
+
expect(campaignMoveNext).toHaveBeenCalledWith(expect.objectContaining({
|
|
67
|
+
campaignId: 10,
|
|
68
|
+
actionId: 5,
|
|
69
|
+
personIds: [100, 200],
|
|
70
|
+
cdpPort: 9222,
|
|
71
|
+
}));
|
|
72
|
+
});
|
|
73
|
+
it("returns error for non-existent campaign", async () => {
|
|
74
|
+
const { server, getHandler } = createMockServer();
|
|
75
|
+
registerCampaignMoveNext(server);
|
|
76
|
+
vi.mocked(campaignMoveNext).mockRejectedValue(new CampaignNotFoundError(999));
|
|
77
|
+
const handler = getHandler("campaign-move-next");
|
|
78
|
+
const result = await handler({
|
|
79
|
+
campaignId: 999,
|
|
80
|
+
actionId: 5,
|
|
81
|
+
personIds: [100],
|
|
82
|
+
cdpPort: 9222,
|
|
83
|
+
});
|
|
84
|
+
expect(result).toEqual({
|
|
85
|
+
isError: true,
|
|
86
|
+
content: [
|
|
87
|
+
{
|
|
88
|
+
type: "text",
|
|
89
|
+
text: "Campaign 999 not found.",
|
|
90
|
+
},
|
|
91
|
+
],
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
it("returns error for non-existent action", async () => {
|
|
95
|
+
const { server, getHandler } = createMockServer();
|
|
96
|
+
registerCampaignMoveNext(server);
|
|
97
|
+
vi.mocked(campaignMoveNext).mockRejectedValue(new ActionNotFoundError(999, 10));
|
|
98
|
+
const handler = getHandler("campaign-move-next");
|
|
99
|
+
const result = await handler({
|
|
100
|
+
campaignId: 10,
|
|
101
|
+
actionId: 999,
|
|
102
|
+
personIds: [100],
|
|
103
|
+
cdpPort: 9222,
|
|
104
|
+
});
|
|
105
|
+
expect(result).toEqual({
|
|
106
|
+
isError: true,
|
|
107
|
+
content: [
|
|
108
|
+
{
|
|
109
|
+
type: "text",
|
|
110
|
+
text: "Action 999 not found in campaign 10.",
|
|
111
|
+
},
|
|
112
|
+
],
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
it("returns error for last action in chain", async () => {
|
|
116
|
+
const { server, getHandler } = createMockServer();
|
|
117
|
+
registerCampaignMoveNext(server);
|
|
118
|
+
vi.mocked(campaignMoveNext).mockRejectedValue(new NoNextActionError(7, 10));
|
|
119
|
+
const handler = getHandler("campaign-move-next");
|
|
120
|
+
const result = await handler({
|
|
121
|
+
campaignId: 10,
|
|
122
|
+
actionId: 7,
|
|
123
|
+
personIds: [100],
|
|
124
|
+
cdpPort: 9222,
|
|
125
|
+
});
|
|
126
|
+
expect(result).toEqual({
|
|
127
|
+
isError: true,
|
|
128
|
+
content: [
|
|
129
|
+
{
|
|
130
|
+
type: "text",
|
|
131
|
+
text: "Action 7 is the last action in campaign 10.",
|
|
132
|
+
},
|
|
133
|
+
],
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
describeInfrastructureErrors(registerCampaignMoveNext, "campaign-move-next", () => ({ campaignId: 10, actionId: 5, personIds: [100], cdpPort: 9222 }), (error) => vi.mocked(campaignMoveNext).mockRejectedValue(error));
|
|
137
|
+
});
|
|
138
|
+
//# sourceMappingURL=campaign-move-next.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-move-next.test.js","sourceRoot":"","sources":["../../src/tools/campaign-move-next.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#campaign-remove-action | campaign-remove-action} MCP tool. */
|
|
3
|
+
export declare function registerCampaignRemoveAction(server: McpServer): void;
|
|
4
|
+
//# sourceMappingURL=campaign-remove-action.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-remove-action.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-remove-action.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AASzE,+HAA+H;AAC/H,wBAAgB,4BAA4B,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAgCpE"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { ActionNotFoundError, CampaignExecutionError, campaignRemoveAction, } 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-remove-action | campaign-remove-action} MCP tool. */
|
|
7
|
+
export function registerCampaignRemoveAction(server) {
|
|
8
|
+
server.tool("campaign-remove-action", "Remove an action from a campaign's action chain", {
|
|
9
|
+
campaignId: z
|
|
10
|
+
.number()
|
|
11
|
+
.int()
|
|
12
|
+
.positive()
|
|
13
|
+
.describe("Campaign ID"),
|
|
14
|
+
actionId: z
|
|
15
|
+
.number()
|
|
16
|
+
.int()
|
|
17
|
+
.positive()
|
|
18
|
+
.describe("Action ID to remove"),
|
|
19
|
+
...cdpConnectionSchema,
|
|
20
|
+
}, async ({ campaignId, actionId, cdpPort, cdpHost, allowRemote }) => {
|
|
21
|
+
try {
|
|
22
|
+
const result = await campaignRemoveAction({ campaignId, actionId, cdpPort, cdpHost, allowRemote });
|
|
23
|
+
return mcpSuccess(JSON.stringify(result, null, 2));
|
|
24
|
+
}
|
|
25
|
+
catch (error) {
|
|
26
|
+
if (error instanceof ActionNotFoundError) {
|
|
27
|
+
return mcpError(`Action ${String(actionId)} not found in campaign ${String(campaignId)}.`);
|
|
28
|
+
}
|
|
29
|
+
if (error instanceof CampaignExecutionError) {
|
|
30
|
+
return mcpError(`Failed to remove action: ${error.message}`);
|
|
31
|
+
}
|
|
32
|
+
return mcpCatchAll(error, "Failed to remove action");
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=campaign-remove-action.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-remove-action.js","sourceRoot":"","sources":["../../src/tools/campaign-remove-action.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAGpC,OAAO,EACL,mBAAmB,EACnB,sBAAsB,EACtB,oBAAoB,GACrB,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,+HAA+H;AAC/H,MAAM,UAAU,4BAA4B,CAAC,MAAiB;IAC5D,MAAM,CAAC,IAAI,CACT,wBAAwB,EACxB,iDAAiD,EACjD;QACE,UAAU,EAAE,CAAC;aACV,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,aAAa,CAAC;QAC1B,QAAQ,EAAE,CAAC;aACR,MAAM,EAAE;aACR,GAAG,EAAE;aACL,QAAQ,EAAE;aACV,QAAQ,CAAC,qBAAqB,CAAC;QAClC,GAAG,mBAAmB;KACvB,EACD,KAAK,EAAE,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;QAChE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,oBAAoB,CAAC,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;YACnG,OAAO,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,KAAK,YAAY,mBAAmB,EAAE,CAAC;gBACzC,OAAO,QAAQ,CAAC,UAAU,MAAM,CAAC,QAAQ,CAAC,0BAA0B,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YAC7F,CAAC;YACD,IAAI,KAAK,YAAY,sBAAsB,EAAE,CAAC;gBAC5C,OAAO,QAAQ,CAAC,4BAA4B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YAC/D,CAAC;YACD,OAAO,WAAW,CAAC,KAAK,EAAE,yBAAyB,CAAC,CAAC;QACvD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign-remove-action.test.d.ts","sourceRoot":"","sources":["../../src/tools/campaign-remove-action.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,130 @@
|
|
|
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
|
+
campaignRemoveAction: vi.fn(),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
import { ActionNotFoundError, CampaignExecutionError, CampaignNotFoundError, InstanceNotRunningError, campaignRemoveAction, } from "@lhremote/core";
|
|
12
|
+
import { registerCampaignRemoveAction } from "./campaign-remove-action.js";
|
|
13
|
+
import { describeInfrastructureErrors } from "./testing/infrastructure-errors.js";
|
|
14
|
+
import { createMockServer } from "./testing/mock-server.js";
|
|
15
|
+
const REMOVE_RESULT = { success: true, campaignId: 15, removedActionId: 50 };
|
|
16
|
+
describe("registerCampaignRemoveAction", () => {
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
vi.clearAllMocks();
|
|
19
|
+
});
|
|
20
|
+
afterEach(() => {
|
|
21
|
+
vi.restoreAllMocks();
|
|
22
|
+
});
|
|
23
|
+
it("registers a tool named campaign-remove-action", () => {
|
|
24
|
+
const { server } = createMockServer();
|
|
25
|
+
registerCampaignRemoveAction(server);
|
|
26
|
+
expect(server.tool).toHaveBeenCalledOnce();
|
|
27
|
+
expect(server.tool).toHaveBeenCalledWith("campaign-remove-action", expect.any(String), expect.any(Object), expect.any(Function));
|
|
28
|
+
});
|
|
29
|
+
it("successfully removes action", async () => {
|
|
30
|
+
const { server, getHandler } = createMockServer();
|
|
31
|
+
registerCampaignRemoveAction(server);
|
|
32
|
+
vi.mocked(campaignRemoveAction).mockResolvedValue(REMOVE_RESULT);
|
|
33
|
+
const handler = getHandler("campaign-remove-action");
|
|
34
|
+
const result = await handler({
|
|
35
|
+
campaignId: 15,
|
|
36
|
+
actionId: 50,
|
|
37
|
+
cdpPort: 9222,
|
|
38
|
+
});
|
|
39
|
+
expect(result).toEqual({
|
|
40
|
+
content: [
|
|
41
|
+
{
|
|
42
|
+
type: "text",
|
|
43
|
+
text: JSON.stringify(REMOVE_RESULT, null, 2),
|
|
44
|
+
},
|
|
45
|
+
],
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
it("returns error for non-existent campaign", async () => {
|
|
49
|
+
const { server, getHandler } = createMockServer();
|
|
50
|
+
registerCampaignRemoveAction(server);
|
|
51
|
+
vi.mocked(campaignRemoveAction).mockRejectedValue(new CampaignNotFoundError(999));
|
|
52
|
+
const handler = getHandler("campaign-remove-action");
|
|
53
|
+
const result = await handler({
|
|
54
|
+
campaignId: 999,
|
|
55
|
+
actionId: 50,
|
|
56
|
+
cdpPort: 9222,
|
|
57
|
+
});
|
|
58
|
+
expect(result).toEqual({
|
|
59
|
+
isError: true,
|
|
60
|
+
content: [
|
|
61
|
+
{
|
|
62
|
+
type: "text",
|
|
63
|
+
text: "Campaign 999 not found.",
|
|
64
|
+
},
|
|
65
|
+
],
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
it("returns error for non-existent action", async () => {
|
|
69
|
+
const { server, getHandler } = createMockServer();
|
|
70
|
+
registerCampaignRemoveAction(server);
|
|
71
|
+
vi.mocked(campaignRemoveAction).mockRejectedValue(new ActionNotFoundError(999, 15));
|
|
72
|
+
const handler = getHandler("campaign-remove-action");
|
|
73
|
+
const result = await handler({
|
|
74
|
+
campaignId: 15,
|
|
75
|
+
actionId: 999,
|
|
76
|
+
cdpPort: 9222,
|
|
77
|
+
});
|
|
78
|
+
expect(result).toEqual({
|
|
79
|
+
isError: true,
|
|
80
|
+
content: [
|
|
81
|
+
{
|
|
82
|
+
type: "text",
|
|
83
|
+
text: "Action 999 not found in campaign 15.",
|
|
84
|
+
},
|
|
85
|
+
],
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
describeInfrastructureErrors(registerCampaignRemoveAction, "campaign-remove-action", () => ({ campaignId: 15, actionId: 50, cdpPort: 9222 }), (error) => vi.mocked(campaignRemoveAction).mockRejectedValue(error));
|
|
89
|
+
it("returns error when instance is not running", async () => {
|
|
90
|
+
const { server, getHandler } = createMockServer();
|
|
91
|
+
registerCampaignRemoveAction(server);
|
|
92
|
+
vi.mocked(campaignRemoveAction).mockRejectedValue(new InstanceNotRunningError("Instance not running"));
|
|
93
|
+
const handler = getHandler("campaign-remove-action");
|
|
94
|
+
const result = await handler({
|
|
95
|
+
campaignId: 15,
|
|
96
|
+
actionId: 50,
|
|
97
|
+
cdpPort: 9222,
|
|
98
|
+
});
|
|
99
|
+
expect(result).toEqual({
|
|
100
|
+
isError: true,
|
|
101
|
+
content: [
|
|
102
|
+
{
|
|
103
|
+
type: "text",
|
|
104
|
+
text: "Failed to remove action: Instance not running",
|
|
105
|
+
},
|
|
106
|
+
],
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
it("returns error when CDP call fails", async () => {
|
|
110
|
+
const { server, getHandler } = createMockServer();
|
|
111
|
+
registerCampaignRemoveAction(server);
|
|
112
|
+
vi.mocked(campaignRemoveAction).mockRejectedValue(new CampaignExecutionError("Failed to remove action 50 from campaign 15: UI error", 15));
|
|
113
|
+
const handler = getHandler("campaign-remove-action");
|
|
114
|
+
const result = await handler({
|
|
115
|
+
campaignId: 15,
|
|
116
|
+
actionId: 50,
|
|
117
|
+
cdpPort: 9222,
|
|
118
|
+
});
|
|
119
|
+
expect(result).toEqual({
|
|
120
|
+
isError: true,
|
|
121
|
+
content: [
|
|
122
|
+
{
|
|
123
|
+
type: "text",
|
|
124
|
+
text: "Failed to remove action: Failed to remove action 50 from campaign 15: UI error",
|
|
125
|
+
},
|
|
126
|
+
],
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
//# sourceMappingURL=campaign-remove-action.test.js.map
|