@lhremote/cli 0.5.0 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (66) hide show
  1. package/dist/handlers/add-people-to-collection.d.ts +10 -0
  2. package/dist/handlers/add-people-to-collection.d.ts.map +1 -0
  3. package/dist/handlers/add-people-to-collection.js +79 -0
  4. package/dist/handlers/add-people-to-collection.js.map +1 -0
  5. package/dist/handlers/add-people-to-collection.test.d.ts +2 -0
  6. package/dist/handlers/add-people-to-collection.test.d.ts.map +1 -0
  7. package/dist/handlers/add-people-to-collection.test.js +107 -0
  8. package/dist/handlers/add-people-to-collection.test.js.map +1 -0
  9. package/dist/handlers/collect-people.d.ts +12 -0
  10. package/dist/handlers/collect-people.d.ts.map +1 -0
  11. package/dist/handlers/collect-people.js +39 -0
  12. package/dist/handlers/collect-people.js.map +1 -0
  13. package/dist/handlers/collect-people.test.d.ts +2 -0
  14. package/dist/handlers/collect-people.test.d.ts.map +1 -0
  15. package/dist/handlers/collect-people.test.js +92 -0
  16. package/dist/handlers/collect-people.test.js.map +1 -0
  17. package/dist/handlers/create-collection.d.ts +8 -0
  18. package/dist/handlers/create-collection.d.ts.map +1 -0
  19. package/dist/handlers/create-collection.js +26 -0
  20. package/dist/handlers/create-collection.js.map +1 -0
  21. package/dist/handlers/create-collection.test.d.ts +2 -0
  22. package/dist/handlers/create-collection.test.d.ts.map +1 -0
  23. package/dist/handlers/create-collection.test.js +55 -0
  24. package/dist/handlers/create-collection.test.js.map +1 -0
  25. package/dist/handlers/delete-collection.d.ts +8 -0
  26. package/dist/handlers/delete-collection.d.ts.map +1 -0
  27. package/dist/handlers/delete-collection.js +31 -0
  28. package/dist/handlers/delete-collection.js.map +1 -0
  29. package/dist/handlers/delete-collection.test.d.ts +2 -0
  30. package/dist/handlers/delete-collection.test.d.ts.map +1 -0
  31. package/dist/handlers/delete-collection.test.js +74 -0
  32. package/dist/handlers/delete-collection.test.js.map +1 -0
  33. package/dist/handlers/import-people-from-collection.d.ts +8 -0
  34. package/dist/handlers/import-people-from-collection.d.ts.map +1 -0
  35. package/dist/handlers/import-people-from-collection.js +55 -0
  36. package/dist/handlers/import-people-from-collection.js.map +1 -0
  37. package/dist/handlers/import-people-from-collection.test.d.ts +2 -0
  38. package/dist/handlers/import-people-from-collection.test.d.ts.map +1 -0
  39. package/dist/handlers/import-people-from-collection.test.js +121 -0
  40. package/dist/handlers/import-people-from-collection.test.js.map +1 -0
  41. package/dist/handlers/index.d.ts +7 -0
  42. package/dist/handlers/index.d.ts.map +1 -1
  43. package/dist/handlers/index.js +7 -0
  44. package/dist/handlers/index.js.map +1 -1
  45. package/dist/handlers/list-collections.d.ts +5 -0
  46. package/dist/handlers/list-collections.d.ts.map +1 -0
  47. package/dist/handlers/list-collections.js +48 -0
  48. package/dist/handlers/list-collections.js.map +1 -0
  49. package/dist/handlers/list-collections.test.d.ts +2 -0
  50. package/dist/handlers/list-collections.test.d.ts.map +1 -0
  51. package/dist/handlers/list-collections.test.js +111 -0
  52. package/dist/handlers/list-collections.test.js.map +1 -0
  53. package/dist/handlers/remove-people-from-collection.d.ts +10 -0
  54. package/dist/handlers/remove-people-from-collection.d.ts.map +1 -0
  55. package/dist/handlers/remove-people-from-collection.js +75 -0
  56. package/dist/handlers/remove-people-from-collection.js.map +1 -0
  57. package/dist/handlers/remove-people-from-collection.test.d.ts +2 -0
  58. package/dist/handlers/remove-people-from-collection.test.d.ts.map +1 -0
  59. package/dist/handlers/remove-people-from-collection.test.js +100 -0
  60. package/dist/handlers/remove-people-from-collection.test.js.map +1 -0
  61. package/dist/program.d.ts.map +1 -1
  62. package/dist/program.js +70 -1
  63. package/dist/program.js.map +1 -1
  64. package/dist/program.test.js +8 -1
  65. package/dist/program.test.js.map +1 -1
  66. package/package.json +2 -2
