@tellescope/sdk 1.252.0 → 1.252.2

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 (60) hide show
  1. package/lib/cjs/tests/api_tests/calendar_canvas_coding_clear.test.d.ts +6 -0
  2. package/lib/cjs/tests/api_tests/calendar_canvas_coding_clear.test.d.ts.map +1 -0
  3. package/lib/cjs/tests/api_tests/calendar_canvas_coding_clear.test.js +139 -0
  4. package/lib/cjs/tests/api_tests/calendar_canvas_coding_clear.test.js.map +1 -0
  5. package/lib/cjs/tests/api_tests/integrations_redacted.test.d.ts.map +1 -1
  6. package/lib/cjs/tests/api_tests/integrations_redacted.test.js +30 -20
  7. package/lib/cjs/tests/api_tests/integrations_redacted.test.js.map +1 -1
  8. package/lib/cjs/tests/api_tests/mdi_webhooks.test.d.ts +22 -0
  9. package/lib/cjs/tests/api_tests/mdi_webhooks.test.d.ts.map +1 -0
  10. package/lib/cjs/tests/api_tests/mdi_webhooks.test.js +162 -0
  11. package/lib/cjs/tests/api_tests/mdi_webhooks.test.js.map +1 -0
  12. package/lib/cjs/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.d.ts +32 -0
  13. package/lib/cjs/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.d.ts.map +1 -0
  14. package/lib/cjs/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.js +237 -0
  15. package/lib/cjs/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.js.map +1 -0
  16. package/lib/cjs/tests/api_tests/security/F-0076-self-admin-role-assignment.test.d.ts +38 -0
  17. package/lib/cjs/tests/api_tests/security/F-0076-self-admin-role-assignment.test.d.ts.map +1 -0
  18. package/lib/cjs/tests/api_tests/security/F-0076-self-admin-role-assignment.test.js +222 -0
  19. package/lib/cjs/tests/api_tests/security/F-0076-self-admin-role-assignment.test.js.map +1 -0
  20. package/lib/cjs/tests/api_tests/user_portal_settings.test.d.ts +6 -0
  21. package/lib/cjs/tests/api_tests/user_portal_settings.test.d.ts.map +1 -0
  22. package/lib/cjs/tests/api_tests/user_portal_settings.test.js +301 -0
  23. package/lib/cjs/tests/api_tests/user_portal_settings.test.js.map +1 -0
  24. package/lib/cjs/tests/tests.d.ts.map +1 -1
  25. package/lib/cjs/tests/tests.js +154 -142
  26. package/lib/cjs/tests/tests.js.map +1 -1
  27. package/lib/esm/tests/api_tests/calendar_canvas_coding_clear.test.d.ts +6 -0
  28. package/lib/esm/tests/api_tests/calendar_canvas_coding_clear.test.d.ts.map +1 -0
  29. package/lib/esm/tests/api_tests/calendar_canvas_coding_clear.test.js +135 -0
  30. package/lib/esm/tests/api_tests/calendar_canvas_coding_clear.test.js.map +1 -0
  31. package/lib/esm/tests/api_tests/integrations_redacted.test.d.ts.map +1 -1
  32. package/lib/esm/tests/api_tests/integrations_redacted.test.js +30 -20
  33. package/lib/esm/tests/api_tests/integrations_redacted.test.js.map +1 -1
  34. package/lib/esm/tests/api_tests/mdi_webhooks.test.d.ts +22 -0
  35. package/lib/esm/tests/api_tests/mdi_webhooks.test.d.ts.map +1 -0
  36. package/lib/esm/tests/api_tests/mdi_webhooks.test.js +158 -0
  37. package/lib/esm/tests/api_tests/mdi_webhooks.test.js.map +1 -0
  38. package/lib/esm/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.d.ts +32 -0
  39. package/lib/esm/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.d.ts.map +1 -0
  40. package/lib/esm/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.js +233 -0
  41. package/lib/esm/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.js.map +1 -0
  42. package/lib/esm/tests/api_tests/security/F-0076-self-admin-role-assignment.test.d.ts +38 -0
  43. package/lib/esm/tests/api_tests/security/F-0076-self-admin-role-assignment.test.d.ts.map +1 -0
  44. package/lib/esm/tests/api_tests/security/F-0076-self-admin-role-assignment.test.js +218 -0
  45. package/lib/esm/tests/api_tests/security/F-0076-self-admin-role-assignment.test.js.map +1 -0
  46. package/lib/esm/tests/api_tests/user_portal_settings.test.d.ts +6 -0
  47. package/lib/esm/tests/api_tests/user_portal_settings.test.d.ts.map +1 -0
  48. package/lib/esm/tests/api_tests/user_portal_settings.test.js +297 -0
  49. package/lib/esm/tests/api_tests/user_portal_settings.test.js.map +1 -0
  50. package/lib/esm/tests/tests.d.ts.map +1 -1
  51. package/lib/esm/tests/tests.js +154 -142
  52. package/lib/esm/tests/tests.js.map +1 -1
  53. package/lib/tsconfig.tsbuildinfo +1 -1
  54. package/package.json +10 -10
  55. package/src/tests/api_tests/integrations_redacted.test.ts +8 -0
  56. package/src/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.ts +161 -0
  57. package/src/tests/api_tests/security/F-0076-self-admin-role-assignment.test.ts +165 -0
  58. package/src/tests/api_tests/user_portal_settings.test.ts +217 -0
  59. package/src/tests/tests.ts +6 -0
  60. package/test_generated.pdf +0 -0
