@indigoai-us/hq-cloud 5.1.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 (108) hide show
  1. package/dist/auth.d.ts +21 -0
  2. package/dist/auth.d.ts.map +1 -0
  3. package/dist/auth.js +116 -0
  4. package/dist/auth.js.map +1 -0
  5. package/dist/cli/accept.d.ts +29 -0
  6. package/dist/cli/accept.d.ts.map +1 -0
  7. package/dist/cli/accept.js +67 -0
  8. package/dist/cli/accept.js.map +1 -0
  9. package/dist/cli/conflict.d.ts +33 -0
  10. package/dist/cli/conflict.d.ts.map +1 -0
  11. package/dist/cli/conflict.js +91 -0
  12. package/dist/cli/conflict.js.map +1 -0
  13. package/dist/cli/index.d.ts +19 -0
  14. package/dist/cli/index.d.ts.map +1 -0
  15. package/dist/cli/index.js +14 -0
  16. package/dist/cli/index.js.map +1 -0
  17. package/dist/cli/invite.d.ts +51 -0
  18. package/dist/cli/invite.d.ts.map +1 -0
  19. package/dist/cli/invite.js +120 -0
  20. package/dist/cli/invite.js.map +1 -0
  21. package/dist/cli/invite.test.d.ts +5 -0
  22. package/dist/cli/invite.test.d.ts.map +1 -0
  23. package/dist/cli/invite.test.js +175 -0
  24. package/dist/cli/invite.test.js.map +1 -0
  25. package/dist/cli/promote.d.ts +30 -0
  26. package/dist/cli/promote.d.ts.map +1 -0
  27. package/dist/cli/promote.js +79 -0
  28. package/dist/cli/promote.js.map +1 -0
  29. package/dist/cli/share.d.ts +33 -0
  30. package/dist/cli/share.d.ts.map +1 -0
  31. package/dist/cli/share.js +153 -0
  32. package/dist/cli/share.js.map +1 -0
  33. package/dist/cli/share.test.d.ts +5 -0
  34. package/dist/cli/share.test.d.ts.map +1 -0
  35. package/dist/cli/share.test.js +121 -0
  36. package/dist/cli/share.test.js.map +1 -0
  37. package/dist/cli/sync.d.ts +30 -0
  38. package/dist/cli/sync.d.ts.map +1 -0
  39. package/dist/cli/sync.js +138 -0
  40. package/dist/cli/sync.js.map +1 -0
  41. package/dist/cli/sync.test.d.ts +5 -0
  42. package/dist/cli/sync.test.d.ts.map +1 -0
  43. package/dist/cli/sync.test.js +172 -0
  44. package/dist/cli/sync.test.js.map +1 -0
  45. package/dist/cognito-auth.d.ts +70 -0
  46. package/dist/cognito-auth.d.ts.map +1 -0
  47. package/dist/cognito-auth.js +280 -0
  48. package/dist/cognito-auth.js.map +1 -0
  49. package/dist/context.d.ts +30 -0
  50. package/dist/context.d.ts.map +1 -0
  51. package/dist/context.js +117 -0
  52. package/dist/context.js.map +1 -0
  53. package/dist/context.test.d.ts +7 -0
  54. package/dist/context.test.d.ts.map +1 -0
  55. package/dist/context.test.js +148 -0
  56. package/dist/context.test.js.map +1 -0
  57. package/dist/daemon-worker.d.ts +6 -0
  58. package/dist/daemon-worker.d.ts.map +1 -0
  59. package/dist/daemon-worker.js +26 -0
  60. package/dist/daemon-worker.js.map +1 -0
  61. package/dist/daemon.d.ts +10 -0
  62. package/dist/daemon.d.ts.map +1 -0
  63. package/dist/daemon.js +88 -0
  64. package/dist/daemon.js.map +1 -0
  65. package/dist/ignore.d.ts +10 -0
  66. package/dist/ignore.d.ts.map +1 -0
  67. package/dist/ignore.js +54 -0
  68. package/dist/ignore.js.map +1 -0
  69. package/dist/index.d.ts +33 -0
  70. package/dist/index.d.ts.map +1 -0
  71. package/dist/index.js +138 -0
  72. package/dist/index.js.map +1 -0
  73. package/dist/journal.d.ts +12 -0
  74. package/dist/journal.d.ts.map +1 -0
  75. package/dist/journal.js +42 -0
  76. package/dist/journal.js.map +1 -0
  77. package/dist/s3.d.ts +15 -0
  78. package/dist/s3.d.ts.map +1 -0
  79. package/dist/s3.js +129 -0
  80. package/dist/s3.js.map +1 -0
  81. package/dist/types.d.ts +52 -0
  82. package/dist/types.d.ts.map +1 -0
  83. package/dist/types.js +5 -0
  84. package/dist/types.js.map +1 -0
  85. package/dist/vault-client.d.ts +164 -0
  86. package/dist/vault-client.d.ts.map +1 -0
  87. package/dist/vault-client.js +209 -0
  88. package/dist/vault-client.js.map +1 -0
  89. package/dist/vault-client.test.d.ts +7 -0
  90. package/dist/vault-client.test.d.ts.map +1 -0
  91. package/dist/vault-client.test.js +257 -0
  92. package/dist/vault-client.test.js.map +1 -0
  93. package/dist/watcher.d.ts +18 -0
  94. package/dist/watcher.d.ts.map +1 -0
  95. package/dist/watcher.js +106 -0
  96. package/dist/watcher.js.map +1 -0
  97. package/package.json +32 -0
  98. package/src/auth.ts +146 -0
  99. package/src/cognito-auth.ts +375 -0
  100. package/src/daemon-worker.ts +32 -0
  101. package/src/daemon.ts +97 -0
  102. package/src/ignore.ts +61 -0
  103. package/src/index.ts +182 -0
  104. package/src/journal.ts +63 -0
  105. package/src/s3.ts +178 -0
  106. package/src/types.ts +59 -0
  107. package/src/watcher.ts +130 -0
  108. package/tsconfig.json +8 -0