@@ -0,0 +1,10 @@
1
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#add-people-to-collection | add-people-to-collection} CLI command. */
2
+ export declare function handleAddPeopleToCollection(collectionId: number, options: {
3
+ personIds?: string;
4
+ personIdsFile?: string;
5
+ cdpPort?: number;
6
+ cdpHost?: string;
7
+ allowRemote?: boolean;
8
+ json?: boolean;
9
+ }): Promise<void>;
10
+ //# sourceMappingURL=add-people-to-collection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"add-people-to-collection.d.ts","sourceRoot":"","sources":["../../src/handlers/add-people-to-collection.ts"],"names":[],"mappings":"AA8BA,oIAAoI;AACpI,wBAAsB,2BAA2B,CAC/C,YAAY,EAAE,MAAM,EACpB,OAAO,EAAE;IACP,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,aAAa,CAAC,EAAE,MAAM,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,GACA,OAAO,CAAC,IAAI,CAAC,CAwDf"}
@@ -0,0 +1,79 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { readFileSync } from "node:fs";
4
+ import { DEFAULT_CDP_PORT, addPeopleToCollection, errorMessage, } from "@lhremote/core";
5
+ function parsePersonIds(raw) {
6
+ return raw
7
+ .split(",")
8
+ .map((s) => s.trim())
9
+ .filter((s) => s.length > 0)
10
+ .map(Number)
11
+ .filter((n) => Number.isInteger(n) && n > 0);
12
+ }
13
+ function readPersonIdsFile(filePath) {
14
+ const content = readFileSync(filePath, "utf-8");
15
+ return content
16
+ .split(/[\n,]/)
17
+ .map((s) => s.trim())
18
+ .filter((s) => s.length > 0)
19
+ .map(Number)
20
+ .filter((n) => Number.isInteger(n) && n > 0);
21
+ }
22
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#add-people-to-collection | add-people-to-collection} CLI command. */
23
+ export async function handleAddPeopleToCollection(collectionId, options) {
24
+ if (options.personIds && options.personIdsFile) {
25
+ process.stderr.write("Use only one of --person-ids or --person-ids-file.\n");
26
+ process.exitCode = 1;
27
+ return;
28
+ }
29
+ let personIds;
30
+ if (options.personIds) {
31
+ personIds = parsePersonIds(options.personIds);
32
+ }
33
+ else if (options.personIdsFile) {
34
+ try {
35
+ personIds = readPersonIdsFile(options.personIdsFile);
36
+ }
37
+ catch (error) {
38
+ const message = errorMessage(error);
39
+ process.stderr.write(`${message}\n`);
40
+ process.exitCode = 1;
41
+ return;
42
+ }
43
+ }
44
+ else {
45
+ process.stderr.write("Either --person-ids or --person-ids-file is required.\n");
46
+ process.exitCode = 1;
47
+ return;
48
+ }
49
+ if (personIds.length === 0) {
50
+ process.stderr.write("No valid person IDs provided.\n");
51
+ process.exitCode = 1;
52
+ return;
53
+ }
54
+ try {
55
+ const result = await addPeopleToCollection({
56
+ collectionId,
57
+ personIds,
58
+ cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
59
+ cdpHost: options.cdpHost,
60
+ allowRemote: options.allowRemote,
61
+ });
62
+ if (options.json) {
63
+ process.stdout.write(JSON.stringify(result, null, 2) + "\n");
64
+ }
65
+ else {
66
+ process.stdout.write(`Added ${String(result.added)} people to collection #${String(collectionId)}.` +
67
+ (result.alreadyInCollection > 0
68
+ ? ` ${String(result.alreadyInCollection)} already in collection.`
69
+ : "") +
70
+ "\n");
71
+ }
72
+ }
73
+ catch (error) {
74
+ const message = errorMessage(error);
75
+ process.stderr.write(`${message}\n`);
76
+ process.exitCode = 1;
77
+ }
78
+ }
79
+ //# sourceMappingURL=add-people-to-collection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"add-people-to-collection.js","sourceRoot":"","sources":["../../src/handlers/add-people-to-collection.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=add-people-to-collection.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"add-people-to-collection.test.d.ts","sourceRoot":"","sources":["../../src/handlers/add-people-to-collection.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,107 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
4
+ vi.mock("@lhremote/core", async (importOriginal) => {
5
+ const actual = await importOriginal();
6
+ return {
7
+ ...actual,
8
+ addPeopleToCollection: vi.fn(),
9
+ };
10
+ });
11
+ vi.mock("node:fs", async (importOriginal) => {
12
+ const actual = await importOriginal();
13
+ return {
14
+ ...actual,
15
+ readFileSync: vi.fn(),
16
+ };
17
+ });
18
+ import { addPeopleToCollection, } from "@lhremote/core";
19
+ import { readFileSync } from "node:fs";
20
+ import { handleAddPeopleToCollection } from "./add-people-to-collection.js";
21
+ import { getStdout } from "./testing/mock-helpers.js";
22
+ function mockResult(added, alreadyInCollection = 0) {
23
+ return {
24
+ success: true,
25
+ collectionId: 1,
26
+ added,
27
+ alreadyInCollection,
28
+ };
29
+ }
30
+ describe("handleAddPeopleToCollection", () => {
31
+ const originalExitCode = process.exitCode;
32
+ let stdoutSpy;
33
+ let stderrSpy;
34
+ beforeEach(() => {
35
+ process.exitCode = undefined;
36
+ vi.clearAllMocks();
37
+ stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
38
+ stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
39
+ });
40
+ afterEach(() => {
41
+ process.exitCode = originalExitCode;
42
+ vi.restoreAllMocks();
43
+ });
44
+ it("adds people with --person-ids and prints result", async () => {
45
+ vi.mocked(addPeopleToCollection).mockResolvedValue(mockResult(2));
46
+ await handleAddPeopleToCollection(1, { personIds: "100,200" });
47
+ expect(process.exitCode).toBeUndefined();
48
+ expect(getStdout(stdoutSpy)).toContain("Added 2 people to collection #1.");
49
+ });
50
+ it("reports alreadyInCollection count", async () => {
51
+ vi.mocked(addPeopleToCollection).mockResolvedValue(mockResult(1, 1));
52
+ await handleAddPeopleToCollection(1, { personIds: "100,200" });
53
+ const output = getStdout(stdoutSpy);
54
+ expect(output).toContain("Added 1 people");
55
+ expect(output).toContain("1 already in collection.");
56
+ });
57
+ it("reads from --person-ids-file", async () => {
58
+ vi.mocked(readFileSync).mockReturnValue("100\n200\n300");
59
+ vi.mocked(addPeopleToCollection).mockResolvedValue(mockResult(3));
60
+ await handleAddPeopleToCollection(1, { personIdsFile: "ids.txt" });
61
+ expect(process.exitCode).toBeUndefined();
62
+ });
63
+ it("prints JSON with --json", async () => {
64
+ vi.mocked(addPeopleToCollection).mockResolvedValue(mockResult(2));
65
+ await handleAddPeopleToCollection(1, { personIds: "100,200", json: true });
66
+ expect(process.exitCode).toBeUndefined();
67
+ const parsed = JSON.parse(getStdout(stdoutSpy));
68
+ expect(parsed.success).toBe(true);
69
+ expect(parsed.collectionId).toBe(1);
70
+ expect(parsed.added).toBe(2);
71
+ expect(parsed.alreadyInCollection).toBe(0);
72
+ });
73
+ it("sets exitCode 1 when both person-ids options provided", async () => {
74
+ await handleAddPeopleToCollection(1, {
75
+ personIds: "100",
76
+ personIdsFile: "ids.txt",
77
+ });
78
+ expect(process.exitCode).toBe(1);
79
+ expect(stderrSpy).toHaveBeenCalledWith("Use only one of --person-ids or --person-ids-file.\n");
80
+ });
81
+ it("sets exitCode 1 when no person-ids option provided", async () => {
82
+ await handleAddPeopleToCollection(1, {});
83
+ expect(process.exitCode).toBe(1);
84
+ expect(stderrSpy).toHaveBeenCalledWith("Either --person-ids or --person-ids-file is required.\n");
85
+ });
86
+ it("sets exitCode 1 when person IDs are empty", async () => {
87
+ vi.mocked(readFileSync).mockReturnValue("");
88
+ await handleAddPeopleToCollection(1, { personIdsFile: "empty.txt" });
89
+ expect(process.exitCode).toBe(1);
90
+ expect(stderrSpy).toHaveBeenCalledWith("No valid person IDs provided.\n");
91
+ });
92
+ it("sets exitCode 1 on error", async () => {
93
+ vi.mocked(addPeopleToCollection).mockRejectedValue(new Error("timeout"));
94
+ await handleAddPeopleToCollection(1, { personIds: "100" });
95
+ expect(process.exitCode).toBe(1);
96
+ expect(stderrSpy).toHaveBeenCalledWith("timeout\n");
97
+ });
98
+ it("sets exitCode 1 when person-ids-file read fails", async () => {
99
+ vi.mocked(readFileSync).mockImplementation(() => {
100
+ throw new Error("ENOENT: no such file");
101
+ });
102
+ await handleAddPeopleToCollection(1, { personIdsFile: "missing.txt" });
103
+ expect(process.exitCode).toBe(1);
104
+ expect(stderrSpy).toHaveBeenCalledWith(expect.stringContaining("ENOENT"));
105
+ });
106
+ });
107
+ //# sourceMappingURL=add-people-to-collection.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"add-people-to-collection.test.js","sourceRoot":"","sources":["../../src/handlers/add-people-to-collection.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#collect-people | collect-people} CLI command. */
2
+ export declare function handleCollectPeople(campaignId: number, sourceUrl: string, options: {
3
+ limit?: number;
4
+ maxPages?: number;
5
+ pageSize?: number;
6
+ sourceType?: string;
7
+ cdpPort?: number;
8
+ cdpHost?: string;
9
+ allowRemote?: boolean;
10
+ json?: boolean;
11
+ }): Promise<void>;
12
+ //# sourceMappingURL=collect-people.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"collect-people.d.ts","sourceRoot":"","sources":["../../src/handlers/collect-people.ts"],"names":[],"mappings":"AAWA,gHAAgH;AAChH,wBAAsB,mBAAmB,CACvC,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,MAAM,EACjB,OAAO,EAAE;IACP,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,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,CAkCf"}
@@ -0,0 +1,39 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { CollectionBusyError, CollectionError, DEFAULT_CDP_PORT, collectPeople, errorMessage, } from "@lhremote/core";
4
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#collect-people | collect-people} CLI command. */
5
+ export async function handleCollectPeople(campaignId, sourceUrl, options) {
6
+ try {
7
+ const result = await collectPeople({
8
+ campaignId,
9
+ sourceUrl,
10
+ ...(options.limit !== undefined && { limit: options.limit }),
11
+ ...(options.maxPages !== undefined && { maxPages: options.maxPages }),
12
+ ...(options.pageSize !== undefined && { pageSize: options.pageSize }),
13
+ ...(options.sourceType !== undefined && { sourceType: options.sourceType }),
14
+ cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
15
+ ...(options.cdpHost !== undefined && { cdpHost: options.cdpHost }),
16
+ ...(options.allowRemote !== undefined && { allowRemote: options.allowRemote }),
17
+ });
18
+ if (options.json) {
19
+ process.stdout.write(JSON.stringify(result, null, 2) + "\n");
20
+ }
21
+ else {
22
+ process.stdout.write(`Started collecting people (${result.sourceType}) into campaign ${String(campaignId)}.\n`);
23
+ }
24
+ }
25
+ catch (error) {
26
+ if (error instanceof CollectionBusyError) {
27
+ process.stderr.write(`Cannot collect — instance is busy (state: ${error.runnerState}).\n`);
28
+ }
29
+ else if (error instanceof CollectionError) {
30
+ process.stderr.write(`${error.message}\n`);
31
+ }
32
+ else {
33
+ const message = errorMessage(error);
34
+ process.stderr.write(`${message}\n`);
35
+ }
36
+ process.exitCode = 1;
37
+ }
38
+ }
39
+ //# sourceMappingURL=collect-people.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"collect-people.js","sourceRoot":"","sources":["../../src/handlers/collect-people.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,mBAAmB,EACnB,eAAe,EACf,gBAAgB,EAChB,aAAa,EACb,YAAY,GACb,MAAM,gBAAgB,CAAC;AAExB,gHAAgH;AAChH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,UAAkB,EAClB,SAAiB,EACjB,OASC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC;YACjC,UAAU;YACV,SAAS;YACT,GAAG,CAAC,OAAO,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC;YAC5D,GAAG,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrE,GAAG,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrE,GAAG,CAAC,OAAO,CAAC,UAAU,KAAK,SAAS,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,UAAU,EAAE,CAAC;YAC3E,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,gBAAgB;YAC5C,GAAG,CAAC,OAAO,CAAC,OAAO,KAAK,SAAS,IAAI,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,EAAE,CAAC;YAClE,GAAG,CAAC,OAAO,CAAC,WAAW,KAAK,SAAS,IAAI,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,CAAC;SAC/E,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;QAC/D,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,8BAA8B,MAAM,CAAC,UAAU,mBAAmB,MAAM,CAAC,UAAU,CAAC,KAAK,CAC1F,CAAC;QACJ,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,mBAAmB,EAAE,CAAC;YACzC,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,6CAA6C,KAAK,CAAC,WAAW,MAAM,CACrE,CAAC;QACJ,CAAC;aAAM,IAAI,KAAK,YAAY,eAAe,EAAE,CAAC;YAC5C,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,OAAO,IAAI,CAAC,CAAC;QAC7C,CAAC;aAAM,CAAC;YACN,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;YACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,IAAI,CAAC,CAAC;QACvC,CAAC;QACD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=collect-people.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"collect-people.test.d.ts","sourceRoot":"","sources":["../../src/handlers/collect-people.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,92 @@
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
+ collectPeople: vi.fn(),
9
+ };
10
+ });
11
+ import { CollectionBusyError, CollectionError, collectPeople, } from "@lhremote/core";
12
+ import { handleCollectPeople } from "./collect-people.js";
13
+ import { getStdout } from "./testing/mock-helpers.js";
14
+ const MOCK_RESULT = {
15
+ success: true,
16
+ campaignId: 42,
17
+ sourceType: "SearchPage",
18
+ };
19
+ describe("handleCollectPeople", () => {
20
+ const originalExitCode = process.exitCode;
21
+ let stdoutSpy;
22
+ let stderrSpy;
23
+ beforeEach(() => {
24
+ process.exitCode = undefined;
25
+ vi.clearAllMocks();
26
+ stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
27
+ stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
28
+ });
29
+ afterEach(() => {
30
+ process.exitCode = originalExitCode;
31
+ vi.restoreAllMocks();
32
+ });
33
+ it("prints success message with detected source type", async () => {
34
+ vi.mocked(collectPeople).mockResolvedValue(MOCK_RESULT);
35
+ await handleCollectPeople(42, "https://www.linkedin.com/search/results/people/", {});
36
+ expect(process.exitCode).toBeUndefined();
37
+ expect(getStdout(stdoutSpy)).toContain("Started collecting people (SearchPage) into campaign 42.");
38
+ });
39
+ it("prints JSON with --json", async () => {
40
+ vi.mocked(collectPeople).mockResolvedValue(MOCK_RESULT);
41
+ await handleCollectPeople(42, "https://www.linkedin.com/search/results/people/", {
42
+ json: true,
43
+ });
44
+ expect(process.exitCode).toBeUndefined();
45
+ const parsed = JSON.parse(getStdout(stdoutSpy));
46
+ expect(parsed.success).toBe(true);
47
+ expect(parsed.campaignId).toBe(42);
48
+ expect(parsed.sourceType).toBe("SearchPage");
49
+ });
50
+ it("passes all options to operation", async () => {
51
+ vi.mocked(collectPeople).mockResolvedValue(MOCK_RESULT);
52
+ await handleCollectPeople(42, "https://www.linkedin.com/search/results/people/", {
53
+ limit: 100,
54
+ maxPages: 5,
55
+ pageSize: 25,
56
+ sourceType: "SearchPage",
57
+ cdpPort: 1234,
58
+ cdpHost: "192.168.1.1",
59
+ allowRemote: true,
60
+ });
61
+ expect(collectPeople).toHaveBeenCalledWith(expect.objectContaining({
62
+ campaignId: 42,
63
+ sourceUrl: "https://www.linkedin.com/search/results/people/",
64
+ limit: 100,
65
+ maxPages: 5,
66
+ pageSize: 25,
67
+ sourceType: "SearchPage",
68
+ cdpPort: 1234,
69
+ cdpHost: "192.168.1.1",
70
+ allowRemote: true,
71
+ }));
72
+ });
73
+ it("sets exitCode 1 when instance is busy", async () => {
74
+ vi.mocked(collectPeople).mockRejectedValue(new CollectionBusyError("running"));
75
+ await handleCollectPeople(42, "https://www.linkedin.com/search/results/people/", {});
76
+ expect(process.exitCode).toBe(1);
77
+ expect(stderrSpy).toHaveBeenCalledWith("Cannot collect — instance is busy (state: running).\n");
78
+ });
79
+ it("sets exitCode 1 on CollectionError", async () => {
80
+ vi.mocked(collectPeople).mockRejectedValue(new CollectionError("Unrecognized source URL: https://example.com — cannot determine LinkedIn page type"));
81
+ await handleCollectPeople(42, "https://example.com", {});
82
+ expect(process.exitCode).toBe(1);
83
+ expect(stderrSpy).toHaveBeenCalledWith("Unrecognized source URL: https://example.com — cannot determine LinkedIn page type\n");
84
+ });
85
+ it("sets exitCode 1 on generic error", async () => {
86
+ vi.mocked(collectPeople).mockRejectedValue(new Error("timeout"));
87
+ await handleCollectPeople(42, "https://www.linkedin.com/search/results/people/", {});
88
+ expect(process.exitCode).toBe(1);
89
+ expect(stderrSpy).toHaveBeenCalledWith("timeout\n");
90
+ });
91
+ });
92
+ //# sourceMappingURL=collect-people.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"collect-people.test.js","sourceRoot":"","sources":["../../src/handlers/collect-people.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#create-collection | create-collection} CLI command. */
2
+ export declare function handleCreateCollection(name: string, options: {
3
+ cdpPort?: number;
4
+ cdpHost?: string;
5
+ allowRemote?: boolean;
6
+ json?: boolean;
7
+ }): Promise<void>;
8
+ //# sourceMappingURL=create-collection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-collection.d.ts","sourceRoot":"","sources":["../../src/handlers/create-collection.ts"],"names":[],"mappings":"AASA,sHAAsH;AACtH,wBAAsB,sBAAsB,CAC1C,IAAI,EAAE,MAAM,EACZ,OAAO,EAAE;IACP,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,CAqBf"}
@@ -0,0 +1,26 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { DEFAULT_CDP_PORT, createCollection, errorMessage, } from "@lhremote/core";
4
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#create-collection | create-collection} CLI command. */
5
+ export async function handleCreateCollection(name, options) {
6
+ try {
7
+ const result = await createCollection({
8
+ name,
9
+ cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
10
+ cdpHost: options.cdpHost,
11
+ allowRemote: options.allowRemote,
12
+ });
13
+ if (options.json) {
14
+ process.stdout.write(JSON.stringify(result, null, 2) + "\n");
15
+ }
16
+ else {
17
+ process.stdout.write(`Created collection #${String(result.collectionId)} "${result.name}".\n`);
18
+ }
19
+ }
20
+ catch (error) {
21
+ const message = errorMessage(error);
22
+ process.stderr.write(`${message}\n`);
23
+ process.exitCode = 1;
24
+ }
25
+ }
26
+ //# sourceMappingURL=create-collection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-collection.js","sourceRoot":"","sources":["../../src/handlers/create-collection.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,gBAAgB,EAChB,gBAAgB,EAChB,YAAY,GACb,MAAM,gBAAgB,CAAC;AAExB,sHAAsH;AACtH,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAC1C,IAAY,EACZ,OAKC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,gBAAgB,CAAC;YACpC,IAAI;YACJ,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,gBAAgB;YAC5C,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,WAAW,EAAE,OAAO,CAAC,WAAW;SACjC,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;QAC/D,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,uBAAuB,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,KAAK,MAAM,CAAC,IAAI,MAAM,CACzE,CAAC;QACJ,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,IAAI,CAAC,CAAC;QACrC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=create-collection.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-collection.test.d.ts","sourceRoot":"","sources":["../../src/handlers/create-collection.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,55 @@
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
+ createCollection: vi.fn(),
9
+ };
10
+ });
11
+ import { createCollection, } from "@lhremote/core";
12
+ import { handleCreateCollection } from "./create-collection.js";
13
+ import { getStdout } from "./testing/mock-helpers.js";
14
+ const MOCK_RESULT = {
15
+ success: true,
16
+ collectionId: 3,
17
+ name: "New Prospects",
18
+ };
19
+ describe("handleCreateCollection", () => {
20
+ const originalExitCode = process.exitCode;
21
+ let stdoutSpy;
22
+ let stderrSpy;
23
+ beforeEach(() => {
24
+ process.exitCode = undefined;
25
+ vi.clearAllMocks();
26
+ stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
27
+ stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
28
+ });
29
+ afterEach(() => {
30
+ process.exitCode = originalExitCode;
31
+ vi.restoreAllMocks();
32
+ });
33
+ it("creates collection and prints confirmation", async () => {
34
+ vi.mocked(createCollection).mockResolvedValue(MOCK_RESULT);
35
+ await handleCreateCollection("New Prospects", {});
36
+ expect(process.exitCode).toBeUndefined();
37
+ expect(getStdout(stdoutSpy)).toContain('Created collection #3 "New Prospects".');
38
+ });
39
+ it("prints JSON with --json", async () => {
40
+ vi.mocked(createCollection).mockResolvedValue(MOCK_RESULT);
41
+ await handleCreateCollection("New Prospects", { json: true });
42
+ expect(process.exitCode).toBeUndefined();
43
+ const parsed = JSON.parse(getStdout(stdoutSpy));
44
+ expect(parsed.success).toBe(true);
45
+ expect(parsed.collectionId).toBe(3);
46
+ expect(parsed.name).toBe("New Prospects");
47
+ });
48
+ it("sets exitCode 1 on error", async () => {
49
+ vi.mocked(createCollection).mockRejectedValue(new Error("connection error"));
50
+ await handleCreateCollection("New Prospects", {});
51
+ expect(process.exitCode).toBe(1);
52
+ expect(stderrSpy).toHaveBeenCalledWith("connection error\n");
53
+ });
54
+ });
55
+ //# sourceMappingURL=create-collection.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-collection.test.js","sourceRoot":"","sources":["../../src/handlers/create-collection.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#delete-collection | delete-collection} CLI command. */
2
+ export declare function handleDeleteCollection(collectionId: number, options: {
3
+ cdpPort?: number;
4
+ cdpHost?: string;
5
+ allowRemote?: boolean;
6
+ json?: boolean;
7
+ }): Promise<void>;
8
+ //# sourceMappingURL=delete-collection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"delete-collection.d.ts","sourceRoot":"","sources":["../../src/handlers/delete-collection.ts"],"names":[],"mappings":"AASA,sHAAsH;AACtH,wBAAsB,sBAAsB,CAC1C,YAAY,EAAE,MAAM,EACpB,OAAO,EAAE;IACP,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,CA2Bf"}
@@ -0,0 +1,31 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { DEFAULT_CDP_PORT, deleteCollection, errorMessage, } from "@lhremote/core";
4
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#delete-collection | delete-collection} CLI command. */
5
+ export async function handleDeleteCollection(collectionId, options) {
6
+ try {
7
+ const result = await deleteCollection({
8
+ collectionId,
9
+ cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
10
+ cdpHost: options.cdpHost,
11
+ allowRemote: options.allowRemote,
12
+ });
13
+ if (options.json) {
14
+ process.stdout.write(JSON.stringify(result, null, 2) + "\n");
15
+ }
16
+ else {
17
+ if (result.deleted) {
18
+ process.stdout.write(`Deleted collection #${String(collectionId)}.\n`);
19
+ }
20
+ else {
21
+ process.stdout.write(`Collection #${String(collectionId)} not found.\n`);
22
+ }
23
+ }
24
+ }
25
+ catch (error) {
26
+ const message = errorMessage(error);
27
+ process.stderr.write(`${message}\n`);
28
+ process.exitCode = 1;
29
+ }
30
+ }
31
+ //# sourceMappingURL=delete-collection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"delete-collection.js","sourceRoot":"","sources":["../../src/handlers/delete-collection.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EACL,gBAAgB,EAChB,gBAAgB,EAChB,YAAY,GACb,MAAM,gBAAgB,CAAC;AAExB,sHAAsH;AACtH,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAC1C,YAAoB,EACpB,OAKC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,gBAAgB,CAAC;YACpC,YAAY;YACZ,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,gBAAgB;YAC5C,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,WAAW,EAAE,OAAO,CAAC,WAAW;SACjC,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;QAC/D,CAAC;aAAM,CAAC;YACN,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;gBACnB,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,uBAAuB,MAAM,CAAC,YAAY,CAAC,KAAK,CACjD,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,eAAe,MAAM,CAAC,YAAY,CAAC,eAAe,CACnD,CAAC;YACJ,CAAC;QACH,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,IAAI,CAAC,CAAC;QACrC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=delete-collection.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"delete-collection.test.d.ts","sourceRoot":"","sources":["../../src/handlers/delete-collection.test.ts"],"names":[],"mappings":""}