@@ -0,0 +1,233 @@
1
+ var __assign = (this && this.__assign) || function () {
2
+ __assign = Object.assign || function(t) {
3
+ for (var s, i = 1, n = arguments.length; i < n; i++) {
4
+ s = arguments[i];
5
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
6
+ t[p] = s[p];
7
+ }
8
+ return t;
9
+ };
10
+ return __assign.apply(this, arguments);
11
+ };
12
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
13
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
14
+ return new (P || (P = Promise))(function (resolve, reject) {
15
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
16
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
17
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
18
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
19
+ });
20
+ };
21
+ var __generator = (this && this.__generator) || function (thisArg, body) {
22
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
23
+ return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
24
+ function verb(n) { return function (v) { return step([n, v]); }; }
25
+ function step(op) {
26
+ if (f) throw new TypeError("Generator is already executing.");
27
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
28
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
29
+ if (y = 0, t) op = [op[0] & 2, t.value];
30
+ switch (op[0]) {
31
+ case 0: case 1: t = op; break;
32
+ case 4: _.label++; return { value: op[1], done: false };
33
+ case 5: _.label++; y = op[1]; op = [0]; continue;
34
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
35
+ default:
36
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
37
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
38
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
39
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
40
+ if (t[2]) _.ops.pop();
41
+ _.trys.pop(); continue;
42
+ }
43
+ op = body.call(thisArg, _);
44
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
45
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
46
+ }
47
+ };
48
+ require('source-map-support').install();
49
+ import { Session } from "../../../sdk";
50
+ import { assert, log_header, wait, } from "@tellescope/testing";
51
+ import { setup_tests } from "../../setup";
52
+ import { PROVIDER_PERMISSIONS } from "@tellescope/constants";
53
+ var host = process.env.API_URL || 'http://localhost:8080';
54
+ // Separate tenant (different businessId), reusing the same hardcoded apiKey that
55
+ // multi_tenant_tests relies on in tests.ts.
56
+ var OTHER_TENANT_API_KEY = "ba745e25162bb95a795c5fa1af70df188d93c4d3aac9c48b34a5c8c9dd7b80f7";
57
+ /**
58
+ * Tenant-boundary guard for cascade_role_rename (relates to security-audit finding F-0053,
59
+ * which was investigated and closed as a FALSE POSITIVE — see that file for the code trace).
60
+ *
61
+ * The `cascade_role_rename` side-effect handler
62
+ * ([event_handlers_v2/role_based_access_permissions.ts](packages/private/api/api/v1/event_handlers_v2/role_based_access_permissions.ts))
63
+ * runs when a `role_based_access_permissions` doc's `role` field changes. It finds every user
64
+ * with the old role name (via `DBUnrestricted.users`) and rewrites their `roles` array, then
65
+ * deauthenticates them. F-0053 hypothesized this query was globally cross-tenant. It is NOT:
66
+ * `DBUnrestricted` bypasses per-user/per-role RBAC but is STILL scoped to the acting session's
67
+ * `businessId` (see `modifyFilterForAccessConstraint` injecting `{ businessId }` at
68
+ * database.ts:1761-1763, reached via the `unrestricted: true` branch at database.ts:2137-2144).
69
+ *
70
+ * This test locks that boundary in place so a future refactor of `DBUnrestricted` semantics
71
+ * can't silently turn the cascade into a cross-tenant write:
72
+ * 1. Tenant A creates a role `ROLE_OLD` and assigns it to a Tenant A user (positive control).
73
+ * 2. Tenant B (separate businessId) has a user whose roles include the SAME `ROLE_OLD`.
74
+ * 3. Tenant A renames the role `ROLE_OLD` -> `ROLE_NEW`.
75
+ * 4. Assert the Tenant B user's roles are UNCHANGED (still `[ROLE_OLD]`) <-- guards the tenant boundary.
76
+ * 5. Assert the Tenant A user's roles ARE renamed to `[ROLE_NEW]` <-- same-tenant cascade works.
77
+ *
78
+ * Expected on current (correct) code: BOTH assertions pass. A regression that drops the
79
+ * `businessId` scoping would flip assertion #4 to red (Tenant B user becomes `[ROLE_NEW]`).
80
+ *
81
+ * A collision-proof unique role name (timestamped) is used so the test never touches real roles.
82
+ */
83
+ export var cascade_role_rename_cross_tenant_tests = function (_a) {
84
+ var sdk = _a.sdk;
85
+ return __awaiter(void 0, void 0, void 0, function () {
86
+ var stamp, ROLE_OLD, ROLE_NEW, sdkOther, rbapId, controlUserId, victimUserId, tenantABusinessId, rbap, controlUser, victimUser, victimBefore, controlBefore, victimAfter, controlAfter, _b, _c, _d;
87
+ return __generator(this, function (_e) {
88
+ switch (_e.label) {
89
+ case 0:
90
+ log_header("F-0053: cascade role rename cross-tenant regression");
91
+ stamp = Date.now();
92
+ ROLE_OLD = "XTenantRename_".concat(stamp);
93
+ ROLE_NEW = "".concat(ROLE_OLD, "_renamed");
94
+ sdkOther = new Session({ host: host, apiKey: OTHER_TENANT_API_KEY });
95
+ _e.label = 1;
96
+ case 1:
97
+ _e.trys.push([1, , 13, 26]);
98
+ tenantABusinessId = sdk.userInfo.businessId;
99
+ return [4 /*yield*/, sdk.api.role_based_access_permissions.createOne({
100
+ role: ROLE_OLD,
101
+ permissions: __assign({}, PROVIDER_PERMISSIONS),
102
+ })];
103
+ case 2:
104
+ rbap = _e.sent();
105
+ rbapId = rbap.id;
106
+ return [4 /*yield*/, sdk.api.users.createOne({
107
+ email: "f0053-control-".concat(stamp, "@example.com"),
108
+ notificationEmailsDisabled: true,
109
+ })];
110
+ case 3:
111
+ controlUser = _e.sent();
112
+ controlUserId = controlUser.id;
113
+ return [4 /*yield*/, sdk.api.users.updateOne(controlUserId, { roles: [ROLE_OLD] }, { replaceObjectFields: true })
114
+ // 3. Tenant B: create a throwaway victim user holding the SAME ROLE_OLD.
115
+ ];
116
+ case 4:
117
+ _e.sent();
118
+ return [4 /*yield*/, sdkOther.api.users.createOne({
119
+ email: "f0053-victim-".concat(stamp, "@example.com"),
120
+ notificationEmailsDisabled: true,
121
+ })];
122
+ case 5:
123
+ victimUser = _e.sent();
124
+ victimUserId = victimUser.id;
125
+ return [4 /*yield*/, sdkOther.api.users.updateOne(victimUserId, { roles: [ROLE_OLD] }, { replaceObjectFields: true })
126
+ // 4. Setup sanity: tenants are distinct and both users actually hold ROLE_OLD before the rename.
127
+ // (Distinguishes a setup failure from the security assertion below.)
128
+ ];
129
+ case 6:
130
+ _e.sent();
131
+ // 4. Setup sanity: tenants are distinct and both users actually hold ROLE_OLD before the rename.
132
+ // (Distinguishes a setup failure from the security assertion below.)
133
+ assert(victimUser.businessId !== tenantABusinessId, "Victim user shares businessId with Tenant A (".concat(victimUser.businessId, ") \u2014 not a cross-tenant scenario"), 'F-0053 setup: tenants are distinct');
134
+ return [4 /*yield*/, sdkOther.api.users.getOne(victimUserId)];
135
+ case 7:
136
+ victimBefore = _e.sent();
137
+ return [4 /*yield*/, sdk.api.users.getOne(controlUserId)];
138
+ case 8:
139
+ controlBefore = _e.sent();
140
+ assert(JSON.stringify(victimBefore.roles) === JSON.stringify([ROLE_OLD])
141
+ && JSON.stringify(controlBefore.roles) === JSON.stringify([ROLE_OLD]), "Setup failed: expected both users to hold [".concat(ROLE_OLD, "] (victim=").concat(JSON.stringify(victimBefore.roles), ", control=").concat(JSON.stringify(controlBefore.roles), ")"), 'F-0053 setup: both users hold ROLE_OLD');
142
+ // 5. Tenant A renames the role. This triggers cascade_role_rename.
143
+ return [4 /*yield*/, sdk.api.role_based_access_permissions.updateOne(rbapId, { role: ROLE_NEW })];
144
+ case 9:
145
+ // 5. Tenant A renames the role. This triggers cascade_role_rename.
146
+ _e.sent();
147
+ return [4 /*yield*/, wait(undefined, 1500)
148
+ // 6. SECURITY ASSERTION — the Tenant B victim must be untouched by Tenant A's rename.
149
+ ]; // let the side effect run
150
+ case 10:
151
+ _e.sent(); // let the side effect run
152
+ return [4 /*yield*/, sdkOther.api.users.getOne(victimUserId)];
153
+ case 11:
154
+ victimAfter = _e.sent();
155
+ assert(JSON.stringify(victimAfter.roles) === JSON.stringify([ROLE_OLD]), "CROSS-TENANT LEAK: Tenant B victim roles changed to ".concat(JSON.stringify(victimAfter.roles), " ")
156
+ + "after Tenant A renamed its role. Expected [".concat(ROLE_OLD, "]."), 'F-0053: Tenant B user roles unaffected by other-tenant role rename');
157
+ return [4 /*yield*/, sdk.api.users.getOne(controlUserId)];
158
+ case 12:
159
+ controlAfter = _e.sent();
160
+ assert(JSON.stringify(controlAfter.roles) === JSON.stringify([ROLE_NEW]), "Same-tenant cascade broken: Tenant A control roles are ".concat(JSON.stringify(controlAfter.roles), ", ")
161
+ + "expected [".concat(ROLE_NEW, "]."), 'F-0053: Tenant A user roles renamed by same-tenant cascade');
162
+ return [3 /*break*/, 26];
163
+ case 13:
164
+ if (!victimUserId) return [3 /*break*/, 17];
165
+ _e.label = 14;
166
+ case 14:
167
+ _e.trys.push([14, 16, , 17]);
168
+ return [4 /*yield*/, sdkOther.api.users.deleteOne(victimUserId)];
169
+ case 15:
170
+ _e.sent();
171
+ return [3 /*break*/, 17];
172
+ case 16:
173
+ _b = _e.sent();
174
+ return [3 /*break*/, 17];
175
+ case 17:
176
+ if (!controlUserId) return [3 /*break*/, 21];
177
+ _e.label = 18;
178
+ case 18:
179
+ _e.trys.push([18, 20, , 21]);
180
+ return [4 /*yield*/, sdk.api.users.deleteOne(controlUserId)];
181
+ case 19:
182
+ _e.sent();
183
+ return [3 /*break*/, 21];
184
+ case 20:
185
+ _c = _e.sent();
186
+ return [3 /*break*/, 21];
187
+ case 21:
188
+ if (!rbapId) return [3 /*break*/, 25];
189
+ _e.label = 22;
190
+ case 22:
191
+ _e.trys.push([22, 24, , 25]);
192
+ return [4 /*yield*/, sdk.api.role_based_access_permissions.deleteOne(rbapId)];
193
+ case 23:
194
+ _e.sent();
195
+ return [3 /*break*/, 25];
196
+ case 24:
197
+ _d = _e.sent();
198
+ return [3 /*break*/, 25];
199
+ case 25: return [7 /*endfinally*/];
200
+ case 26: return [2 /*return*/];
201
+ }
202
+ });
203
+ });
204
+ };
205
+ // Allow running this test file independently
206
+ if (require.main === module) {
207
+ console.log("\uD83C\uDF10 Using API URL: ".concat(host));
208
+ var sdk_1 = new Session({ host: host });
209
+ var sdkNonAdmin_1 = new Session({ host: host });
210
+ var runTests = function () { return __awaiter(void 0, void 0, void 0, function () {
211
+ return __generator(this, function (_a) {
212
+ switch (_a.label) {
213
+ case 0: return [4 /*yield*/, setup_tests(sdk_1, sdkNonAdmin_1)];
214
+ case 1:
215
+ _a.sent();
216
+ return [4 /*yield*/, cascade_role_rename_cross_tenant_tests({ sdk: sdk_1, sdkNonAdmin: sdkNonAdmin_1 })];
217
+ case 2:
218
+ _a.sent();
219
+ return [2 /*return*/];
220
+ }
221
+ });
222
+ }); };
223
+ runTests()
224
+ .then(function () {
225
+ console.log("✅ F-0053 cascade role rename cross-tenant test suite completed successfully");
226
+ process.exit(0);
227
+ })
228
+ .catch(function (error) {
229
+ console.error("❌ F-0053 cascade role rename cross-tenant test suite failed:", error);
230
+ process.exit(1);
231
+ });
232
+ }
233
+ //# sourceMappingURL=F-0053-cascade-role-rename-cross-tenant.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"F-0053-cascade-role-rename-cross-tenant.test.js","sourceRoot":"","sources":["../../../../../src/tests/api_tests/security/F-0053-cascade-role-rename-cross-tenant.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,38 @@
1
+ import { Session } from "../../../sdk";
2
+ /**
3
+ * Self-role privilege-escalation guard (relates to security-audit finding F-0076, which was
4
+ * investigated and closed as a FALSE POSITIVE — see that file for the full code trace).
5
+ *
6
+ * F-0076 hypothesized that a non-admin staff user could `PATCH /v1/users/{their-own-id}` with
7
+ * `{ roles: ['Admin'] }` and self-promote to Admin, because the FIRST `users` relationship
8
+ * constraint ("Only admin users can set the admin role",
9
+ * [schema.ts:3446](packages/public/schema/src/schema.ts#L3446)) has a self-exception
10
+ * (`if (_id === session.id) return`).
11
+ *
12
+ * That analysis missed the SECOND constraint, "Only admin users can update user roles"
13
+ * ([schema.ts:3486](packages/public/schema/src/schema.ts#L3486)), which has NO self-exception.
14
+ * Relationship constraints are AND-evaluated — `validateRelationshipConstraints`
15
+ * ([routing.ts:1240-1252](packages/private/api/api/modules/routing.ts#L1240)) loops the whole
16
+ * array and throws 400 on the FIRST evaluator that returns a string. So a non-admin self-update
17
+ * that includes `roles` passes constraint #1 (self-exception) but is rejected by constraint #2.
18
+ * The self-promotion is blocked.
19
+ *
20
+ * This test locks that boundary in place so a future refactor of the role constraints can't
21
+ * silently reintroduce the escalation. A dedicated throwaway non-admin user is used as the
22
+ * "attacker" (we never mutate the shared sdkNonAdmin's roles):
23
+ * 1. Admin creates a throwaway user and assigns it a non-admin role (`['Provider']`).
24
+ * 2. Authenticate AS that user via a freshly-minted auth token.
25
+ * 3. As the attacker, attempt four self-role mutations — ['Admin'], ['Provider','Admin'],
26
+ * an arbitrary role, and [] — and assert EACH is blocked. <-- the security assertions
27
+ * 4. Confirm (as admin) the attacker's roles are still ['Provider'] — nothing slipped through.
28
+ * 5. Positive control: admin CAN update the throwaway user's roles. <-- guards against an
29
+ * over-restrictive regression that would block legitimate admin role management.
30
+ *
31
+ * Expected on current (correct) code: all assertions pass. A regression that made the self-update
32
+ * path role-writable by non-admins would flip the step-3/step-4 assertions to red.
33
+ */
34
+ export declare const self_admin_role_assignment_tests: ({ sdk }: {
35
+ sdk: Session;
36
+ sdkNonAdmin: Session;
37
+ }) => Promise<void>;
38
+ //# sourceMappingURL=F-0076-self-admin-role-assignment.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"F-0076-self-admin-role-assignment.test.d.ts","sourceRoot":"","sources":["../../../../../src/tests/api_tests/security/F-0076-self-admin-role-assignment.test.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAA;AAUtC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AACH,eAAO,MAAM,gCAAgC;SAA2B,OAAO;iBAAe,OAAO;mBAkGpG,CAAA"}
@@ -0,0 +1,218 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ var __generator = (this && this.__generator) || function (thisArg, body) {
11
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
12
+ return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
13
+ function verb(n) { return function (v) { return step([n, v]); }; }
14
+ function step(op) {
15
+ if (f) throw new TypeError("Generator is already executing.");
16
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
17
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
18
+ if (y = 0, t) op = [op[0] & 2, t.value];
19
+ switch (op[0]) {
20
+ case 0: case 1: t = op; break;
21
+ case 4: _.label++; return { value: op[1], done: false };
22
+ case 5: _.label++; y = op[1]; op = [0]; continue;
23
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
24
+ default:
25
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
26
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
27
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
28
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
29
+ if (t[2]) _.ops.pop();
30
+ _.trys.pop(); continue;
31
+ }
32
+ op = body.call(thisArg, _);
33
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
34
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
+ }
36
+ };
37
+ require('source-map-support').install();
38
+ import { Session } from "../../../sdk";
39
+ import { assert, log_header, wait, } from "@tellescope/testing";
40
+ import { setup_tests } from "../../setup";
41
+ var host = process.env.API_URL || 'http://localhost:8080';
42
+ /**
43
+ * Self-role privilege-escalation guard (relates to security-audit finding F-0076, which was
44
+ * investigated and closed as a FALSE POSITIVE — see that file for the full code trace).
45
+ *
46
+ * F-0076 hypothesized that a non-admin staff user could `PATCH /v1/users/{their-own-id}` with
47
+ * `{ roles: ['Admin'] }` and self-promote to Admin, because the FIRST `users` relationship
48
+ * constraint ("Only admin users can set the admin role",
49
+ * [schema.ts:3446](packages/public/schema/src/schema.ts#L3446)) has a self-exception
50
+ * (`if (_id === session.id) return`).
51
+ *
52
+ * That analysis missed the SECOND constraint, "Only admin users can update user roles"
53
+ * ([schema.ts:3486](packages/public/schema/src/schema.ts#L3486)), which has NO self-exception.
54
+ * Relationship constraints are AND-evaluated — `validateRelationshipConstraints`
55
+ * ([routing.ts:1240-1252](packages/private/api/api/modules/routing.ts#L1240)) loops the whole
56
+ * array and throws 400 on the FIRST evaluator that returns a string. So a non-admin self-update
57
+ * that includes `roles` passes constraint #1 (self-exception) but is rejected by constraint #2.
58
+ * The self-promotion is blocked.
59
+ *
60
+ * This test locks that boundary in place so a future refactor of the role constraints can't
61
+ * silently reintroduce the escalation. A dedicated throwaway non-admin user is used as the
62
+ * "attacker" (we never mutate the shared sdkNonAdmin's roles):
63
+ * 1. Admin creates a throwaway user and assigns it a non-admin role (`['Provider']`).
64
+ * 2. Authenticate AS that user via a freshly-minted auth token.
65
+ * 3. As the attacker, attempt four self-role mutations — ['Admin'], ['Provider','Admin'],
66
+ * an arbitrary role, and [] — and assert EACH is blocked. <-- the security assertions
67
+ * 4. Confirm (as admin) the attacker's roles are still ['Provider'] — nothing slipped through.
68
+ * 5. Positive control: admin CAN update the throwaway user's roles. <-- guards against an
69
+ * over-restrictive regression that would block legitimate admin role management.
70
+ *
71
+ * Expected on current (correct) code: all assertions pass. A regression that made the self-update
72
+ * path role-writable by non-admins would flip the step-3/step-4 assertions to red.
73
+ */
74
+ export var self_admin_role_assignment_tests = function (_a) {
75
+ var sdk = _a.sdk;
76
+ return __awaiter(void 0, void 0, void 0, function () {
77
+ var stamp, NON_ADMIN_ROLE, expect_blocked, attackerId, attacker, attackerBefore, sdkAttacker_1, _b, attackerAfter, afterAdminUpdate, _c;
78
+ var _d;
79
+ var _e;
80
+ return __generator(this, function (_f) {
81
+ switch (_f.label) {
82
+ case 0:
83
+ log_header("F-0076: self-admin role assignment privilege-escalation regression");
84
+ stamp = Date.now();
85
+ NON_ADMIN_ROLE = 'Provider';
86
+ expect_blocked = function (fn, description) { return __awaiter(void 0, void 0, void 0, function () {
87
+ var e_1;
88
+ return __generator(this, function (_a) {
89
+ switch (_a.label) {
90
+ case 0:
91
+ _a.trys.push([0, 2, , 3]);
92
+ return [4 /*yield*/, fn()];
93
+ case 1:
94
+ _a.sent();
95
+ assert(false, "".concat(description, " - SELF-ROLE ESCALATION SUCCEEDED (expected it to be blocked)"));
96
+ return [3 /*break*/, 3];
97
+ case 2:
98
+ e_1 = _a.sent();
99
+ // CRUD relationship-constraint failures surface as 400 { message, info } via SDK parseError.
100
+ assert((e_1 === null || e_1 === void 0 ? void 0 : e_1.code) === 400 || (e_1 === null || e_1 === void 0 ? void 0 : e_1.statusCode) === 400 || typeof (e_1 === null || e_1 === void 0 ? void 0 : e_1.message) === 'string', "".concat(description, " - expected a block error, got: ").concat(JSON.stringify(e_1)), description);
101
+ return [3 /*break*/, 3];
102
+ case 3: return [2 /*return*/];
103
+ }
104
+ });
105
+ }); };
106
+ _f.label = 1;
107
+ case 1:
108
+ _f.trys.push([1, , 15, 20]);
109
+ return [4 /*yield*/, sdk.api.users.createOne({
110
+ email: "f0076-attacker-".concat(stamp, "@example.com"),
111
+ notificationEmailsDisabled: true,
112
+ verifiedEmail: true,
113
+ })];
114
+ case 2:
115
+ attacker = _f.sent();
116
+ attackerId = attacker.id;
117
+ return [4 /*yield*/, sdk.api.users.updateOne(attackerId, { roles: [NON_ADMIN_ROLE] }, { replaceObjectFields: true })];
118
+ case 3:
119
+ _f.sent();
120
+ return [4 /*yield*/, wait(undefined, 2000)
121
+ // Setup sanity: the attacker holds exactly the non-admin role and is NOT an admin.
122
+ ]; // role change triggers a logout; let it propagate before minting a token
123
+ case 4:
124
+ _f.sent(); // role change triggers a logout; let it propagate before minting a token
125
+ return [4 /*yield*/, sdk.api.users.getOne(attackerId)];
126
+ case 5:
127
+ attackerBefore = _f.sent();
128
+ assert(JSON.stringify(attackerBefore.roles) === JSON.stringify([NON_ADMIN_ROLE]), "Setup failed: expected attacker to hold [".concat(NON_ADMIN_ROLE, "], got ").concat(JSON.stringify(attackerBefore.roles)), 'F-0076 setup: attacker holds a non-admin role');
129
+ _b = Session.bind;
130
+ _d = {
131
+ host: host
132
+ };
133
+ return [4 /*yield*/, sdk.api.users.generate_auth_token({ id: attackerId })];
134
+ case 6:
135
+ sdkAttacker_1 = new (_b.apply(Session, [void 0, (_d.authToken = (_f.sent()).authToken,
136
+ _d)]))();
137
+ return [4 /*yield*/, sdkAttacker_1.refresh_session()]; // populate userInfo from the freshly-minted token
138
+ case 7:
139
+ _f.sent(); // populate userInfo from the freshly-minted token
140
+ assert(sdkAttacker_1.userInfo.id === attackerId && !((_e = sdkAttacker_1.userInfo.roles) !== null && _e !== void 0 ? _e : []).includes('Admin'), "Setup failed: attacker session is not the expected non-admin user", 'F-0076 setup: authenticated as the non-admin attacker');
141
+ // 3. SECURITY ASSERTIONS — every self-role mutation by the non-admin must be blocked.
142
+ return [4 /*yield*/, expect_blocked(function () { return sdkAttacker_1.api.users.updateOne(attackerId, { roles: ['Admin'] }, { replaceObjectFields: true }); }, 'F-0076: non-admin self-update to [Admin] is blocked')];
143
+ case 8:
144
+ // 3. SECURITY ASSERTIONS — every self-role mutation by the non-admin must be blocked.
145
+ _f.sent();
146
+ return [4 /*yield*/, expect_blocked(function () { return sdkAttacker_1.api.users.updateOne(attackerId, { roles: [NON_ADMIN_ROLE, 'Admin'] }, { replaceObjectFields: true }); }, 'F-0076: non-admin self-update to [Provider, Admin] is blocked')];
147
+ case 9:
148
+ _f.sent();
149
+ return [4 /*yield*/, expect_blocked(function () { return sdkAttacker_1.api.users.updateOne(attackerId, { roles: ["Arbitrary_".concat(stamp)] }, { replaceObjectFields: true }); }, 'F-0076: non-admin self-update to an arbitrary role is blocked')];
150
+ case 10:
151
+ _f.sent();
152
+ return [4 /*yield*/, expect_blocked(function () { return sdkAttacker_1.api.users.updateOne(attackerId, { roles: [] }, { replaceObjectFields: true }); }, 'F-0076: non-admin self-update to [] (would grant defaults) is blocked')
153
+ // 4. STATE ASSERTION — nothing slipped through; roles are still the original non-admin role.
154
+ ];
155
+ case 11:
156
+ _f.sent();
157
+ return [4 /*yield*/, sdk.api.users.getOne(attackerId)];
158
+ case 12:
159
+ attackerAfter = _f.sent();
160
+ assert(JSON.stringify(attackerAfter.roles) === JSON.stringify([NON_ADMIN_ROLE]), "ESCALATION LEAK: attacker roles changed to ".concat(JSON.stringify(attackerAfter.roles), " ")
161
+ + "after self-update attempts. Expected [".concat(NON_ADMIN_ROLE, "]."), 'F-0076: attacker roles unchanged after all self-escalation attempts');
162
+ // 5. POSITIVE CONTROL — an Admin CAN update the user's roles (mechanism is not over-restricted).
163
+ return [4 /*yield*/, sdk.api.users.updateOne(attackerId, { roles: [NON_ADMIN_ROLE] }, { replaceObjectFields: true })];
164
+ case 13:
165
+ // 5. POSITIVE CONTROL — an Admin CAN update the user's roles (mechanism is not over-restricted).
166
+ _f.sent();
167
+ return [4 /*yield*/, sdk.api.users.getOne(attackerId)];
168
+ case 14:
169
+ afterAdminUpdate = _f.sent();
170
+ assert(JSON.stringify(afterAdminUpdate.roles) === JSON.stringify([NON_ADMIN_ROLE]), "Admin role update failed: roles are ".concat(JSON.stringify(afterAdminUpdate.roles), ", expected [").concat(NON_ADMIN_ROLE, "]"), 'F-0076: admin can manage user roles (positive control)');
171
+ return [3 /*break*/, 20];
172
+ case 15:
173
+ if (!attackerId) return [3 /*break*/, 19];
174
+ _f.label = 16;
175
+ case 16:
176
+ _f.trys.push([16, 18, , 19]);
177
+ return [4 /*yield*/, sdk.api.users.deleteOne(attackerId)];
178
+ case 17:
179
+ _f.sent();
180
+ return [3 /*break*/, 19];
181
+ case 18:
182
+ _c = _f.sent();
183
+ return [3 /*break*/, 19];
184
+ case 19: return [7 /*endfinally*/];
185
+ case 20: return [2 /*return*/];
186
+ }
187
+ });
188
+ });
189
+ };
190
+ // Allow running this test file independently
191
+ if (require.main === module) {
192
+ console.log("\uD83C\uDF10 Using API URL: ".concat(host));
193
+ var sdk_1 = new Session({ host: host });
194
+ var sdkNonAdmin_1 = new Session({ host: host });
195
+ var runTests = function () { return __awaiter(void 0, void 0, void 0, function () {
196
+ return __generator(this, function (_a) {
197
+ switch (_a.label) {
198
+ case 0: return [4 /*yield*/, setup_tests(sdk_1, sdkNonAdmin_1)];
199
+ case 1:
200
+ _a.sent();
201
+ return [4 /*yield*/, self_admin_role_assignment_tests({ sdk: sdk_1, sdkNonAdmin: sdkNonAdmin_1 })];
202
+ case 2:
203
+ _a.sent();
204
+ return [2 /*return*/];
205
+ }
206
+ });
207
+ }); };
208
+ runTests()
209
+ .then(function () {
210
+ console.log("✅ F-0076 self-admin role assignment test suite completed successfully");
211
+ process.exit(0);
212
+ })
213
+ .catch(function (error) {
214
+ console.error("❌ F-0076 self-admin role assignment test suite failed:", error);
215
+ process.exit(1);
216
+ });
217
+ }
218
+ //# sourceMappingURL=F-0076-self-admin-role-assignment.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"F-0076-self-admin-role-assignment.test.js","sourceRoot":"","sources":["../../../../../src/tests/api_tests/security/F-0076-self-admin-role-assignment.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import { Session } from "../../sdk";
2
+ export declare const user_portal_settings_tests: ({ sdk, sdkNonAdmin }: {
3
+ sdk: Session;
4
+ sdkNonAdmin: Session;
5
+ }) => Promise<void>;
6
+ //# sourceMappingURL=user_portal_settings.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"user_portal_settings.test.d.ts","sourceRoot":"","sources":["../../../../src/tests/api_tests/user_portal_settings.test.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAkB,MAAM,WAAW,CAAA;AAanD,eAAO,MAAM,0BAA0B;SAAuC,OAAO;iBAAe,OAAO;mBAmL1G,CAAA"}