@lhremote/cli 0.6.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.
Files changed (176) hide show
  1. package/README.md +5 -4
  2. package/dist/handlers/build-url.d.ts +17 -0
  3. package/dist/handlers/build-url.d.ts.map +1 -0
  4. package/dist/handlers/build-url.js +90 -0
  5. package/dist/handlers/build-url.js.map +1 -0
  6. package/dist/handlers/campaign-delete.d.ts +1 -0
  7. package/dist/handlers/campaign-delete.d.ts.map +1 -1
  8. package/dist/handlers/campaign-delete.js +3 -1
  9. package/dist/handlers/campaign-delete.js.map +1 -1
  10. package/dist/handlers/campaign-delete.test.js +25 -0
  11. package/dist/handlers/campaign-delete.test.js.map +1 -1
  12. package/dist/handlers/campaign-erase.d.ts +8 -0
  13. package/dist/handlers/campaign-erase.d.ts.map +1 -0
  14. package/dist/handlers/campaign-erase.js +39 -0
  15. package/dist/handlers/campaign-erase.js.map +1 -0
  16. package/dist/handlers/campaign-erase.test.d.ts +2 -0
  17. package/dist/handlers/campaign-erase.test.d.ts.map +1 -0
  18. package/dist/handlers/campaign-erase.test.js +71 -0
  19. package/dist/handlers/campaign-erase.test.js.map +1 -0
  20. package/dist/handlers/comment-on-post.d.ts +10 -0
  21. package/dist/handlers/comment-on-post.d.ts.map +1 -0
  22. package/dist/handlers/comment-on-post.js +37 -0
  23. package/dist/handlers/comment-on-post.js.map +1 -0
  24. package/dist/handlers/dismiss-errors.d.ts +8 -0
  25. package/dist/handlers/dismiss-errors.d.ts.map +1 -0
  26. package/dist/handlers/dismiss-errors.js +28 -0
  27. package/dist/handlers/dismiss-errors.js.map +1 -0
  28. package/dist/handlers/dismiss-errors.test.d.ts +2 -0
  29. package/dist/handlers/dismiss-errors.test.d.ts.map +1 -0
  30. package/dist/handlers/dismiss-errors.test.js +90 -0
  31. package/dist/handlers/dismiss-errors.test.js.map +1 -0
  32. package/dist/handlers/endorse-skills.d.ts +14 -0
  33. package/dist/handlers/endorse-skills.d.ts.map +1 -0
  34. package/dist/handlers/endorse-skills.js +44 -0
  35. package/dist/handlers/endorse-skills.js.map +1 -0
  36. package/dist/handlers/endorse-skills.test.d.ts +2 -0
  37. package/dist/handlers/endorse-skills.test.d.ts.map +1 -0
  38. package/dist/handlers/endorse-skills.test.js +65 -0
  39. package/dist/handlers/endorse-skills.test.js.map +1 -0
  40. package/dist/handlers/enrich-profile.d.ts +16 -0
  41. package/dist/handlers/enrich-profile.d.ts.map +1 -0
  42. package/dist/handlers/enrich-profile.js +51 -0
  43. package/dist/handlers/enrich-profile.js.map +1 -0
  44. package/dist/handlers/enrich-profile.test.d.ts +2 -0
  45. package/dist/handlers/enrich-profile.test.d.ts.map +1 -0
  46. package/dist/handlers/enrich-profile.test.js +65 -0
  47. package/dist/handlers/enrich-profile.test.js.map +1 -0
  48. package/dist/handlers/follow-person.d.ts +13 -0
  49. package/dist/handlers/follow-person.d.ts.map +1 -0
  50. package/dist/handlers/follow-person.js +45 -0
  51. package/dist/handlers/follow-person.js.map +1 -0
  52. package/dist/handlers/follow-person.test.d.ts +2 -0
  53. package/dist/handlers/follow-person.test.d.ts.map +1 -0
  54. package/dist/handlers/follow-person.test.js +65 -0
  55. package/dist/handlers/follow-person.test.js.map +1 -0
  56. package/dist/handlers/get-action-budget.d.ts +8 -0
  57. package/dist/handlers/get-action-budget.d.ts.map +1 -0
  58. package/dist/handlers/get-action-budget.js +46 -0
  59. package/dist/handlers/get-action-budget.js.map +1 -0
  60. package/dist/handlers/get-action-budget.test.d.ts +2 -0
  61. package/dist/handlers/get-action-budget.test.d.ts.map +1 -0
  62. package/dist/handlers/get-action-budget.test.js +69 -0
  63. package/dist/handlers/get-action-budget.test.js.map +1 -0
  64. package/dist/handlers/get-errors.d.ts.map +1 -1
  65. package/dist/handlers/get-errors.js +12 -0
  66. package/dist/handlers/get-errors.js.map +1 -1
  67. package/dist/handlers/get-errors.test.js +25 -0
  68. package/dist/handlers/get-errors.test.js.map +1 -1
  69. package/dist/handlers/get-feed.d.ts +10 -0
  70. package/dist/handlers/get-feed.d.ts.map +1 -0
  71. package/dist/handlers/get-feed.js +62 -0
  72. package/dist/handlers/get-feed.js.map +1 -0
  73. package/dist/handlers/get-feed.test.d.ts +2 -0
  74. package/dist/handlers/get-feed.test.d.ts.map +1 -0
  75. package/dist/handlers/get-feed.test.js +126 -0
  76. package/dist/handlers/get-feed.test.js.map +1 -0
  77. package/dist/handlers/get-post-stats.d.ts +8 -0
  78. package/dist/handlers/get-post-stats.d.ts.map +1 -0
  79. package/dist/handlers/get-post-stats.js +37 -0
  80. package/dist/handlers/get-post-stats.js.map +1 -0
  81. package/dist/handlers/get-post.d.ts +10 -0
  82. package/dist/handlers/get-post.d.ts.map +1 -0
  83. package/dist/handlers/get-post.js +66 -0
  84. package/dist/handlers/get-post.js.map +1 -0
  85. package/dist/handlers/get-profile-activity.d.ts +10 -0
  86. package/dist/handlers/get-profile-activity.d.ts.map +1 -0
  87. package/dist/handlers/get-profile-activity.js +56 -0
  88. package/dist/handlers/get-profile-activity.js.map +1 -0
  89. package/dist/handlers/get-throttle-status.d.ts +8 -0
  90. package/dist/handlers/get-throttle-status.d.ts.map +1 -0
  91. package/dist/handlers/get-throttle-status.js +33 -0
  92. package/dist/handlers/get-throttle-status.js.map +1 -0
  93. package/dist/handlers/get-throttle-status.test.d.ts +2 -0
  94. package/dist/handlers/get-throttle-status.test.d.ts.map +1 -0
  95. package/dist/handlers/get-throttle-status.test.js +65 -0
  96. package/dist/handlers/get-throttle-status.test.js.map +1 -0
  97. package/dist/handlers/index.d.ts +23 -0
  98. package/dist/handlers/index.d.ts.map +1 -1
  99. package/dist/handlers/index.js +23 -0
  100. package/dist/handlers/index.js.map +1 -1
  101. package/dist/handlers/like-person-posts.d.ts +18 -0
  102. package/dist/handlers/like-person-posts.d.ts.map +1 -0
  103. package/dist/handlers/like-person-posts.js +59 -0
  104. package/dist/handlers/like-person-posts.js.map +1 -0
  105. package/dist/handlers/like-person-posts.test.d.ts +2 -0
  106. package/dist/handlers/like-person-posts.test.d.ts.map +1 -0
  107. package/dist/handlers/like-person-posts.test.js +71 -0
  108. package/dist/handlers/like-person-posts.test.js.map +1 -0
  109. package/dist/handlers/list-reference-data.d.ts +5 -0
  110. package/dist/handlers/list-reference-data.d.ts.map +1 -0
  111. package/dist/handlers/list-reference-data.js +33 -0
  112. package/dist/handlers/list-reference-data.js.map +1 -0
  113. package/dist/handlers/message-person.d.ts +15 -0
  114. package/dist/handlers/message-person.d.ts.map +1 -0
  115. package/dist/handlers/message-person.js +65 -0
  116. package/dist/handlers/message-person.js.map +1 -0
  117. package/dist/handlers/message-person.test.d.ts +2 -0
  118. package/dist/handlers/message-person.test.d.ts.map +1 -0
  119. package/dist/handlers/message-person.test.js +77 -0
  120. package/dist/handlers/message-person.test.js.map +1 -0
  121. package/dist/handlers/query-messages.integration.test.js +24 -3
  122. package/dist/handlers/query-messages.integration.test.js.map +1 -1
  123. package/dist/handlers/react-to-post.d.ts +9 -0
  124. package/dist/handlers/react-to-post.d.ts.map +1 -0
  125. package/dist/handlers/react-to-post.js +30 -0
  126. package/dist/handlers/react-to-post.js.map +1 -0
  127. package/dist/handlers/remove-connection.d.ts +11 -0
  128. package/dist/handlers/remove-connection.d.ts.map +1 -0
  129. package/dist/handlers/remove-connection.js +41 -0
  130. package/dist/handlers/remove-connection.js.map +1 -0
  131. package/dist/handlers/remove-connection.test.d.ts +2 -0
  132. package/dist/handlers/remove-connection.test.d.ts.map +1 -0
  133. package/dist/handlers/remove-connection.test.js +65 -0
  134. package/dist/handlers/remove-connection.test.js.map +1 -0
  135. package/dist/handlers/resolve-entity.d.ts +9 -0
  136. package/dist/handlers/resolve-entity.d.ts.map +1 -0
  137. package/dist/handlers/resolve-entity.js +48 -0
  138. package/dist/handlers/resolve-entity.js.map +1 -0
  139. package/dist/handlers/search-posts.d.ts +10 -0
  140. package/dist/handlers/search-posts.d.ts.map +1 -0
  141. package/dist/handlers/search-posts.js +61 -0
  142. package/dist/handlers/search-posts.js.map +1 -0
  143. package/dist/handlers/search-posts.test.d.ts +2 -0
  144. package/dist/handlers/search-posts.test.d.ts.map +1 -0
  145. package/dist/handlers/search-posts.test.js +105 -0
  146. package/dist/handlers/search-posts.test.js.map +1 -0
  147. package/dist/handlers/send-inmail.d.ts +15 -0
  148. package/dist/handlers/send-inmail.d.ts.map +1 -0
  149. package/dist/handlers/send-inmail.js +65 -0
  150. package/dist/handlers/send-inmail.js.map +1 -0
  151. package/dist/handlers/send-inmail.test.d.ts +2 -0
  152. package/dist/handlers/send-inmail.test.d.ts.map +1 -0
  153. package/dist/handlers/send-inmail.test.js +77 -0
  154. package/dist/handlers/send-inmail.test.js.map +1 -0
  155. package/dist/handlers/send-invite.d.ts +13 -0
  156. package/dist/handlers/send-invite.d.ts.map +1 -0
  157. package/dist/handlers/send-invite.js +54 -0
  158. package/dist/handlers/send-invite.js.map +1 -0
  159. package/dist/handlers/send-invite.test.d.ts +2 -0
  160. package/dist/handlers/send-invite.test.d.ts.map +1 -0
  161. package/dist/handlers/send-invite.test.js +71 -0
  162. package/dist/handlers/send-invite.test.js.map +1 -0
  163. package/dist/handlers/visit-profile.d.ts +11 -0
  164. package/dist/handlers/visit-profile.d.ts.map +1 -0
  165. package/dist/handlers/visit-profile.js +93 -0
  166. package/dist/handlers/visit-profile.js.map +1 -0
  167. package/dist/handlers/visit-profile.test.d.ts +2 -0
  168. package/dist/handlers/visit-profile.test.d.ts.map +1 -0
  169. package/dist/handlers/visit-profile.test.js +169 -0
  170. package/dist/handlers/visit-profile.test.js.map +1 -0
  171. package/dist/program.d.ts.map +1 -1
  172. package/dist/program.js +279 -2
  173. package/dist/program.js.map +1 -1
  174. package/dist/program.test.js +24 -1
  175. package/dist/program.test.js.map +1 -1
  176. package/package.json +2 -2
