@lhremote/cli 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/handlers/campaign-delete.d.ts +1 -0
- package/dist/handlers/campaign-delete.d.ts.map +1 -1
- package/dist/handlers/campaign-delete.js +3 -1
- package/dist/handlers/campaign-delete.js.map +1 -1
- package/dist/handlers/campaign-delete.test.js +25 -0
- package/dist/handlers/campaign-delete.test.js.map +1 -1
- package/dist/handlers/campaign-erase.d.ts +8 -0
- package/dist/handlers/campaign-erase.d.ts.map +1 -0
- package/dist/handlers/campaign-erase.js +39 -0
- package/dist/handlers/campaign-erase.js.map +1 -0
- package/dist/handlers/campaign-erase.test.d.ts +2 -0
- package/dist/handlers/campaign-erase.test.d.ts.map +1 -0
- package/dist/handlers/campaign-erase.test.js +71 -0
- package/dist/handlers/campaign-erase.test.js.map +1 -0
- package/dist/handlers/comment-on-post.d.ts +10 -0
- package/dist/handlers/comment-on-post.d.ts.map +1 -0
- package/dist/handlers/comment-on-post.js +37 -0
- package/dist/handlers/comment-on-post.js.map +1 -0
- package/dist/handlers/dismiss-errors.d.ts +8 -0
- package/dist/handlers/dismiss-errors.d.ts.map +1 -0
- package/dist/handlers/dismiss-errors.js +28 -0
- package/dist/handlers/dismiss-errors.js.map +1 -0
- package/dist/handlers/dismiss-errors.test.d.ts +2 -0
- package/dist/handlers/dismiss-errors.test.d.ts.map +1 -0
- package/dist/handlers/dismiss-errors.test.js +90 -0
- package/dist/handlers/dismiss-errors.test.js.map +1 -0
- package/dist/handlers/endorse-skills.d.ts +14 -0
- package/dist/handlers/endorse-skills.d.ts.map +1 -0
- package/dist/handlers/endorse-skills.js +44 -0
- package/dist/handlers/endorse-skills.js.map +1 -0
- package/dist/handlers/endorse-skills.test.d.ts +2 -0
- package/dist/handlers/endorse-skills.test.d.ts.map +1 -0
- package/dist/handlers/endorse-skills.test.js +65 -0
- package/dist/handlers/endorse-skills.test.js.map +1 -0
- package/dist/handlers/enrich-profile.d.ts +16 -0
- package/dist/handlers/enrich-profile.d.ts.map +1 -0
- package/dist/handlers/enrich-profile.js +51 -0
- package/dist/handlers/enrich-profile.js.map +1 -0
- package/dist/handlers/enrich-profile.test.d.ts +2 -0
- package/dist/handlers/enrich-profile.test.d.ts.map +1 -0
- package/dist/handlers/enrich-profile.test.js +65 -0
- package/dist/handlers/enrich-profile.test.js.map +1 -0
- package/dist/handlers/follow-person.d.ts +13 -0
- package/dist/handlers/follow-person.d.ts.map +1 -0
- package/dist/handlers/follow-person.js +45 -0
- package/dist/handlers/follow-person.js.map +1 -0
- package/dist/handlers/follow-person.test.d.ts +2 -0
- package/dist/handlers/follow-person.test.d.ts.map +1 -0
- package/dist/handlers/follow-person.test.js +65 -0
- package/dist/handlers/follow-person.test.js.map +1 -0
- package/dist/handlers/get-action-budget.d.ts +8 -0
- package/dist/handlers/get-action-budget.d.ts.map +1 -0
- package/dist/handlers/get-action-budget.js +46 -0
- package/dist/handlers/get-action-budget.js.map +1 -0
- package/dist/handlers/get-action-budget.test.d.ts +2 -0
- package/dist/handlers/get-action-budget.test.d.ts.map +1 -0
- package/dist/handlers/get-action-budget.test.js +69 -0
- package/dist/handlers/get-action-budget.test.js.map +1 -0
- package/dist/handlers/get-errors.d.ts.map +1 -1
- package/dist/handlers/get-errors.js +12 -0
- package/dist/handlers/get-errors.js.map +1 -1
- package/dist/handlers/get-errors.test.js +25 -0
- package/dist/handlers/get-errors.test.js.map +1 -1
- package/dist/handlers/get-feed.d.ts +10 -0
- package/dist/handlers/get-feed.d.ts.map +1 -0
- package/dist/handlers/get-feed.js +62 -0
- package/dist/handlers/get-feed.js.map +1 -0
- package/dist/handlers/get-feed.test.d.ts +2 -0
- package/dist/handlers/get-feed.test.d.ts.map +1 -0
- package/dist/handlers/get-feed.test.js +126 -0
- package/dist/handlers/get-feed.test.js.map +1 -0
- package/dist/handlers/get-post-stats.d.ts +8 -0
- package/dist/handlers/get-post-stats.d.ts.map +1 -0
- package/dist/handlers/get-post-stats.js +37 -0
- package/dist/handlers/get-post-stats.js.map +1 -0
- package/dist/handlers/get-post.d.ts +10 -0
- package/dist/handlers/get-post.d.ts.map +1 -0
- package/dist/handlers/get-post.js +66 -0
- package/dist/handlers/get-post.js.map +1 -0
- package/dist/handlers/get-profile-activity.d.ts +10 -0
- package/dist/handlers/get-profile-activity.d.ts.map +1 -0
- package/dist/handlers/get-profile-activity.js +56 -0
- package/dist/handlers/get-profile-activity.js.map +1 -0
- package/dist/handlers/get-throttle-status.d.ts +8 -0
- package/dist/handlers/get-throttle-status.d.ts.map +1 -0
- package/dist/handlers/get-throttle-status.js +33 -0
- package/dist/handlers/get-throttle-status.js.map +1 -0
- package/dist/handlers/get-throttle-status.test.d.ts +2 -0
- package/dist/handlers/get-throttle-status.test.d.ts.map +1 -0
- package/dist/handlers/get-throttle-status.test.js +65 -0
- package/dist/handlers/get-throttle-status.test.js.map +1 -0
- package/dist/handlers/index.d.ts +20 -0
- package/dist/handlers/index.d.ts.map +1 -1
- package/dist/handlers/index.js +20 -0
- package/dist/handlers/index.js.map +1 -1
- package/dist/handlers/like-person-posts.d.ts +18 -0
- package/dist/handlers/like-person-posts.d.ts.map +1 -0
- package/dist/handlers/like-person-posts.js +59 -0
- package/dist/handlers/like-person-posts.js.map +1 -0
- package/dist/handlers/like-person-posts.test.d.ts +2 -0
- package/dist/handlers/like-person-posts.test.d.ts.map +1 -0
- package/dist/handlers/like-person-posts.test.js +71 -0
- package/dist/handlers/like-person-posts.test.js.map +1 -0
- package/dist/handlers/message-person.d.ts +15 -0
- package/dist/handlers/message-person.d.ts.map +1 -0
- package/dist/handlers/message-person.js +65 -0
- package/dist/handlers/message-person.js.map +1 -0
- package/dist/handlers/message-person.test.d.ts +2 -0
- package/dist/handlers/message-person.test.d.ts.map +1 -0
- package/dist/handlers/message-person.test.js +77 -0
- package/dist/handlers/message-person.test.js.map +1 -0
- package/dist/handlers/query-messages.integration.test.js +24 -3
- package/dist/handlers/query-messages.integration.test.js.map +1 -1
- package/dist/handlers/react-to-post.d.ts +9 -0
- package/dist/handlers/react-to-post.d.ts.map +1 -0
- package/dist/handlers/react-to-post.js +30 -0
- package/dist/handlers/react-to-post.js.map +1 -0
- package/dist/handlers/remove-connection.d.ts +11 -0
- package/dist/handlers/remove-connection.d.ts.map +1 -0
- package/dist/handlers/remove-connection.js +41 -0
- package/dist/handlers/remove-connection.js.map +1 -0
- package/dist/handlers/remove-connection.test.d.ts +2 -0
- package/dist/handlers/remove-connection.test.d.ts.map +1 -0
- package/dist/handlers/remove-connection.test.js +65 -0
- package/dist/handlers/remove-connection.test.js.map +1 -0
- package/dist/handlers/search-posts.d.ts +10 -0
- package/dist/handlers/search-posts.d.ts.map +1 -0
- package/dist/handlers/search-posts.js +61 -0
- package/dist/handlers/search-posts.js.map +1 -0
- package/dist/handlers/search-posts.test.d.ts +2 -0
- package/dist/handlers/search-posts.test.d.ts.map +1 -0
- package/dist/handlers/search-posts.test.js +105 -0
- package/dist/handlers/search-posts.test.js.map +1 -0
- package/dist/handlers/send-inmail.d.ts +15 -0
- package/dist/handlers/send-inmail.d.ts.map +1 -0
- package/dist/handlers/send-inmail.js +65 -0
- package/dist/handlers/send-inmail.js.map +1 -0
- package/dist/handlers/send-inmail.test.d.ts +2 -0
- package/dist/handlers/send-inmail.test.d.ts.map +1 -0
- package/dist/handlers/send-inmail.test.js +77 -0
- package/dist/handlers/send-inmail.test.js.map +1 -0
- package/dist/handlers/send-invite.d.ts +13 -0
- package/dist/handlers/send-invite.d.ts.map +1 -0
- package/dist/handlers/send-invite.js +54 -0
- package/dist/handlers/send-invite.js.map +1 -0
- package/dist/handlers/send-invite.test.d.ts +2 -0
- package/dist/handlers/send-invite.test.d.ts.map +1 -0
- package/dist/handlers/send-invite.test.js +71 -0
- package/dist/handlers/send-invite.test.js.map +1 -0
- package/dist/handlers/visit-profile.d.ts +11 -0
- package/dist/handlers/visit-profile.d.ts.map +1 -0
- package/dist/handlers/visit-profile.js +93 -0
- package/dist/handlers/visit-profile.js.map +1 -0
- package/dist/handlers/visit-profile.test.d.ts +2 -0
- package/dist/handlers/visit-profile.test.d.ts.map +1 -0
- package/dist/handlers/visit-profile.test.js +169 -0
- package/dist/handlers/visit-profile.test.js.map +1 -0
- package/dist/program.d.ts.map +1 -1
- package/dist/program.js +244 -2
- package/dist/program.js.map +1 -1
- package/dist/program.test.js +21 -1
- package/dist/program.test.js.map +1 -1
- package/package.json +2 -2
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { DEFAULT_CDP_PORT, errorMessage, removeConnection, CampaignExecutionError, CampaignTimeoutError, } from "@lhremote/core";
|
|
4
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#remove-connection | remove-connection} CLI command. */
|
|
5
|
+
export async function handleRemoveConnection(options) {
|
|
6
|
+
if ((options.personId == null) === (options.url == null)) {
|
|
7
|
+
process.stderr.write("Exactly one of --person-id or --url must be provided.\n");
|
|
8
|
+
process.exitCode = 1;
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
process.stderr.write("Removing connection...\n");
|
|
12
|
+
let result;
|
|
13
|
+
try {
|
|
14
|
+
result = await removeConnection({
|
|
15
|
+
personId: options.personId,
|
|
16
|
+
url: options.url,
|
|
17
|
+
keepCampaign: options.keepCampaign,
|
|
18
|
+
cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
|
|
19
|
+
cdpHost: options.cdpHost,
|
|
20
|
+
allowRemote: options.allowRemote,
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
catch (error) {
|
|
24
|
+
if (error instanceof CampaignExecutionError || error instanceof CampaignTimeoutError) {
|
|
25
|
+
process.stderr.write(`${error.message}\n`);
|
|
26
|
+
}
|
|
27
|
+
else {
|
|
28
|
+
process.stderr.write(`${errorMessage(error)}\n`);
|
|
29
|
+
}
|
|
30
|
+
process.exitCode = 1;
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
process.stderr.write("Done.\n");
|
|
34
|
+
if (options.json) {
|
|
35
|
+
process.stdout.write(JSON.stringify(result, null, 2) + "\n");
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
process.stdout.write(`Remove connection ${result.success ? "succeeded" : "failed"} (person #${String(result.personId)})\n`);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=remove-connection.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.js","sourceRoot":"","sources":["../../src/handlers/remove-connection.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,gBAAgB,EAChB,YAAY,EACZ,gBAAgB,EAEhB,sBAAsB,EACtB,oBAAoB,GACrB,MAAM,gBAAgB,CAAC;AAExB,sHAAsH;AACtH,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAAC,OAQ5C;IACC,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;QACzD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;QAChF,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;IAEjD,IAAI,MAA6B,CAAC;IAClC,IAAI,CAAC;QACH,MAAM,GAAG,MAAM,gBAAgB,CAAC;YAC9B,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,YAAY,EAAE,OAAO,CAAC,YAAY;YAClC,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,gBAAgB;YAC5C,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,WAAW,EAAE,OAAO,CAAC,WAAW;SACjC,CAAC,CAAC;IACL,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,sBAAsB,IAAI,KAAK,YAAY,oBAAoB,EAAE,CAAC;YACrF,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,OAAO,IAAI,CAAC,CAAC;QAC7C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACnD,CAAC;QACD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAEhC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;IAC/D,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,qBAAqB,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,QAAQ,aAAa,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9H,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.test.d.ts","sourceRoot":"","sources":["../../src/handlers/remove-connection.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,65 @@
|
|
|
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 { CampaignExecutionError, CampaignTimeoutError, removeConnection, } from "@lhremote/core";
|
|
12
|
+
import { handleRemoveConnection } from "./remove-connection.js";
|
|
13
|
+
import { getStderr, getStdout } from "./testing/mock-helpers.js";
|
|
14
|
+
const MOCK_RESULT = {
|
|
15
|
+
success: true,
|
|
16
|
+
personId: 100,
|
|
17
|
+
results: [{ id: 1, actionVersionId: 1, personId: 100, result: 1, platform: null, createdAt: "2026-01-01T00:00:00Z", profile: null }],
|
|
18
|
+
};
|
|
19
|
+
describe("handleRemoveConnection", () => {
|
|
20
|
+
let stdoutSpy;
|
|
21
|
+
let stderrSpy;
|
|
22
|
+
beforeEach(() => {
|
|
23
|
+
vi.clearAllMocks();
|
|
24
|
+
process.exitCode = undefined;
|
|
25
|
+
stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
|
|
26
|
+
stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
|
|
27
|
+
});
|
|
28
|
+
afterEach(() => {
|
|
29
|
+
vi.restoreAllMocks();
|
|
30
|
+
});
|
|
31
|
+
it("outputs JSON result on success", async () => {
|
|
32
|
+
vi.mocked(removeConnection).mockResolvedValue(MOCK_RESULT);
|
|
33
|
+
await handleRemoveConnection({ personId: 100, json: true });
|
|
34
|
+
expect(process.exitCode).toBeUndefined();
|
|
35
|
+
const stdout = getStdout(stdoutSpy);
|
|
36
|
+
const parsed = JSON.parse(stdout);
|
|
37
|
+
expect(parsed.success).toBe(true);
|
|
38
|
+
expect(parsed.personId).toBe(100);
|
|
39
|
+
});
|
|
40
|
+
it("outputs human-readable result on success", async () => {
|
|
41
|
+
vi.mocked(removeConnection).mockResolvedValue(MOCK_RESULT);
|
|
42
|
+
await handleRemoveConnection({ personId: 100 });
|
|
43
|
+
expect(process.exitCode).toBeUndefined();
|
|
44
|
+
expect(getStdout(stdoutSpy)).toContain("succeeded");
|
|
45
|
+
});
|
|
46
|
+
it("returns error when neither personId nor url provided", async () => {
|
|
47
|
+
await handleRemoveConnection({});
|
|
48
|
+
expect(process.exitCode).toBe(1);
|
|
49
|
+
expect(getStderr(stderrSpy)).toContain("Exactly one of --person-id or --url");
|
|
50
|
+
expect(removeConnection).not.toHaveBeenCalled();
|
|
51
|
+
});
|
|
52
|
+
it("handles CampaignExecutionError", async () => {
|
|
53
|
+
vi.mocked(removeConnection).mockRejectedValue(new CampaignExecutionError("Person 100 not found"));
|
|
54
|
+
await handleRemoveConnection({ personId: 100 });
|
|
55
|
+
expect(process.exitCode).toBe(1);
|
|
56
|
+
expect(getStderr(stderrSpy)).toContain("Person 100 not found");
|
|
57
|
+
});
|
|
58
|
+
it("handles CampaignTimeoutError", async () => {
|
|
59
|
+
vi.mocked(removeConnection).mockRejectedValue(new CampaignTimeoutError("Timed out", 42));
|
|
60
|
+
await handleRemoveConnection({ personId: 100 });
|
|
61
|
+
expect(process.exitCode).toBe(1);
|
|
62
|
+
expect(getStderr(stderrSpy)).toContain("Timed out");
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
//# sourceMappingURL=remove-connection.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove-connection.test.js","sourceRoot":"","sources":["../../src/handlers/remove-connection.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#search-posts | search-posts} CLI command. */
|
|
2
|
+
export declare function handleSearchPosts(query: string, options: {
|
|
3
|
+
start?: number;
|
|
4
|
+
count?: number;
|
|
5
|
+
cdpPort?: number;
|
|
6
|
+
cdpHost?: string;
|
|
7
|
+
allowRemote?: boolean;
|
|
8
|
+
json?: boolean;
|
|
9
|
+
}): Promise<void>;
|
|
10
|
+
//# sourceMappingURL=search-posts.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-posts.d.ts","sourceRoot":"","sources":["../../src/handlers/search-posts.ts"],"names":[],"mappings":"AAUA,4GAA4G;AAC5G,wBAAsB,iBAAiB,CACrC,KAAK,EAAE,MAAM,EACb,OAAO,EAAE;IACP,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB,GACA,OAAO,CAAC,IAAI,CAAC,CAgEf"}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { DEFAULT_CDP_PORT, errorMessage, searchPosts, } from "@lhremote/core";
|
|
4
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#search-posts | search-posts} CLI command. */
|
|
5
|
+
export async function handleSearchPosts(query, options) {
|
|
6
|
+
let result;
|
|
7
|
+
try {
|
|
8
|
+
result = await searchPosts({
|
|
9
|
+
query,
|
|
10
|
+
start: options.start,
|
|
11
|
+
count: options.count,
|
|
12
|
+
cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
|
|
13
|
+
cdpHost: options.cdpHost,
|
|
14
|
+
allowRemote: options.allowRemote,
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
catch (error) {
|
|
18
|
+
const message = errorMessage(error);
|
|
19
|
+
process.stderr.write(`${message}\n`);
|
|
20
|
+
process.exitCode = 1;
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
if (options.json) {
|
|
24
|
+
process.stdout.write(JSON.stringify(result, null, 2) + "\n");
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
const { posts, paging } = result;
|
|
28
|
+
process.stdout.write(`Search: "${result.query}" (${String(paging.total)} results)\n\n`);
|
|
29
|
+
if (posts.length === 0) {
|
|
30
|
+
process.stdout.write("No posts found.\n");
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
for (const post of posts) {
|
|
34
|
+
const author = [post.authorFirstName, post.authorLastName]
|
|
35
|
+
.filter(Boolean)
|
|
36
|
+
.join(" ") || "Unknown";
|
|
37
|
+
process.stdout.write(` ${post.postUrn}\n`);
|
|
38
|
+
process.stdout.write(` Author: ${author}`);
|
|
39
|
+
if (post.authorPublicId) {
|
|
40
|
+
process.stdout.write(` (${post.authorPublicId})`);
|
|
41
|
+
}
|
|
42
|
+
process.stdout.write("\n");
|
|
43
|
+
if (post.authorHeadline) {
|
|
44
|
+
process.stdout.write(` Headline: ${post.authorHeadline}\n`);
|
|
45
|
+
}
|
|
46
|
+
if (post.text) {
|
|
47
|
+
const preview = post.text.length > 120
|
|
48
|
+
? post.text.substring(0, 120) + "..."
|
|
49
|
+
: post.text;
|
|
50
|
+
process.stdout.write(` Text: ${preview}\n`);
|
|
51
|
+
}
|
|
52
|
+
process.stdout.write(` Reactions: ${String(post.reactionCount)} Comments: ${String(post.commentCount)}\n`);
|
|
53
|
+
process.stdout.write("\n");
|
|
54
|
+
}
|
|
55
|
+
if (paging.start + posts.length < paging.total) {
|
|
56
|
+
process.stdout.write(`Showing ${String(paging.start + 1)}-${String(paging.start + posts.length)} of ${String(paging.total)}. ` +
|
|
57
|
+
`Use --start ${String(paging.start + posts.length)} for next page.\n`);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
//# sourceMappingURL=search-posts.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-posts.js","sourceRoot":"","sources":["../../src/handlers/search-posts.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-posts.test.d.ts","sourceRoot":"","sources":["../../src/handlers/search-posts.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,105 @@
|
|
|
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
|
+
searchPosts: vi.fn(),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
import { searchPosts } from "@lhremote/core";
|
|
12
|
+
import { handleSearchPosts } from "./search-posts.js";
|
|
13
|
+
import { getStderr, getStdout } from "./testing/mock-helpers.js";
|
|
14
|
+
const MOCK_RESULTS = {
|
|
15
|
+
query: "AI agents",
|
|
16
|
+
posts: [
|
|
17
|
+
{
|
|
18
|
+
postUrn: "urn:li:activity:7123456789012345678",
|
|
19
|
+
text: "Excited about AI agents!",
|
|
20
|
+
authorFirstName: "Jane",
|
|
21
|
+
authorLastName: "Smith",
|
|
22
|
+
authorPublicId: "janesmith",
|
|
23
|
+
authorHeadline: "CEO at Acme Corp",
|
|
24
|
+
reactionCount: 42,
|
|
25
|
+
commentCount: 7,
|
|
26
|
+
},
|
|
27
|
+
{
|
|
28
|
+
postUrn: "urn:li:activity:7234567890123456789",
|
|
29
|
+
text: null,
|
|
30
|
+
authorFirstName: "Bob",
|
|
31
|
+
authorLastName: null,
|
|
32
|
+
authorPublicId: null,
|
|
33
|
+
authorHeadline: null,
|
|
34
|
+
reactionCount: 0,
|
|
35
|
+
commentCount: 0,
|
|
36
|
+
},
|
|
37
|
+
],
|
|
38
|
+
paging: { start: 0, count: 10, total: 42 },
|
|
39
|
+
};
|
|
40
|
+
describe("handleSearchPosts", () => {
|
|
41
|
+
const originalExitCode = process.exitCode;
|
|
42
|
+
let stdoutSpy;
|
|
43
|
+
let stderrSpy;
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
process.exitCode = undefined;
|
|
46
|
+
vi.clearAllMocks();
|
|
47
|
+
stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
|
|
48
|
+
stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
|
|
49
|
+
});
|
|
50
|
+
afterEach(() => {
|
|
51
|
+
process.exitCode = originalExitCode;
|
|
52
|
+
vi.restoreAllMocks();
|
|
53
|
+
});
|
|
54
|
+
it("prints JSON with --json", async () => {
|
|
55
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
56
|
+
await handleSearchPosts("AI agents", { json: true });
|
|
57
|
+
expect(process.exitCode).toBeUndefined();
|
|
58
|
+
const output = JSON.parse(getStdout(stdoutSpy));
|
|
59
|
+
expect(output.query).toBe("AI agents");
|
|
60
|
+
expect(output.posts).toHaveLength(2);
|
|
61
|
+
expect(output.paging.total).toBe(42);
|
|
62
|
+
});
|
|
63
|
+
it("prints human-readable output by default", async () => {
|
|
64
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
65
|
+
await handleSearchPosts("AI agents", {});
|
|
66
|
+
expect(process.exitCode).toBeUndefined();
|
|
67
|
+
const output = getStdout(stdoutSpy);
|
|
68
|
+
expect(output).toContain('"AI agents"');
|
|
69
|
+
expect(output).toContain("42 results");
|
|
70
|
+
expect(output).toContain("Jane Smith");
|
|
71
|
+
expect(output).toContain("janesmith");
|
|
72
|
+
expect(output).toContain("CEO at Acme Corp");
|
|
73
|
+
expect(output).toContain("Excited about AI agents!");
|
|
74
|
+
expect(output).toContain("Reactions: 42");
|
|
75
|
+
expect(output).toContain("Comments: 7");
|
|
76
|
+
});
|
|
77
|
+
it("shows pagination hint when more results available", async () => {
|
|
78
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
79
|
+
await handleSearchPosts("AI agents", {});
|
|
80
|
+
const output = getStdout(stdoutSpy);
|
|
81
|
+
expect(output).toContain("--start 2");
|
|
82
|
+
});
|
|
83
|
+
it("handles empty results", async () => {
|
|
84
|
+
vi.mocked(searchPosts).mockResolvedValue({
|
|
85
|
+
query: "nonexistent",
|
|
86
|
+
posts: [],
|
|
87
|
+
paging: { start: 0, count: 10, total: 0 },
|
|
88
|
+
});
|
|
89
|
+
await handleSearchPosts("nonexistent", {});
|
|
90
|
+
const output = getStdout(stdoutSpy);
|
|
91
|
+
expect(output).toContain("No posts found");
|
|
92
|
+
});
|
|
93
|
+
it("passes pagination options to operation", async () => {
|
|
94
|
+
vi.mocked(searchPosts).mockResolvedValue(MOCK_RESULTS);
|
|
95
|
+
await handleSearchPosts("AI agents", { start: 10, count: 5 });
|
|
96
|
+
expect(searchPosts).toHaveBeenCalledWith(expect.objectContaining({ query: "AI agents", start: 10, count: 5 }));
|
|
97
|
+
});
|
|
98
|
+
it("sets exitCode on error", async () => {
|
|
99
|
+
vi.mocked(searchPosts).mockRejectedValue(new Error("connection refused"));
|
|
100
|
+
await handleSearchPosts("AI agents", {});
|
|
101
|
+
expect(process.exitCode).toBe(1);
|
|
102
|
+
expect(getStderr(stderrSpy)).toContain("connection refused");
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
//# sourceMappingURL=search-posts.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-posts.test.js","sourceRoot":"","sources":["../../src/handlers/search-posts.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#send-inmail | send-inmail} CLI command. */
|
|
2
|
+
export declare function handleSendInmail(options: {
|
|
3
|
+
personId?: number;
|
|
4
|
+
url?: string;
|
|
5
|
+
messageTemplate: string;
|
|
6
|
+
subjectTemplate?: string;
|
|
7
|
+
rejectIfReplied?: boolean;
|
|
8
|
+
proceedOnOutOfCredits?: boolean;
|
|
9
|
+
keepCampaign?: boolean;
|
|
10
|
+
cdpPort?: number;
|
|
11
|
+
cdpHost?: string;
|
|
12
|
+
allowRemote?: boolean;
|
|
13
|
+
json?: boolean;
|
|
14
|
+
}): Promise<void>;
|
|
15
|
+
//# sourceMappingURL=send-inmail.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-inmail.d.ts","sourceRoot":"","sources":["../../src/handlers/send-inmail.ts"],"names":[],"mappings":"AAYA,0GAA0G;AAC1G,wBAAsB,gBAAgB,CAAC,OAAO,EAAE;IAC9C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,eAAe,EAAE,MAAM,CAAC;IACxB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,eAAe,CAAC,EAAE,OAAO,CAAC;IAC1B,qBAAqB,CAAC,EAAE,OAAO,CAAC;IAChC,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB,GAAG,OAAO,CAAC,IAAI,CAAC,CA4DhB"}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { DEFAULT_CDP_PORT, errorMessage, sendInmail, CampaignExecutionError, CampaignTimeoutError, } from "@lhremote/core";
|
|
4
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#send-inmail | send-inmail} CLI command. */
|
|
5
|
+
export async function handleSendInmail(options) {
|
|
6
|
+
if ((options.personId == null) === (options.url == null)) {
|
|
7
|
+
process.stderr.write("Exactly one of --person-id or --url must be provided.\n");
|
|
8
|
+
process.exitCode = 1;
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
let parsedMessageTemplate;
|
|
12
|
+
try {
|
|
13
|
+
parsedMessageTemplate = JSON.parse(options.messageTemplate);
|
|
14
|
+
}
|
|
15
|
+
catch {
|
|
16
|
+
process.stderr.write("Invalid JSON in --message-template.\n");
|
|
17
|
+
process.exitCode = 1;
|
|
18
|
+
return;
|
|
19
|
+
}
|
|
20
|
+
let parsedSubjectTemplate;
|
|
21
|
+
if (options.subjectTemplate) {
|
|
22
|
+
try {
|
|
23
|
+
parsedSubjectTemplate = JSON.parse(options.subjectTemplate);
|
|
24
|
+
}
|
|
25
|
+
catch {
|
|
26
|
+
process.stderr.write("Invalid JSON in --subject-template.\n");
|
|
27
|
+
process.exitCode = 1;
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
process.stderr.write("Sending InMail...\n");
|
|
32
|
+
let result;
|
|
33
|
+
try {
|
|
34
|
+
result = await sendInmail({
|
|
35
|
+
personId: options.personId,
|
|
36
|
+
url: options.url,
|
|
37
|
+
messageTemplate: parsedMessageTemplate,
|
|
38
|
+
subjectTemplate: parsedSubjectTemplate,
|
|
39
|
+
rejectIfReplied: options.rejectIfReplied,
|
|
40
|
+
proceedOnOutOfCredits: options.proceedOnOutOfCredits,
|
|
41
|
+
keepCampaign: options.keepCampaign,
|
|
42
|
+
cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
|
|
43
|
+
cdpHost: options.cdpHost,
|
|
44
|
+
allowRemote: options.allowRemote,
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
catch (error) {
|
|
48
|
+
if (error instanceof CampaignExecutionError || error instanceof CampaignTimeoutError) {
|
|
49
|
+
process.stderr.write(`${error.message}\n`);
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
process.stderr.write(`${errorMessage(error)}\n`);
|
|
53
|
+
}
|
|
54
|
+
process.exitCode = 1;
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
process.stderr.write("Done.\n");
|
|
58
|
+
if (options.json) {
|
|
59
|
+
process.stdout.write(JSON.stringify(result, null, 2) + "\n");
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
process.stdout.write(`InMail ${result.success ? "sent" : "failed"} (person #${String(result.personId)})\n`);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=send-inmail.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-inmail.js","sourceRoot":"","sources":["../../src/handlers/send-inmail.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-inmail.test.d.ts","sourceRoot":"","sources":["../../src/handlers/send-inmail.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,77 @@
|
|
|
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 { CampaignExecutionError, CampaignTimeoutError, sendInmail, } from "@lhremote/core";
|
|
12
|
+
import { handleSendInmail } from "./send-inmail.js";
|
|
13
|
+
import { getStderr, getStdout } from "./testing/mock-helpers.js";
|
|
14
|
+
const MOCK_RESULT = {
|
|
15
|
+
success: true,
|
|
16
|
+
personId: 100,
|
|
17
|
+
results: [{ id: 1, actionVersionId: 1, personId: 100, result: 1, platform: null, createdAt: "2026-01-01T00:00:00Z", profile: null }],
|
|
18
|
+
};
|
|
19
|
+
describe("handleSendInmail", () => {
|
|
20
|
+
let stdoutSpy;
|
|
21
|
+
let stderrSpy;
|
|
22
|
+
beforeEach(() => {
|
|
23
|
+
vi.clearAllMocks();
|
|
24
|
+
process.exitCode = undefined;
|
|
25
|
+
stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
|
|
26
|
+
stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
|
|
27
|
+
});
|
|
28
|
+
afterEach(() => {
|
|
29
|
+
vi.restoreAllMocks();
|
|
30
|
+
});
|
|
31
|
+
it("outputs JSON result on success", async () => {
|
|
32
|
+
vi.mocked(sendInmail).mockResolvedValue(MOCK_RESULT);
|
|
33
|
+
await handleSendInmail({ personId: 100, messageTemplate: '{"type":"text","value":"Hello"}', json: true });
|
|
34
|
+
expect(process.exitCode).toBeUndefined();
|
|
35
|
+
const stdout = getStdout(stdoutSpy);
|
|
36
|
+
const parsed = JSON.parse(stdout);
|
|
37
|
+
expect(parsed.success).toBe(true);
|
|
38
|
+
expect(parsed.personId).toBe(100);
|
|
39
|
+
});
|
|
40
|
+
it("outputs human-readable result on success", async () => {
|
|
41
|
+
vi.mocked(sendInmail).mockResolvedValue(MOCK_RESULT);
|
|
42
|
+
await handleSendInmail({ personId: 100, messageTemplate: '{"type":"text","value":"Hello"}' });
|
|
43
|
+
expect(process.exitCode).toBeUndefined();
|
|
44
|
+
expect(getStdout(stdoutSpy)).toContain("sent");
|
|
45
|
+
});
|
|
46
|
+
it("returns error when neither personId nor url provided", async () => {
|
|
47
|
+
await handleSendInmail({ messageTemplate: '{"type":"text","value":"Hello"}' });
|
|
48
|
+
expect(process.exitCode).toBe(1);
|
|
49
|
+
expect(getStderr(stderrSpy)).toContain("Exactly one of --person-id or --url");
|
|
50
|
+
expect(sendInmail).not.toHaveBeenCalled();
|
|
51
|
+
});
|
|
52
|
+
it("returns error for invalid JSON in messageTemplate", async () => {
|
|
53
|
+
await handleSendInmail({ personId: 100, messageTemplate: "not json" });
|
|
54
|
+
expect(process.exitCode).toBe(1);
|
|
55
|
+
expect(getStderr(stderrSpy)).toContain("Invalid JSON in --message-template");
|
|
56
|
+
expect(sendInmail).not.toHaveBeenCalled();
|
|
57
|
+
});
|
|
58
|
+
it("returns error for invalid JSON in subjectTemplate", async () => {
|
|
59
|
+
await handleSendInmail({ personId: 100, messageTemplate: '{"type":"text","value":"Hello"}', subjectTemplate: "not json" });
|
|
60
|
+
expect(process.exitCode).toBe(1);
|
|
61
|
+
expect(getStderr(stderrSpy)).toContain("Invalid JSON in --subject-template");
|
|
62
|
+
expect(sendInmail).not.toHaveBeenCalled();
|
|
63
|
+
});
|
|
64
|
+
it("handles CampaignExecutionError", async () => {
|
|
65
|
+
vi.mocked(sendInmail).mockRejectedValue(new CampaignExecutionError("Person 100 not found"));
|
|
66
|
+
await handleSendInmail({ personId: 100, messageTemplate: '{"type":"text","value":"Hello"}' });
|
|
67
|
+
expect(process.exitCode).toBe(1);
|
|
68
|
+
expect(getStderr(stderrSpy)).toContain("Person 100 not found");
|
|
69
|
+
});
|
|
70
|
+
it("handles CampaignTimeoutError", async () => {
|
|
71
|
+
vi.mocked(sendInmail).mockRejectedValue(new CampaignTimeoutError("Timed out", 42));
|
|
72
|
+
await handleSendInmail({ personId: 100, messageTemplate: '{"type":"text","value":"Hello"}' });
|
|
73
|
+
expect(process.exitCode).toBe(1);
|
|
74
|
+
expect(getStderr(stderrSpy)).toContain("Timed out");
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
//# sourceMappingURL=send-inmail.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-inmail.test.js","sourceRoot":"","sources":["../../src/handlers/send-inmail.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#send-invite | send-invite} CLI command. */
|
|
2
|
+
export declare function handleSendInvite(options: {
|
|
3
|
+
personId?: number;
|
|
4
|
+
url?: string;
|
|
5
|
+
messageTemplate?: string;
|
|
6
|
+
saveAsLeadSn?: boolean;
|
|
7
|
+
keepCampaign?: boolean;
|
|
8
|
+
cdpPort?: number;
|
|
9
|
+
cdpHost?: string;
|
|
10
|
+
allowRemote?: boolean;
|
|
11
|
+
json?: boolean;
|
|
12
|
+
}): Promise<void>;
|
|
13
|
+
//# sourceMappingURL=send-invite.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-invite.d.ts","sourceRoot":"","sources":["../../src/handlers/send-invite.ts"],"names":[],"mappings":"AAYA,0GAA0G;AAC1G,wBAAsB,gBAAgB,CAAC,OAAO,EAAE;IAC9C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB,GAAG,OAAO,CAAC,IAAI,CAAC,CAiDhB"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
2
|
+
// Copyright (C) 2026 Oleksii PELYKH
|
|
3
|
+
import { DEFAULT_CDP_PORT, errorMessage, sendInvite, CampaignExecutionError, CampaignTimeoutError, } from "@lhremote/core";
|
|
4
|
+
/** Handle the {@link https://github.com/alexey-pelykh/lhremote#send-invite | send-invite} CLI command. */
|
|
5
|
+
export async function handleSendInvite(options) {
|
|
6
|
+
if ((options.personId == null) === (options.url == null)) {
|
|
7
|
+
process.stderr.write("Exactly one of --person-id or --url must be provided.\n");
|
|
8
|
+
process.exitCode = 1;
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
let parsedMessageTemplate;
|
|
12
|
+
if (options.messageTemplate) {
|
|
13
|
+
try {
|
|
14
|
+
parsedMessageTemplate = JSON.parse(options.messageTemplate);
|
|
15
|
+
}
|
|
16
|
+
catch {
|
|
17
|
+
process.stderr.write("Invalid JSON in --message-template.\n");
|
|
18
|
+
process.exitCode = 1;
|
|
19
|
+
return;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
process.stderr.write("Sending invite...\n");
|
|
23
|
+
let result;
|
|
24
|
+
try {
|
|
25
|
+
result = await sendInvite({
|
|
26
|
+
personId: options.personId,
|
|
27
|
+
url: options.url,
|
|
28
|
+
messageTemplate: parsedMessageTemplate,
|
|
29
|
+
saveAsLeadSN: options.saveAsLeadSn,
|
|
30
|
+
keepCampaign: options.keepCampaign,
|
|
31
|
+
cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
|
|
32
|
+
cdpHost: options.cdpHost,
|
|
33
|
+
allowRemote: options.allowRemote,
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
catch (error) {
|
|
37
|
+
if (error instanceof CampaignExecutionError || error instanceof CampaignTimeoutError) {
|
|
38
|
+
process.stderr.write(`${error.message}\n`);
|
|
39
|
+
}
|
|
40
|
+
else {
|
|
41
|
+
process.stderr.write(`${errorMessage(error)}\n`);
|
|
42
|
+
}
|
|
43
|
+
process.exitCode = 1;
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
process.stderr.write("Done.\n");
|
|
47
|
+
if (options.json) {
|
|
48
|
+
process.stdout.write(JSON.stringify(result, null, 2) + "\n");
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
process.stdout.write(`Invite ${result.success ? "sent" : "failed"} (person #${String(result.personId)})\n`);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=send-invite.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-invite.js","sourceRoot":"","sources":["../../src/handlers/send-invite.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,gBAAgB,EAChB,YAAY,EACZ,UAAU,EAEV,sBAAsB,EACtB,oBAAoB,GACrB,MAAM,gBAAgB,CAAC;AAExB,0GAA0G;AAC1G,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,OAUtC;IACC,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;QACzD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;QAChF,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,qBAA0D,CAAC;IAC/D,IAAI,OAAO,CAAC,eAAe,EAAE,CAAC;QAC5B,IAAI,CAAC;YACH,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,eAAe,CAA4B,CAAC;QACzF,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,uCAAuC,CAAC,CAAC;YAC9D,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;IACH,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;IAE5C,IAAI,MAA6B,CAAC;IAClC,IAAI,CAAC;QACH,MAAM,GAAG,MAAM,UAAU,CAAC;YACxB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,eAAe,EAAE,qBAAqB;YACtC,YAAY,EAAE,OAAO,CAAC,YAAY;YAClC,YAAY,EAAE,OAAO,CAAC,YAAY;YAClC,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,gBAAgB;YAC5C,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,WAAW,EAAE,OAAO,CAAC,WAAW;SACjC,CAAC,CAAC;IACL,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,sBAAsB,IAAI,KAAK,YAAY,oBAAoB,EAAE,CAAC;YACrF,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,OAAO,IAAI,CAAC,CAAC;QAC7C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACnD,CAAC;QACD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAEhC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;IAC/D,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,aAAa,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9G,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"send-invite.test.d.ts","sourceRoot":"","sources":["../../src/handlers/send-invite.test.ts"],"names":[],"mappings":""}
|