@@ -0,0 +1,175 @@
1
+ /**
2
+ * invite CLI command tests (VLT-7 US-002).
3
+ */
4
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
5
+ import { invite, listInvites, revokeInvite } from "./invite.js";
6
+ // ---------------------------------------------------------------------------
7
+ // Helpers
8
+ // ---------------------------------------------------------------------------
9
+ function jsonResponse(status, body) {
10
+ return new Response(JSON.stringify(body), {
11
+ status,
12
+ headers: { "Content-Type": "application/json" },
13
+ });
14
+ }
15
+ const VAULT_CONFIG = {
16
+ apiUrl: "https://vault.test.example.com",
17
+ authToken: "test-token",
18
+ };
19
+ let fetchSpy;
20
+ beforeEach(() => {
21
+ fetchSpy = vi.spyOn(globalThis, "fetch");
22
+ });
23
+ afterEach(() => {
24
+ vi.restoreAllMocks();
25
+ });
26
+ // ---------------------------------------------------------------------------
27
+ // invite()
28
+ // ---------------------------------------------------------------------------
29
+ describe("invite", () => {
30
+ it("creates invite for email target and returns magic link", async () => {
31
+ // First call: entity.findBySlug to resolve company
32
+ fetchSpy
33
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
34
+ // Second call: createInvite
35
+ .mockResolvedValueOnce(jsonResponse(200, {
36
+ membership: { membershipKey: "psn_1#cmp_abc", role: "member", status: "pending" },
37
+ inviteToken: "tok_secure123",
38
+ }));
39
+ const result = await invite({
40
+ target: "alice@example.com",
41
+ role: "member",
42
+ company: "acme",
43
+ vaultConfig: VAULT_CONFIG,
44
+ callerUid: "psn_admin",
45
+ });
46
+ expect(result.magicLink).toBe("hq://accept/tok_secure123");
47
+ expect(result.inviteToken).toBe("tok_secure123");
48
+ expect(result.membership.status).toBe("pending");
49
+ });
50
+ it("creates invite for person UID target", async () => {
51
+ // Company is already a UID — no entity lookup needed
52
+ fetchSpy.mockResolvedValueOnce(jsonResponse(200, {
53
+ membership: { membershipKey: "psn_bob#cmp_abc", role: "admin", status: "pending" },
54
+ inviteToken: "tok_456",
55
+ }));
56
+ const result = await invite({
57
+ target: "psn_bob",
58
+ role: "admin",
59
+ company: "cmp_abc",
60
+ vaultConfig: VAULT_CONFIG,
61
+ callerUid: "psn_admin",
62
+ });
63
+ expect(result.magicLink).toBe("hq://accept/tok_456");
64
+ // Should have called createInvite with personUid, not inviteeEmail
65
+ const body = JSON.parse(fetchSpy.mock.calls[0][1]?.body);
66
+ expect(body.personUid).toBe("psn_bob");
67
+ expect(body.inviteeEmail).toBeUndefined();
68
+ });
69
+ it("rejects --paths without --role guest", async () => {
70
+ await expect(invite({
71
+ target: "alice@example.com",
72
+ role: "member",
73
+ paths: "docs/",
74
+ company: "acme",
75
+ vaultConfig: VAULT_CONFIG,
76
+ callerUid: "psn_admin",
77
+ })).rejects.toThrow("--paths is only valid with --role guest");
78
+ });
79
+ it("allows --paths with --role guest", async () => {
80
+ fetchSpy
81
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
82
+ .mockResolvedValueOnce(jsonResponse(200, {
83
+ membership: { membershipKey: "psn_1#cmp_abc", role: "guest", status: "pending", allowedPrefixes: ["docs/", "shared/"] },
84
+ inviteToken: "tok_guest",
85
+ }));
86
+ const result = await invite({
87
+ target: "alice@example.com",
88
+ role: "guest",
89
+ paths: "docs/, shared/",
90
+ company: "acme",
91
+ vaultConfig: VAULT_CONFIG,
92
+ callerUid: "psn_admin",
93
+ });
94
+ expect(result.membership.allowedPrefixes).toEqual(["docs/", "shared/"]);
95
+ // Verify allowedPrefixes sent to API
96
+ const body = JSON.parse(fetchSpy.mock.calls[1][1]?.body);
97
+ expect(body.allowedPrefixes).toEqual(["docs/", "shared/"]);
98
+ });
99
+ it("maps VaultPermissionDeniedError to human-readable message", async () => {
100
+ fetchSpy
101
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
102
+ .mockResolvedValueOnce(jsonResponse(403, { message: "Admin required" }));
103
+ await expect(invite({
104
+ target: "alice@example.com",
105
+ company: "acme",
106
+ vaultConfig: VAULT_CONFIG,
107
+ callerUid: "psn_member",
108
+ })).rejects.toThrow("Permission denied — only admins and owners can invite members");
109
+ });
110
+ it("throws when no company specified", async () => {
111
+ await expect(invite({
112
+ target: "alice@example.com",
113
+ vaultConfig: VAULT_CONFIG,
114
+ callerUid: "psn_admin",
115
+ })).rejects.toThrow("No company specified");
116
+ });
117
+ it("maps VaultConflictError to human-readable message", async () => {
118
+ fetchSpy
119
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
120
+ .mockResolvedValueOnce(jsonResponse(409, { message: "Already exists" }));
121
+ await expect(invite({
122
+ target: "alice@example.com",
123
+ company: "acme",
124
+ vaultConfig: VAULT_CONFIG,
125
+ callerUid: "psn_admin",
126
+ })).rejects.toThrow("already has a membership or pending invite");
127
+ });
128
+ });
129
+ // ---------------------------------------------------------------------------
130
+ // listInvites()
131
+ // ---------------------------------------------------------------------------
132
+ describe("listInvites", () => {
133
+ it("returns pending invites for a company", async () => {
134
+ fetchSpy
135
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
136
+ .mockResolvedValueOnce(jsonResponse(200, {
137
+ invites: [
138
+ { membershipKey: "psn_1#cmp_abc", status: "pending", role: "member" },
139
+ { membershipKey: "psn_2#cmp_abc", status: "pending", role: "guest" },
140
+ ],
141
+ }));
142
+ const invites = await listInvites({
143
+ company: "acme",
144
+ vaultConfig: VAULT_CONFIG,
145
+ callerUid: "psn_admin",
146
+ });
147
+ expect(invites).toHaveLength(2);
148
+ });
149
+ });
150
+ // ---------------------------------------------------------------------------
151
+ // revokeInvite()
152
+ // ---------------------------------------------------------------------------
153
+ describe("revokeInvite", () => {
154
+ it("revokes a pending invite", async () => {
155
+ fetchSpy
156
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
157
+ .mockResolvedValueOnce(new Response(null, { status: 204 }));
158
+ await expect(revokeInvite({
159
+ tokenOrKey: "psn_1#cmp_abc",
160
+ company: "acme",
161
+ vaultConfig: VAULT_CONFIG,
162
+ })).resolves.toBeUndefined();
163
+ });
164
+ it("maps 404 to human-readable message", async () => {
165
+ fetchSpy
166
+ .mockResolvedValueOnce(jsonResponse(200, { entity: { uid: "cmp_abc", slug: "acme", type: "company", status: "active" } }))
167
+ .mockResolvedValueOnce(jsonResponse(404, { message: "Not found" }));
168
+ await expect(revokeInvite({
169
+ tokenOrKey: "psn_1#cmp_abc",
170
+ company: "acme",
171
+ vaultConfig: VAULT_CONFIG,
172
+ })).rejects.toThrow("Invite not found");
173
+ });
174
+ });
175
+ //# sourceMappingURL=invite.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"invite.test.js","sourceRoot":"","sources":["../../src/cli/invite.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,30 @@
1
+ /**
2
+ * `hq promote` command — change an existing member's role (VLT-7 US-003).
3
+ *
4
+ * Admin+ only. Surfaces last-owner demotion errors as human messages.
5
+ */
6
+ import type { VaultServiceConfig } from "../types.js";
7
+ import type { MembershipRole, Membership } from "../vault-client.js";
8
+ export interface PromoteOptions {
9
+ /** Person slug or UID of the member to promote */
10
+ target: string;
11
+ /** New role to assign */
12
+ newRole: MembershipRole;
13
+ /** Allowed prefixes (only valid with guest role) */
14
+ paths?: string;
15
+ /** Company slug or UID */
16
+ company?: string;
17
+ /** Caller's person UID */
18
+ callerUid: string;
19
+ /** Vault service config */
20
+ vaultConfig: VaultServiceConfig;
21
+ }
22
+ export interface PromoteResult {
23
+ membership: Membership;
24
+ previousRole?: MembershipRole;
25
+ }
26
+ /**
27
+ * Change a member's role.
28
+ */
29
+ export declare function promote(options: PromoteOptions): Promise<PromoteResult>;
30
+ //# sourceMappingURL=promote.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"promote.d.ts","sourceRoot":"","sources":["../../src/cli/promote.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAQtD,OAAO,KAAK,EAAE,cAAc,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAErE,MAAM,WAAW,cAAc;IAC7B,kDAAkD;IAClD,MAAM,EAAE,MAAM,CAAC;IACf,yBAAyB;IACzB,OAAO,EAAE,cAAc,CAAC;IACxB,oDAAoD;IACpD,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,0BAA0B;IAC1B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,0BAA0B;IAC1B,SAAS,EAAE,MAAM,CAAC;IAClB,2BAA2B;IAC3B,WAAW,EAAE,kBAAkB,CAAC;CACjC;AAED,MAAM,WAAW,aAAa;IAC5B,UAAU,EAAE,UAAU,CAAC;IACvB,YAAY,CAAC,EAAE,cAAc,CAAC;CAC/B;AAED;;GAEG;AACH,wBAAsB,OAAO,CAAC,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,aAAa,CAAC,CA6C7E"}
@@ -0,0 +1,79 @@
1
+ /**
2
+ * `hq promote` command — change an existing member's role (VLT-7 US-003).
3
+ *
4
+ * Admin+ only. Surfaces last-owner demotion errors as human messages.
5
+ */
6
+ import { VaultClient, VaultAuthError, VaultPermissionDeniedError, VaultNotFoundError, VaultConflictError, } from "../vault-client.js";
7
+ /**
8
+ * Change a member's role.
9
+ */
10
+ export async function promote(options) {
11
+ const { target, newRole, paths, company, callerUid, vaultConfig } = options;
12
+ // Validate: --paths only with guest role
13
+ if (paths && newRole !== "guest") {
14
+ throw new Error("--paths is only valid with --role guest (allowedPrefixes are only meaningful for the guest role)");
15
+ }
16
+ const client = new VaultClient(vaultConfig);
17
+ // Resolve company UID
18
+ const companyUid = await resolveCompanyUid(client, company);
19
+ // Build membership key from target + company
20
+ const membershipKey = buildMembershipKey(target, companyUid);
21
+ const allowedPrefixes = paths
22
+ ? paths.split(",").map((p) => p.trim()).filter(Boolean)
23
+ : undefined;
24
+ try {
25
+ const membership = await client.updateRole({
26
+ membershipKey,
27
+ newRole,
28
+ allowedPrefixes,
29
+ updaterUid: callerUid,
30
+ companyUid,
31
+ });
32
+ return { membership };
33
+ }
34
+ catch (err) {
35
+ if (err instanceof VaultAuthError) {
36
+ throw new Error("Authentication failed — run `hq auth` to refresh your session");
37
+ }
38
+ if (err instanceof VaultPermissionDeniedError) {
39
+ throw new Error("Permission denied — only admins and owners can change member roles");
40
+ }
41
+ if (err instanceof VaultNotFoundError) {
42
+ throw new Error(`Member "${target}" not found in this company`);
43
+ }
44
+ if (err instanceof VaultConflictError) {
45
+ throw new Error("Cannot leave company without an owner — promote another member to owner first");
46
+ }
47
+ throw err;
48
+ }
49
+ }
50
+ // ---------------------------------------------------------------------------
51
+ // Helpers
52
+ // ---------------------------------------------------------------------------
53
+ function buildMembershipKey(personRef, companyUid) {
54
+ // If already a composite key, use as-is
55
+ if (personRef.includes("#")) {
56
+ return personRef;
57
+ }
58
+ // Build composite key: personUid#companyUid
59
+ return `${personRef}#${companyUid}`;
60
+ }
61
+ async function resolveCompanyUid(client, companyRef) {
62
+ if (!companyRef) {
63
+ throw new Error("No company specified. Use --company <slug> or set up .hq/config.json");
64
+ }
65
+ if (companyRef.startsWith("cmp_")) {
66
+ return companyRef;
67
+ }
68
+ try {
69
+ const entity = await client.entity.findBySlug("company", companyRef);
70
+ return entity.uid;
71
+ }
72
+ catch (err) {
73
+ if (err instanceof VaultNotFoundError) {
74
+ throw new Error(`Company "${companyRef}" not found in the vault registry`);
75
+ }
76
+ throw err;
77
+ }
78
+ }
79
+ //# sourceMappingURL=promote.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"promote.js","sourceRoot":"","sources":["../../src/cli/promote.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ /**
2
+ * `hq share` command — selective push to entity vault (VLT-5 US-002).
3
+ *
4
+ * Broadcasts local file(s) to the company's S3 vault bucket.
5
+ * Refuses to overwrite a newer remote version without prompting.
6
+ */
7
+ import type { VaultServiceConfig } from "../types.js";
8
+ import type { ConflictStrategy } from "./conflict.js";
9
+ export interface ShareOptions {
10
+ /** Path(s) to share (files or directories) */
11
+ paths: string[];
12
+ /** Company slug or UID (defaults to active company from config) */
13
+ company?: string;
14
+ /** Optional message attached to journal entries */
15
+ message?: string;
16
+ /** Non-interactive conflict strategy */
17
+ onConflict?: ConflictStrategy;
18
+ /** Vault service config */
19
+ vaultConfig: VaultServiceConfig;
20
+ /** HQ root directory */
21
+ hqRoot: string;
22
+ }
23
+ export interface ShareResult {
24
+ filesUploaded: number;
25
+ bytesUploaded: number;
26
+ filesSkipped: number;
27
+ aborted: boolean;
28
+ }
29
+ /**
30
+ * Share local file(s) to the entity vault.
31
+ */
32
+ export declare function share(options: ShareOptions): Promise<ShareResult>;
33
+ //# sourceMappingURL=share.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"share.d.ts","sourceRoot":"","sources":["../../src/cli/share.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAMtD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AAEtD,MAAM,WAAW,YAAY;IAC3B,8CAA8C;IAC9C,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,mEAAmE;IACnE,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,mDAAmD;IACnD,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,wCAAwC;IACxC,UAAU,CAAC,EAAE,gBAAgB,CAAC;IAC9B,2BAA2B;IAC3B,WAAW,EAAE,kBAAkB,CAAC;IAChC,wBAAwB;IACxB,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,WAAW;IAC1B,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;IACrB,OAAO,EAAE,OAAO,CAAC;CAClB;AAED;;GAEG;AACH,wBAAsB,KAAK,CAAC,OAAO,EAAE,YAAY,GAAG,OAAO,CAAC,WAAW,CAAC,CA+FvE"}
@@ -0,0 +1,153 @@
1
+ /**
2
+ * `hq share` command — selective push to entity vault (VLT-5 US-002).
3
+ *
4
+ * Broadcasts local file(s) to the company's S3 vault bucket.
5
+ * Refuses to overwrite a newer remote version without prompting.
6
+ */
7
+ import * as fs from "fs";
8
+ import * as path from "path";
9
+ import { resolveEntityContext, isExpiringSoon, refreshEntityContext } from "../context.js";
10
+ import { uploadFile, headRemoteFile } from "../s3.js";
11
+ import { readJournal, writeJournal, hashFile, updateEntry } from "../journal.js";
12
+ import { createIgnoreFilter, isWithinSizeLimit } from "../ignore.js";
13
+ import { resolveConflict } from "./conflict.js";
14
+ /**
15
+ * Share local file(s) to the entity vault.
16
+ */
17
+ export async function share(options) {
18
+ const { paths, company, message, onConflict, vaultConfig, hqRoot } = options;
19
+ // Resolve company — slug, UID, or from active config
20
+ const companyRef = company ?? resolveActiveCompany(hqRoot);
21
+ if (!companyRef) {
22
+ throw new Error("No company specified and no active company found. " +
23
+ "Use --company <slug> or set up .hq/config.json.");
24
+ }
25
+ // Resolve entity context (handles STS vending + caching)
26
+ let ctx = await resolveEntityContext(companyRef, vaultConfig);
27
+ const shouldSync = createIgnoreFilter(hqRoot);
28
+ const journal = readJournal(hqRoot);
29
+ let filesUploaded = 0;
30
+ let bytesUploaded = 0;
31
+ let filesSkipped = 0;
32
+ // Collect all files to share
33
+ const filesToShare = collectFiles(paths, hqRoot, shouldSync);
34
+ for (const { absolutePath, relativePath } of filesToShare) {
35
+ if (!isWithinSizeLimit(absolutePath)) {
36
+ console.error(` Skipped (too large): ${relativePath}`);
37
+ filesSkipped++;
38
+ continue;
39
+ }
40
+ // Auto-refresh context if credentials expiring
41
+ if (isExpiringSoon(ctx.expiresAt)) {
42
+ ctx = await refreshEntityContext(companyRef, vaultConfig);
43
+ }
44
+ // Check for remote conflict — refuse to overwrite newer remote version
45
+ const remoteMeta = await headRemoteFile(ctx, relativePath);
46
+ if (remoteMeta) {
47
+ const journalEntry = journal.files[relativePath];
48
+ const localHash = hashFile(absolutePath);
49
+ // If remote has changed since our last sync, it's a conflict
50
+ if (journalEntry && journalEntry.hash !== localHash) {
51
+ // Local has changes — check if remote also changed
52
+ const resolution = await resolveConflict({
53
+ path: relativePath,
54
+ localHash,
55
+ remoteModified: remoteMeta.lastModified,
56
+ direction: "push",
57
+ }, onConflict);
58
+ if (resolution === "abort") {
59
+ return { filesUploaded, bytesUploaded, filesSkipped, aborted: true };
60
+ }
61
+ if (resolution === "keep" || resolution === "skip") {
62
+ filesSkipped++;
63
+ continue;
64
+ }
65
+ // "overwrite" falls through to upload
66
+ }
67
+ }
68
+ // Upload
69
+ try {
70
+ const stat = fs.statSync(absolutePath);
71
+ const hash = hashFile(absolutePath);
72
+ await uploadFile(ctx, absolutePath, relativePath);
73
+ // Update journal with optional message
74
+ updateEntry(journal, relativePath, hash, stat.size, "up");
75
+ if (message) {
76
+ journal.files[relativePath] = {
77
+ ...journal.files[relativePath],
78
+ message,
79
+ };
80
+ }
81
+ filesUploaded++;
82
+ bytesUploaded += stat.size;
83
+ console.log(` ✓ ${relativePath}`);
84
+ }
85
+ catch (err) {
86
+ console.error(` ✗ ${relativePath} — ${err instanceof Error ? err.message : err}`);
87
+ }
88
+ }
89
+ writeJournal(hqRoot, journal);
90
+ return { filesUploaded, bytesUploaded, filesSkipped, aborted: false };
91
+ }
92
+ /**
93
+ * Resolve active company from .hq/config.json or parent directory chain.
94
+ */
95
+ function resolveActiveCompany(hqRoot) {
96
+ const configPath = path.join(hqRoot, ".hq", "config.json");
97
+ if (fs.existsSync(configPath)) {
98
+ try {
99
+ const config = JSON.parse(fs.readFileSync(configPath, "utf-8"));
100
+ return config.activeCompany ?? config.companySlug;
101
+ }
102
+ catch {
103
+ // Ignore parse errors
104
+ }
105
+ }
106
+ return undefined;
107
+ }
108
+ /**
109
+ * Collect files from paths (expanding directories recursively).
110
+ */
111
+ function collectFiles(paths, hqRoot, filter) {
112
+ const results = [];
113
+ for (const p of paths) {
114
+ const absolutePath = path.isAbsolute(p) ? p : path.resolve(hqRoot, p);
115
+ if (!fs.existsSync(absolutePath)) {
116
+ console.error(` Warning: ${p} does not exist, skipping.`);
117
+ continue;
118
+ }
119
+ const stat = fs.statSync(absolutePath);
120
+ if (stat.isDirectory()) {
121
+ results.push(...walkDir(absolutePath, hqRoot, filter));
122
+ }
123
+ else if (stat.isFile()) {
124
+ const relativePath = path.relative(hqRoot, absolutePath);
125
+ if (filter(absolutePath)) {
126
+ results.push({ absolutePath, relativePath });
127
+ }
128
+ }
129
+ }
130
+ return results;
131
+ }
132
+ function walkDir(dir, root, filter) {
133
+ const results = [];
134
+ if (!fs.existsSync(dir))
135
+ return results;
136
+ const entries = fs.readdirSync(dir, { withFileTypes: true });
137
+ for (const entry of entries) {
138
+ const absolutePath = path.join(dir, entry.name);
139
+ if (!filter(absolutePath))
140
+ continue;
141
+ if (entry.isDirectory()) {
142
+ results.push(...walkDir(absolutePath, root, filter));
143
+ }
144
+ else if (entry.isFile()) {
145
+ results.push({
146
+ absolutePath,
147
+ relativePath: path.relative(root, absolutePath),
148
+ });
149
+ }
150
+ }
151
+ return results;
152
+ }
153
+ //# sourceMappingURL=share.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"share.js","sourceRoot":"","sources":["../../src/cli/share.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Unit tests for hq share command (VLT-5 US-002).
3
+ */
4
+ export {};
5
+ //# sourceMappingURL=share.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"share.test.d.ts","sourceRoot":"","sources":["../../src/cli/share.test.ts"],"names":[],"mappings":"AAAA;;GAEG"}
@@ -0,0 +1,121 @@
1
+ /**
2
+ * Unit tests for hq share command (VLT-5 US-002).
3
+ */
4
+ import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
5
+ import * as fs from "fs";
6
+ import * as path from "path";
7
+ import * as os from "os";
8
+ import { clearContextCache } from "../context.js";
9
+ // Mock s3 module at the top level
10
+ vi.mock("../s3.js", () => ({
11
+ uploadFile: vi.fn().mockResolvedValue(undefined),
12
+ downloadFile: vi.fn().mockResolvedValue(undefined),
13
+ listRemoteFiles: vi.fn().mockResolvedValue([]),
14
+ deleteRemoteFile: vi.fn().mockResolvedValue(undefined),
15
+ headRemoteFile: vi.fn().mockResolvedValue(null),
16
+ }));
17
+ import { share } from "./share.js";
18
+ import { headRemoteFile } from "../s3.js";
19
+ const mockConfig = {
20
+ apiUrl: "https://vault-api.test",
21
+ authToken: "test-jwt-token",
22
+ region: "us-east-1",
23
+ };
24
+ const mockEntity = {
25
+ uid: "cmp_01ABCDEF",
26
+ slug: "acme",
27
+ bucketName: "hq-vault-acme-123",
28
+ status: "active",
29
+ };
30
+ const mockVendResponse = {
31
+ credentials: {
32
+ accessKeyId: "ASIA_TEST_KEY",
33
+ secretAccessKey: "test-secret",
34
+ sessionToken: "test-session-token",
35
+ expiration: new Date(Date.now() + 15 * 60 * 1000).toISOString(),
36
+ },
37
+ expiresAt: new Date(Date.now() + 15 * 60 * 1000).toISOString(),
38
+ };
39
+ function setupFetchMock() {
40
+ const fetchMock = vi.fn().mockImplementation(async (url) => {
41
+ const urlStr = String(url);
42
+ if (urlStr.includes("/entity/by-slug/")) {
43
+ return { ok: true, status: 200, json: async () => ({ entity: mockEntity }), text: async () => "" };
44
+ }
45
+ if (urlStr.includes("/sts/vend")) {
46
+ return { ok: true, status: 200, json: async () => mockVendResponse, text: async () => "" };
47
+ }
48
+ return { ok: false, status: 404, text: async () => "Not found" };
49
+ });
50
+ vi.stubGlobal("fetch", fetchMock);
51
+ return fetchMock;
52
+ }
53
+ describe("share", () => {
54
+ let tmpDir;
55
+ beforeEach(() => {
56
+ clearContextCache();
57
+ tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), "hq-share-test-"));
58
+ setupFetchMock();
59
+ vi.mocked(headRemoteFile).mockResolvedValue(null);
60
+ });
61
+ afterEach(() => {
62
+ vi.restoreAllMocks();
63
+ fs.rmSync(tmpDir, { recursive: true, force: true });
64
+ });
65
+ it("shares a single file", async () => {
66
+ const testFile = path.join(tmpDir, "test.md");
67
+ fs.writeFileSync(testFile, "# Hello World");
68
+ const result = await share({
69
+ paths: [testFile],
70
+ company: "acme",
71
+ vaultConfig: mockConfig,
72
+ hqRoot: tmpDir,
73
+ });
74
+ expect(result.filesUploaded).toBe(1);
75
+ expect(result.aborted).toBe(false);
76
+ });
77
+ it("respects ignore rules", async () => {
78
+ fs.mkdirSync(path.join(tmpDir, ".git"));
79
+ fs.writeFileSync(path.join(tmpDir, ".git", "config"), "git config");
80
+ fs.writeFileSync(path.join(tmpDir, "readme.md"), "readme");
81
+ const result = await share({
82
+ paths: [tmpDir],
83
+ company: "acme",
84
+ vaultConfig: mockConfig,
85
+ hqRoot: tmpDir,
86
+ });
87
+ expect(result.filesUploaded).toBe(1);
88
+ });
89
+ it("shares a directory of files", async () => {
90
+ fs.mkdirSync(path.join(tmpDir, "docs"));
91
+ fs.writeFileSync(path.join(tmpDir, "docs", "a.md"), "doc a");
92
+ fs.writeFileSync(path.join(tmpDir, "docs", "b.md"), "doc b");
93
+ const result = await share({
94
+ paths: [path.join(tmpDir, "docs")],
95
+ company: "acme",
96
+ vaultConfig: mockConfig,
97
+ hqRoot: tmpDir,
98
+ });
99
+ expect(result.filesUploaded).toBe(2);
100
+ });
101
+ it("throws when no company specified and no active company", async () => {
102
+ fs.writeFileSync(path.join(tmpDir, "test.md"), "test");
103
+ await expect(share({
104
+ paths: [path.join(tmpDir, "test.md")],
105
+ vaultConfig: mockConfig,
106
+ hqRoot: tmpDir,
107
+ })).rejects.toThrow(/No company specified/);
108
+ });
109
+ it("resolves active company from .hq/config.json", async () => {
110
+ fs.mkdirSync(path.join(tmpDir, ".hq"), { recursive: true });
111
+ fs.writeFileSync(path.join(tmpDir, ".hq", "config.json"), JSON.stringify({ activeCompany: "acme" }));
112
+ fs.writeFileSync(path.join(tmpDir, "test.md"), "test");
113
+ const result = await share({
114
+ paths: [path.join(tmpDir, "test.md")],
115
+ vaultConfig: mockConfig,
116
+ hqRoot: tmpDir,
117
+ });
118
+ expect(result.filesUploaded).toBe(1);
119
+ });
120
+ });
121
+ //# sourceMappingURL=share.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"share.test.js","sourceRoot":"","sources":["../../src/cli/share.test.ts"],"names":[],"mappings":"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"}