@@ -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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=send-inmail.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-inmail.test.d.ts","sourceRoot":"","sources":["../../src/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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=send-invite.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-invite.test.d.ts","sourceRoot":"","sources":["../../src/handlers/send-invite.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,71 @@
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
+ sendInvite: vi.fn(),
9
+ };
10
+ });
11
+ import { CampaignExecutionError, CampaignTimeoutError, sendInvite, } from "@lhremote/core";
12
+ import { handleSendInvite } from "./send-invite.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("handleSendInvite", () => {
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(sendInvite).mockResolvedValue(MOCK_RESULT);
33
+ await handleSendInvite({ 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(sendInvite).mockResolvedValue(MOCK_RESULT);
42
+ await handleSendInvite({ personId: 100 });
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 handleSendInvite({});
48
+ expect(process.exitCode).toBe(1);
49
+ expect(getStderr(stderrSpy)).toContain("Exactly one of --person-id or --url");
50
+ expect(sendInvite).not.toHaveBeenCalled();
51
+ });
52
+ it("returns error for invalid JSON in messageTemplate", async () => {
53
+ await handleSendInvite({ personId: 100, messageTemplate: "not json" });
54
+ expect(process.exitCode).toBe(1);
55
+ expect(getStderr(stderrSpy)).toContain("Invalid JSON in --message-template");
56
+ expect(sendInvite).not.toHaveBeenCalled();
57
+ });
58
+ it("handles CampaignExecutionError", async () => {
59
+ vi.mocked(sendInvite).mockRejectedValue(new CampaignExecutionError("Person 100 not found"));
60
+ await handleSendInvite({ personId: 100 });
61
+ expect(process.exitCode).toBe(1);
62
+ expect(getStderr(stderrSpy)).toContain("Person 100 not found");
63
+ });
64
+ it("handles CampaignTimeoutError", async () => {
65
+ vi.mocked(sendInvite).mockRejectedValue(new CampaignTimeoutError("Timed out", 42));
66
+ await handleSendInvite({ personId: 100 });
67
+ expect(process.exitCode).toBe(1);
68
+ expect(getStderr(stderrSpy)).toContain("Timed out");
69
+ });
70
+ });
71
+ //# sourceMappingURL=send-invite.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send-invite.test.js","sourceRoot":"","sources":["../../src/handlers/send-invite.test.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,oCAAoC;AAEpC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAEzE,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE;IACjD,MAAM,MAAM,GAAG,MAAM,cAAc,EAAmC,CAAC;IACvE,OAAO;QACL,GAAG,MAAM;QACT,UAAU,EAAE,EAAE,CAAC,EAAE,EAAE;KACpB,CAAC;AACJ,CAAC,CAAC,CAAC;AAEH,OAAO,EAEL,sBAAsB,EACtB,oBAAoB,EACpB,UAAU,GACX,MAAM,gBAAgB,CAAC;AAExB,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AAEjE,MAAM,WAAW,GAA0B;IACzC,OAAO,EAAE,IAAI;IACb,QAAQ,EAAE,GAAG;IACb,OAAO,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,eAAe,EAAE,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,sBAAsB,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;CACrI,CAAC;AAEF,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,IAAI,SAAsC,CAAC;IAC3C,IAAI,SAAsC,CAAC;IAE3C,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;QACnB,OAAO,CAAC,QAAQ,GAAG,SAAS,CAAC;QAC7B,SAAS,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QACpE,SAAS,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;IACtE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,eAAe,EAAE,CAAC;IACvB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,KAAK,IAAI,EAAE;QAC9C,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;QAErD,MAAM,gBAAgB,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;QAEtD,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,aAAa,EAAE,CAAC;QACzC,MAAM,MAAM,GAAG,SAAS,CAAC,SAAS,CAAC,CAAC;QACpC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAA0B,CAAC;QAC3D,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,KAAK,IAAI,EAAE;QACxD,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;QAErD,MAAM,gBAAgB,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,CAAC,CAAC;QAE1C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,aAAa,EAAE,CAAC;QACzC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sDAAsD,EAAE,KAAK,IAAI,EAAE;QACpE,MAAM,gBAAgB,CAAC,EAAE,CAAC,CAAC;QAE3B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,qCAAqC,CAAC,CAAC;QAC9E,MAAM,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;QACjE,MAAM,gBAAgB,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,eAAe,EAAE,UAAU,EAAE,CAAC,CAAC;QAEvE,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,oCAAoC,CAAC,CAAC;QAC7E,MAAM,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,KAAK,IAAI,EAAE;QAC9C,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,iBAAiB,CACrC,IAAI,sBAAsB,CAAC,sBAAsB,CAAC,CACnD,CAAC;QAEF,MAAM,gBAAgB,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,CAAC,CAAC;QAE1C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,sBAAsB,CAAC,CAAC;IACjE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,KAAK,IAAI,EAAE;QAC5C,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,iBAAiB,CACrC,IAAI,oBAAoB,CAAC,WAAW,EAAE,EAAE,CAAC,CAC1C,CAAC;QAEF,MAAM,gBAAgB,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,CAAC,CAAC;QAE1C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;IACtD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,11 @@
1
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#visit-profile | visit-profile} CLI command. */
2
+ export declare function handleVisitProfile(options: {
3
+ personId?: number;
4
+ url?: string;
5
+ extractCurrentOrganizations?: boolean;
6
+ cdpPort?: number;
7
+ cdpHost?: string;
8
+ allowRemote?: boolean;
9
+ json?: boolean;
10
+ }): Promise<void>;
11
+ //# sourceMappingURL=visit-profile.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"visit-profile.d.ts","sourceRoot":"","sources":["../../src/handlers/visit-profile.ts"],"names":[],"mappings":"AAYA,8GAA8G;AAC9G,wBAAsB,kBAAkB,CAAC,OAAO,EAAE;IAChD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,2BAA2B,CAAC,EAAE,OAAO,CAAC;IACtC,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,CAuChB"}
@@ -0,0 +1,93 @@
1
+ // SPDX-License-Identifier: AGPL-3.0-only
2
+ // Copyright (C) 2026 Oleksii PELYKH
3
+ import { DEFAULT_CDP_PORT, errorMessage, InstanceNotRunningError, visitProfile, } from "@lhremote/core";
4
+ /** Handle the {@link https://github.com/alexey-pelykh/lhremote#visit-profile | visit-profile} CLI command. */
5
+ export async function handleVisitProfile(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("Visiting profile...\n");
12
+ let result;
13
+ try {
14
+ result = await visitProfile({
15
+ personId: options.personId,
16
+ url: options.url,
17
+ extractCurrentOrganizations: options.extractCurrentOrganizations,
18
+ cdpPort: options.cdpPort ?? DEFAULT_CDP_PORT,
19
+ cdpHost: options.cdpHost,
20
+ allowRemote: options.allowRemote,
21
+ });
22
+ }
23
+ catch (error) {
24
+ if (error instanceof InstanceNotRunningError) {
25
+ process.stderr.write(`${error.message}\n`);
26
+ }
27
+ else {
28
+ const message = errorMessage(error);
29
+ process.stderr.write(`${message}\n`);
30
+ }
31
+ process.exitCode = 1;
32
+ return;
33
+ }
34
+ process.stderr.write("Done.\n");
35
+ if (options.json) {
36
+ process.stdout.write(JSON.stringify(result, null, 2) + "\n");
37
+ }
38
+ else {
39
+ printProfile(result.profile);
40
+ }
41
+ }
42
+ function printProfile(profile) {
43
+ const name = [profile.miniProfile.firstName, profile.miniProfile.lastName]
44
+ .filter(Boolean)
45
+ .join(" ");
46
+ process.stdout.write(`${name} (#${String(profile.id)})\n`);
47
+ if (profile.miniProfile.headline) {
48
+ process.stdout.write(`${profile.miniProfile.headline}\n`);
49
+ }
50
+ if (profile.currentPosition) {
51
+ const parts = [
52
+ profile.currentPosition.title,
53
+ profile.currentPosition.company,
54
+ ].filter(Boolean);
55
+ if (parts.length > 0) {
56
+ process.stdout.write(`\nCurrent: ${parts.join(" at ")}\n`);
57
+ }
58
+ }
59
+ if (profile.positions && profile.positions.length > 0) {
60
+ process.stdout.write("\nPositions:\n");
61
+ for (const pos of profile.positions) {
62
+ const role = [pos.title, pos.company].filter(Boolean).join(" at ");
63
+ const dates = [pos.startDate ?? "?", pos.endDate ?? "present"].join(" – ");
64
+ process.stdout.write(` ${role} (${dates})\n`);
65
+ }
66
+ }
67
+ if (profile.education.length > 0) {
68
+ process.stdout.write("\nEducation:\n");
69
+ for (const edu of profile.education) {
70
+ const parts = [edu.degree, edu.field].filter(Boolean).join(" in ");
71
+ const dates = [edu.startDate, edu.endDate].filter(Boolean).join(" – ");
72
+ const school = edu.school ?? "";
73
+ const schoolWithDates = dates
74
+ ? [school, `(${dates})`].filter(Boolean).join(" ")
75
+ : school;
76
+ const line = [parts, schoolWithDates].filter(Boolean).join(", ");
77
+ if (line) {
78
+ process.stdout.write(` ${line}\n`);
79
+ }
80
+ }
81
+ }
82
+ if (profile.skills.length > 0) {
83
+ process.stdout.write(`\nSkills: ${profile.skills.map((s) => s.name).join(", ")}\n`);
84
+ }
85
+ if (profile.emails.length > 0) {
86
+ process.stdout.write(`Email: ${profile.emails.join(", ")}\n`);
87
+ }
88
+ const publicExtId = profile.externalIds.find((e) => e.typeGroup === "public");
89
+ if (publicExtId) {
90
+ process.stdout.write(`\nLinkedIn: linkedin.com/in/${publicExtId.externalId}\n`);
91
+ }
92
+ }
93
+ //# sourceMappingURL=visit-profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"visit-profile.js","sourceRoot":"","sources":["../../src/handlers/visit-profile.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=visit-profile.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"visit-profile.test.d.ts","sourceRoot":"","sources":["../../src/handlers/visit-profile.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,169 @@
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
+ visitProfile: vi.fn(),
9
+ };
10
+ });
11
+ import { InstanceNotRunningError, visitProfile, } from "@lhremote/core";
12
+ import { handleVisitProfile } from "./visit-profile.js";
13
+ import { getStderr, getStdout } from "./testing/mock-helpers.js";
14
+ const MOCK_PROFILE = {
15
+ id: 100,
16
+ miniProfile: {
17
+ firstName: "Jane",
18
+ lastName: "Doe",
19
+ headline: "Software Engineer at Acme",
20
+ avatar: null,
21
+ },
22
+ externalIds: [{ externalId: "jane-doe-123", typeGroup: "public", isMemberId: false }],
23
+ currentPosition: { company: "Acme Corp", title: "Senior Engineer" },
24
+ positions: [
25
+ {
26
+ company: "Acme Corp",
27
+ title: "Senior Engineer",
28
+ startDate: "2023-01",
29
+ endDate: null,
30
+ isCurrent: true,
31
+ },
32
+ ],
33
+ education: [
34
+ {
35
+ school: "MIT",
36
+ degree: "BS",
37
+ field: "Computer Science",
38
+ startDate: "2015",
39
+ endDate: "2019",
40
+ },
41
+ ],
42
+ skills: [{ name: "TypeScript" }, { name: "Node.js" }],
43
+ emails: ["jane@example.com"],
44
+ };
45
+ const MOCK_RESULT = {
46
+ success: true,
47
+ actionType: "VisitAndExtract",
48
+ profile: MOCK_PROFILE,
49
+ };
50
+ describe("handleVisitProfile", () => {
51
+ const originalExitCode = process.exitCode;
52
+ let stdoutSpy;
53
+ let stderrSpy;
54
+ beforeEach(() => {
55
+ process.exitCode = undefined;
56
+ vi.clearAllMocks();
57
+ stdoutSpy = vi.spyOn(process.stdout, "write").mockReturnValue(true);
58
+ stderrSpy = vi.spyOn(process.stderr, "write").mockReturnValue(true);
59
+ });
60
+ afterEach(() => {
61
+ process.exitCode = originalExitCode;
62
+ vi.restoreAllMocks();
63
+ });
64
+ it("exits with error when neither personId nor url provided", async () => {
65
+ await handleVisitProfile({});
66
+ expect(process.exitCode).toBe(1);
67
+ expect(getStderr(stderrSpy)).toContain("Exactly one of --person-id or --url must be provided");
68
+ expect(visitProfile).not.toHaveBeenCalled();
69
+ });
70
+ it("exits with error when both personId and url provided", async () => {
71
+ await handleVisitProfile({ personId: 100, url: "https://www.linkedin.com/in/jane" });
72
+ expect(process.exitCode).toBe(1);
73
+ expect(getStderr(stderrSpy)).toContain("Exactly one of --person-id or --url must be provided");
74
+ expect(visitProfile).not.toHaveBeenCalled();
75
+ });
76
+ it("prints JSON with --json", async () => {
77
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
78
+ await handleVisitProfile({ personId: 100, json: true });
79
+ expect(process.exitCode).toBeUndefined();
80
+ const output = JSON.parse(getStdout(stdoutSpy));
81
+ expect(output.success).toBe(true);
82
+ expect(output.actionType).toBe("VisitAndExtract");
83
+ expect(output.profile.id).toBe(100);
84
+ });
85
+ it("prints human-readable output by default", async () => {
86
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
87
+ await handleVisitProfile({ personId: 100 });
88
+ expect(process.exitCode).toBeUndefined();
89
+ const output = getStdout(stdoutSpy);
90
+ expect(output).toContain("Jane Doe (#100)");
91
+ expect(output).toContain("Software Engineer at Acme");
92
+ expect(output).toContain("Senior Engineer at Acme Corp");
93
+ expect(output).toContain("TypeScript, Node.js");
94
+ expect(output).toContain("jane@example.com");
95
+ expect(output).toContain("linkedin.com/in/jane-doe-123");
96
+ });
97
+ it("prints positions in human-readable output", async () => {
98
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
99
+ await handleVisitProfile({ personId: 100 });
100
+ const output = getStdout(stdoutSpy);
101
+ expect(output).toContain("Positions:");
102
+ expect(output).toContain("Senior Engineer at Acme Corp");
103
+ });
104
+ it("prints education in human-readable output", async () => {
105
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
106
+ await handleVisitProfile({ personId: 100 });
107
+ const output = getStdout(stdoutSpy);
108
+ expect(output).toContain("Education:");
109
+ expect(output).toContain("BS in Computer Science");
110
+ expect(output).toContain("MIT");
111
+ });
112
+ it("handles null school in education", async () => {
113
+ vi.mocked(visitProfile).mockResolvedValue({
114
+ ...MOCK_RESULT,
115
+ profile: {
116
+ ...MOCK_PROFILE,
117
+ education: [
118
+ { school: null, degree: "MBA", field: null, startDate: "2020", endDate: "2022" },
119
+ ],
120
+ },
121
+ });
122
+ await handleVisitProfile({ personId: 100 });
123
+ const output = getStdout(stdoutSpy);
124
+ expect(output).toContain("Education:");
125
+ expect(output).toContain("MBA");
126
+ expect(output).not.toContain("null");
127
+ });
128
+ it("passes url to operation when provided", async () => {
129
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
130
+ await handleVisitProfile({ url: "https://www.linkedin.com/in/jane-doe-123" });
131
+ expect(visitProfile).toHaveBeenCalledWith(expect.objectContaining({
132
+ url: "https://www.linkedin.com/in/jane-doe-123",
133
+ }));
134
+ });
135
+ it("prints progress to stderr", async () => {
136
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
137
+ await handleVisitProfile({ personId: 100 });
138
+ const stderr = getStderr(stderrSpy);
139
+ expect(stderr).toContain("Visiting profile...");
140
+ expect(stderr).toContain("Done.");
141
+ });
142
+ it("passes extractCurrentOrganizations when provided", async () => {
143
+ vi.mocked(visitProfile).mockResolvedValue(MOCK_RESULT);
144
+ await handleVisitProfile({ personId: 100, extractCurrentOrganizations: true });
145
+ expect(visitProfile).toHaveBeenCalledWith(expect.objectContaining({
146
+ personId: 100,
147
+ extractCurrentOrganizations: true,
148
+ }));
149
+ });
150
+ it("sets exitCode on error", async () => {
151
+ vi.mocked(visitProfile).mockRejectedValue(new Error("No accounts found."));
152
+ await handleVisitProfile({ personId: 100 });
153
+ expect(process.exitCode).toBe(1);
154
+ expect(getStderr(stderrSpy)).toContain("No accounts found.");
155
+ });
156
+ it("sets exitCode when no instance running", async () => {
157
+ vi.mocked(visitProfile).mockRejectedValue(new InstanceNotRunningError("No LinkedHelper instance is running. Use start-instance first."));
158
+ await handleVisitProfile({ personId: 100 });
159
+ expect(process.exitCode).toBe(1);
160
+ expect(getStderr(stderrSpy)).toContain("No LinkedHelper instance is running. Use start-instance first.");
161
+ });
162
+ it("sets exitCode on unexpected error", async () => {
163
+ vi.mocked(visitProfile).mockRejectedValue(new Error("connection reset"));
164
+ await handleVisitProfile({ personId: 100 });
165
+ expect(process.exitCode).toBe(1);
166
+ expect(getStderr(stderrSpy)).toContain("connection reset");
167
+ });
168
+ });
169
+ //# sourceMappingURL=visit-profile.test.js.map