@dalmore/api-contracts 1.0.5 → 1.0.6

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 (137) hide show
  1. package/package.json +4 -5
  2. package/src/common/types/account-contact.types.ts +98 -0
  3. package/src/common/types/account-detail.types.ts +27 -0
  4. package/src/common/types/account-integration.types.ts +143 -0
  5. package/src/common/types/account-manager.types.ts +124 -0
  6. package/src/common/types/account.types.ts +296 -0
  7. package/src/common/types/activity.types.ts +274 -0
  8. package/src/common/types/address.spec.ts +203 -0
  9. package/src/common/types/address.types.ts +41 -0
  10. package/src/common/types/aic.types.ts +246 -0
  11. package/src/common/types/aml.types.ts +18 -0
  12. package/src/common/types/api-key-logs.types.ts +66 -0
  13. package/src/common/types/api-keys.types.ts +69 -0
  14. package/src/common/types/asset.types.ts +338 -0
  15. package/src/common/types/auth.types.ts +370 -0
  16. package/src/common/types/batch-jobs.types.ts +151 -0
  17. package/src/common/types/bonus-tier.types.ts +147 -0
  18. package/src/common/types/cart.types.ts +18 -0
  19. package/src/common/types/checklist-items.types.ts +70 -0
  20. package/src/common/types/checklist.types.ts +97 -0
  21. package/src/common/types/common.types.spec.ts +336 -0
  22. package/src/common/types/common.types.ts +1520 -0
  23. package/src/common/types/comply-advantage-api.types.ts +316 -0
  24. package/src/common/types/comply-advantage.types.ts +25 -0
  25. package/src/common/types/contact-us.types.ts +107 -0
  26. package/src/common/types/contract-helpers.ts +205 -0
  27. package/src/common/types/countries.types.ts +375 -0
  28. package/src/common/types/covered-person.types.ts +274 -0
  29. package/src/common/types/dashboard.types.ts +799 -0
  30. package/src/common/types/data-record.types.ts +325 -0
  31. package/src/common/types/data-room.types.ts +242 -0
  32. package/src/common/types/default-theme-config.types.ts +87 -0
  33. package/src/common/types/disbursement-adjustment.types.ts +32 -0
  34. package/src/common/types/disbursement-approval-user.types.ts +100 -0
  35. package/src/common/types/disbursement-review.types.ts +110 -0
  36. package/src/common/types/disbursement-transaction.types.ts +72 -0
  37. package/src/common/types/disbursements.types.ts +310 -0
  38. package/src/common/types/domain-filter.types.ts +55 -0
  39. package/src/common/types/email-theme.types.ts +442 -0
  40. package/src/common/types/entity.types.ts +15 -0
  41. package/src/common/types/error-responses.types.ts +135 -0
  42. package/src/common/types/escrow-account.types.ts +104 -0
  43. package/src/common/types/exchange-api-key.types.ts +121 -0
  44. package/src/common/types/exchange-import.types.ts +36 -0
  45. package/src/common/types/exchange-provider.types.ts +329 -0
  46. package/src/common/types/file.types.ts +461 -0
  47. package/src/common/types/files.types.spec.ts +154 -0
  48. package/src/common/types/health.types.ts +29 -0
  49. package/src/common/types/index.ts +48 -0
  50. package/src/common/types/individuals.types.ts +554 -0
  51. package/src/common/types/investor-account.types.ts +1239 -0
  52. package/src/common/types/investorAccountIdSchema.type.ts +0 -0
  53. package/src/common/types/investors-offering.types.ts +65 -0
  54. package/src/common/types/invite.types.ts +133 -0
  55. package/src/common/types/issuer-bank-account.types.ts +107 -0
  56. package/src/common/types/issuer-offering.types.ts +306 -0
  57. package/src/common/types/issuer-payment-method.types.spec.ts +612 -0
  58. package/src/common/types/issuer-payment-method.types.ts +341 -0
  59. package/src/common/types/issuer.types.ts +312 -0
  60. package/src/common/types/job-item.types.ts +119 -0
  61. package/src/common/types/jobs.types.ts +171 -0
  62. package/src/common/types/kyb.types.ts +53 -0
  63. package/src/common/types/kyc.types.ts +188 -0
  64. package/src/common/types/legal-entity.types.ts +185 -0
  65. package/src/common/types/login-history.types.ts +46 -0
  66. package/src/common/types/mail-template.types.ts +436 -0
  67. package/src/common/types/north-cap-integration.types.ts +190 -0
  68. package/src/common/types/note.types.ts +109 -0
  69. package/src/common/types/notification.types.ts +58 -0
  70. package/src/common/types/notion-api.types.ts +374 -0
  71. package/src/common/types/notion-database.types.ts +125 -0
  72. package/src/common/types/notion-page.types.ts +267 -0
  73. package/src/common/types/offering-reports.types.ts +153 -0
  74. package/src/common/types/offering-submission.types.ts +314 -0
  75. package/src/common/types/offering.types.spec.ts +91 -0
  76. package/src/common/types/offering.types.ts +590 -0
  77. package/src/common/types/page-revision.types.ts +86 -0
  78. package/src/common/types/page.types.ts +436 -0
  79. package/src/common/types/password.type.ts +15 -0
  80. package/src/common/types/payment-methods.types.ts +298 -0
  81. package/src/common/types/phone.spec.ts +76 -0
  82. package/src/common/types/phone.type.ts +27 -0
  83. package/src/common/types/portfolio.types.ts +50 -0
  84. package/src/common/types/privacy-policy-and-tos.types.ts +231 -0
  85. package/src/common/types/queue.types.ts +112 -0
  86. package/src/common/types/registered-reps.types.ts +25 -0
  87. package/src/common/types/rejection-reasons.types.ts +56 -0
  88. package/src/common/types/reminder-config.types.ts +40 -0
  89. package/src/common/types/review.types.ts +133 -0
  90. package/src/common/types/role.types.ts +26 -0
  91. package/src/common/types/secondary-customer.types.ts +66 -0
  92. package/src/common/types/secondary-issuer.types.ts +50 -0
  93. package/src/common/types/secondary-order.types.ts +58 -0
  94. package/src/common/types/secondary-security.types.ts +60 -0
  95. package/src/common/types/secondary-trade.entity.ts +16 -0
  96. package/src/common/types/secondary-trade.types.ts +95 -0
  97. package/src/common/types/secure-request.types.ts +68 -0
  98. package/src/common/types/signer.types.ts +651 -0
  99. package/src/common/types/site-link.types.spec.ts +134 -0
  100. package/src/common/types/site-link.types.ts +166 -0
  101. package/src/common/types/site-settings.types.ts +726 -0
  102. package/src/common/types/site.types.ts +270 -0
  103. package/src/common/types/sms.types.ts +30 -0
  104. package/src/common/types/state-machine.types.ts +177 -0
  105. package/src/common/types/states.types.ts +163 -0
  106. package/src/common/types/subdoc-preview.types.ts +35 -0
  107. package/src/common/types/task.types.ts +258 -0
  108. package/src/common/types/trade-adjustment.type.ts +33 -0
  109. package/src/common/types/trade-line-item.type.ts +132 -0
  110. package/src/common/types/trade.types.ts +929 -0
  111. package/src/common/types/transaction.types.ts +198 -0
  112. package/src/common/types/trusted-contact.types.ts +122 -0
  113. package/src/common/types/typography.types.ts +75 -0
  114. package/src/common/types/user-manual.types.ts +290 -0
  115. package/src/common/types/user-setting.types.ts +133 -0
  116. package/src/common/types/user.types.ts +320 -0
  117. package/src/common/types/webhook.types.ts +588 -0
  118. package/src/common/types/zip.type.ts +36 -0
  119. package/src/contracts/clients/accounts/index.ts +61 -0
  120. package/src/contracts/clients/aic/index.ts +59 -0
  121. package/src/contracts/clients/api-key-logs/index.ts +53 -0
  122. package/src/contracts/clients/api-keys/index.ts +73 -0
  123. package/src/contracts/clients/assets/index.ts +102 -0
  124. package/src/contracts/clients/auth/index.ts +50 -0
  125. package/src/contracts/clients/files/index.ts +166 -0
  126. package/src/contracts/clients/files-public/index.ts +166 -0
  127. package/src/contracts/clients/index.ts +44 -0
  128. package/src/contracts/clients/individuals/index.ts +93 -0
  129. package/src/contracts/clients/investor-accounts/index.ts +93 -0
  130. package/src/contracts/clients/issuers/index.ts +94 -0
  131. package/src/contracts/clients/legal-entities/index.ts +93 -0
  132. package/src/contracts/clients/offerings/index.ts +117 -0
  133. package/src/contracts/clients/secure-requests/index.ts +34 -0
  134. package/src/contracts/clients/sites/index.ts +56 -0
  135. package/src/contracts/clients/trades/index.ts +122 -0
  136. package/index.d.mts +0 -17
  137. package/index.d.ts +0 -17
@@ -0,0 +1,461 @@
1
+ import { z } from 'zod';
2
+ import { TypeID } from 'typeid-js';
3
+ import { extendZodWithOpenApi } from '@anatine/zod-openapi';
4
+ import { IBaseEntity } from './entity.types';
5
+ import {
6
+ ComplianceReview,
7
+ FileLabels,
8
+ FileLabelsEnum,
9
+ FileStatus,
10
+ IPaginationMeta,
11
+ metadataSchema,
12
+ TargetTableConfig,
13
+ TargetTableEnum,
14
+ } from './common.types';
15
+ import { KycZod } from './kyc.types';
16
+ import { KybZod } from './kyb.types';
17
+ import { IIndividualZod, individualIdSchema } from './individuals.types';
18
+ import { LegalEntityZod } from './legal-entity.types';
19
+ import { IInvestorAccount } from './investor-account.types';
20
+ import { TradeZod } from './trade.types';
21
+
22
+ extendZodWithOpenApi(z);
23
+
24
+ export const allowedFileExtension = [
25
+ '.pdf',
26
+ '.doc',
27
+ '.docx',
28
+ '.xls',
29
+ '.xlsx',
30
+ '.csv',
31
+ '.gif',
32
+ '.png',
33
+ '.jpg',
34
+ '.jpeg',
35
+ '.txt',
36
+ '.webp',
37
+ '.html',
38
+ '.svg',
39
+ ];
40
+
41
+ export enum ADFFileProcessingPipeline {
42
+ NONE = 'NONE',
43
+ SUBDOCS_PHOENIX = 'SUBDOCS_PHOENIX',
44
+ SUBDOCS_GENERAL = 'SUBDOCS_GENERAL',
45
+ AIC_TAX_RETURNS = 'AIC_TAX_RETURNS',
46
+ AIC_ACCOUNT_STATEMENTS = 'AIC_ACCOUNT_STATEMENTS',
47
+ PAY_STUBS = 'PAY_STUBS',
48
+ }
49
+
50
+ /**
51
+ * BoldSign brand logo supported file extensions
52
+ * @see https://developers.boldsign.com/branding/create-brand/?region=us
53
+ */
54
+ export const boldSignAllowedFileExtension = ['.jpg', '.jpeg', '.png', '.svg'];
55
+
56
+ export const maxFileSize = 50 * 1024 * 1024; // 50MB
57
+
58
+ export const fileIdSchema = z.string().refine(
59
+ (value) => {
60
+ try {
61
+ const tid = TypeID.fromString(value);
62
+ return tid.getType() === 'file';
63
+ } catch {
64
+ return false;
65
+ }
66
+ },
67
+ {
68
+ message:
69
+ 'Invalid file ID format. Must be a valid TypeID with "file" prefix. Example: file_01j5y5ghx8fvc83dmx3pznq7hv',
70
+ },
71
+ );
72
+
73
+ export const FileZod = IBaseEntity.extend({
74
+ id: fileIdSchema,
75
+ tid: z.string().nullable(),
76
+ url: z.string().url().openapi({ example: 'https://file-location.com' }),
77
+ fileType: z.string().openapi({ example: 'pdf' }),
78
+ fileName: z.string().openapi({ example: 'test.pdf' }),
79
+ name: z.string().openapi({ example: 'test' }),
80
+ category: z.string().openapi({ example: 'application' }),
81
+ label: z.string().openapi({ example: 'other' }),
82
+ private: z.boolean(),
83
+ complianceReview: z.nativeEnum(ComplianceReview),
84
+ targetTable: z.enum(TargetTableEnum).nullable(),
85
+ targetId: z.string().nullable(),
86
+ linkedFrom: z.string().nullable(),
87
+ accountId: z
88
+ .string()
89
+ .openapi({ example: 'account_01j6aqmtfyfwy9spjdcnh7yqk7' }),
90
+ userId: z.string().openapi({ example: 'user_01j6aqmtfyfwy9spjdcnh7yqk7' }),
91
+ kycs: z.array(KycZod).optional().nullable(),
92
+ kybs: z.array(KybZod).optional().nullable(),
93
+ individuals: z.array(IIndividualZod).optional().nullable(),
94
+ legalEntities: z.array(LegalEntityZod).optional().nullable(),
95
+ investorAccounts: z.array(IInvestorAccount).optional().nullable(),
96
+ trades: z.array(TradeZod).optional().nullable(),
97
+ metadata: z.string().nullable(),
98
+ status: z.nativeEnum(FileStatus),
99
+ reviewLog: z.string().nullable(),
100
+ pipeline: z.nativeEnum(ADFFileProcessingPipeline),
101
+ errorMessage: z.string().nullable(),
102
+ });
103
+ export type FileZod = z.infer<typeof FileZod>;
104
+
105
+ export const LinkFileZod = z.object({
106
+ targetId: z
107
+ .string()
108
+ .refine(
109
+ (value) => {
110
+ try {
111
+ const tid = TypeID.fromString(value);
112
+ return Object.values(TargetTableConfig).some(
113
+ (config) => config.idPrefix === tid.getType(),
114
+ );
115
+ } catch {
116
+ return false;
117
+ }
118
+ },
119
+ {
120
+ message: `Invalid target ID format. Must match the corresponding table's ID prefix. Valid prefix: ${TargetTableEnum.toString().toLowerCase()}. Example: disbursement_01j6aqmtfyfwy9spjdcnh7yqk7`,
121
+ },
122
+ )
123
+ .openapi({ example: 'trades_01j6aqmtfyfwy9spjdcnh7yqk7' }),
124
+ label: FileLabelsEnum.openapi({ example: FileLabels.OTHER }).optional(),
125
+ category: z.string().min(1).max(50).optional(),
126
+ });
127
+ export type LinkFileZod = z.infer<typeof LinkFileZod>;
128
+
129
+ export const IPaginatedFile = z.object({
130
+ items: z.array(FileZod),
131
+ meta: IPaginationMeta,
132
+ });
133
+ export type IPaginatedFile = z.infer<typeof IPaginatedFile>;
134
+
135
+ export const PatchFile = z.object({
136
+ name: z.string().max(50),
137
+ category: z.string().max(50),
138
+ label: FileLabelsEnum.openapi({ example: FileLabels.OTHER }),
139
+ metadata: metadataSchema.nullable().optional(),
140
+ });
141
+ export type PatchFile = z.infer<typeof PatchFile>;
142
+
143
+ export const PostFileQueryParams = z.object({
144
+ name: z.string().openapi({ example: 'file_name' }),
145
+ category: z.string().max(50).openapi({ example: 'application' }),
146
+ label: FileLabelsEnum.openapi({ example: FileLabels.OTHER }),
147
+ targetId: z
148
+ .string()
149
+ .refine(
150
+ (value) => {
151
+ try {
152
+ const tid = TypeID.fromString(value);
153
+ return Object.values(TargetTableConfig).some(
154
+ (config) => config.idPrefix === tid.getType(),
155
+ );
156
+ } catch {
157
+ return false;
158
+ }
159
+ },
160
+ {
161
+ message: `Invalid target ID format. Must match the corresponding table's ID prefix. Valid prefix: ${TargetTableEnum.toString().toLowerCase()}. Example: disbursement_01j6aqmtfyfwy9spjdcnh7yqk7`,
162
+ },
163
+ )
164
+ .openapi({ example: 'trades_01j6aqmtfyfwy9spjdcnh7yqk7' }),
165
+ metadata: metadataSchema.nullable().optional(),
166
+ });
167
+ export type PostFileQueryParams = z.infer<typeof PostFileQueryParams>;
168
+
169
+ /**
170
+ * CLIENT portal specific schema for file uploads
171
+ * Only allows INDIVIDUALS as target for file uploads
172
+ */
173
+ export const ClientPostFileQueryParams = z.object({
174
+ name: z.string().min(1).max(100).openapi({ example: 'file_name' }),
175
+ category: z.string().max(50).openapi({ example: 'application' }),
176
+ label: FileLabelsEnum.openapi({ example: FileLabels.OTHER }),
177
+ targetId: individualIdSchema.openapi({
178
+ example: 'individual_01kcrsny60fb9rjc8bbqc3b80c',
179
+ }),
180
+ metadata: metadataSchema.nullable().optional(),
181
+ });
182
+ export type ClientPostFileQueryParams = z.infer<
183
+ typeof ClientPostFileQueryParams
184
+ >;
185
+
186
+ export const FileFilters = z.object({
187
+ name: z.string().optional(),
188
+ category: z.string().optional(),
189
+ label: z.string().optional(),
190
+ targetId: z.string().optional(),
191
+ fileType: z.string().optional(),
192
+ targetTable: z.enum(TargetTableEnum).optional(),
193
+ includeLinkedFrom: z
194
+ .string()
195
+ .optional()
196
+ .openapi({
197
+ description:
198
+ 'if false it only returns files that have linkedFrom=null. Defaults to false',
199
+ })
200
+ .refine((v) => !v || v === 'true' || v === 'false', {
201
+ message: 'includeLinkedFrom must be a boolean string',
202
+ })
203
+ .transform((v) => {
204
+ if (!v) return false;
205
+ return v === 'true';
206
+ }),
207
+ });
208
+ export const filesInclude = z.enum(['user', 'account']);
209
+ /**
210
+ * @description Query parameters for including related entities
211
+ * @xample in contract us as -> query: z.object({}).merge(AccountsIncludeQuery),
212
+ */
213
+ export const FilesIncludeQuery = z.object({
214
+ include: z
215
+ .string()
216
+ .optional()
217
+ .transform((str) => (str ? str.split(',') : []))
218
+ .refine(
219
+ (includes) =>
220
+ includes.every((include) =>
221
+ filesInclude.options.includes(include as any),
222
+ ),
223
+ {
224
+ message: `Invalid include option provided. Valid options are: ${filesInclude.options.join(',')}`,
225
+ },
226
+ )
227
+ .openapi({
228
+ example: `${filesInclude.options.join(',')}`,
229
+ }),
230
+ targetTable: z.enum(TargetTableEnum).optional(),
231
+ targetId: z
232
+ .string()
233
+ .optional()
234
+ .refine(
235
+ (value) => {
236
+ try {
237
+ if (!value) return true;
238
+ const tid = TypeID.fromString(value);
239
+ return Object.values(TargetTableConfig).some(
240
+ (config) => config.idPrefix === tid.getType(),
241
+ );
242
+ } catch {
243
+ return false;
244
+ }
245
+ },
246
+ {
247
+ message: `Invalid target ID format. Must match the corresponding table's ID prefix. Valid prefix: ${TargetTableEnum.toString().toLowerCase()}. Example: disbursement_01j6aqmtfyfwy9spjdcnh7yqk7`,
248
+ },
249
+ ),
250
+ });
251
+ export interface FilesIncludeQuery extends z.infer<typeof FilesIncludeQuery> {}
252
+
253
+ export const ComplianceFilesIncludeQuery = z.object({
254
+ include: z
255
+ .string()
256
+ .optional()
257
+ .transform((str) => (str ? str.split(',') : []))
258
+ .refine(
259
+ (includes) =>
260
+ includes.every((include) =>
261
+ filesInclude.options.includes(include as any),
262
+ ),
263
+ {
264
+ message: `Invalid include option provided. Valid options are: ${filesInclude.options.join(',')}`,
265
+ },
266
+ )
267
+ .openapi({
268
+ example: `${filesInclude.options.join(',')}`,
269
+ }),
270
+ targetTable: z.enum(TargetTableEnum).optional(),
271
+ targetId: z
272
+ .string()
273
+ .optional()
274
+ .transform((str) => (str ? str.split(',') : []))
275
+ .refine(
276
+ (tradeIds) => {
277
+ try {
278
+ if (!tradeIds.length) return true;
279
+ return tradeIds.every((tradeId) => {
280
+ const tid = TypeID.fromString(tradeId);
281
+ return Object.values(TargetTableConfig).some(
282
+ (config) => config.idPrefix === tid.getType(),
283
+ );
284
+ });
285
+ } catch {
286
+ return false;
287
+ }
288
+ },
289
+ {
290
+ message: `Invalid target ID format. Must match the corresponding table's ID prefix. Valid prefix: ${TargetTableEnum.toString().toLowerCase()}. Example: disbursement_01j6aqmtfyfwy9spjdcnh7yqk7`,
291
+ },
292
+ ),
293
+ status: z.nativeEnum(FileStatus).optional(),
294
+ label: z.nativeEnum(FileLabels).optional(),
295
+ accountId: z.string().optional(),
296
+ complianceReview: z.nativeEnum(ComplianceReview).optional(),
297
+ pipeline: z.nativeEnum(ADFFileProcessingPipeline).optional(),
298
+ });
299
+ export interface ComplianceFilesIncludeQuery
300
+ extends z.infer<typeof ComplianceFilesIncludeQuery> {}
301
+
302
+ export const reviewFiles = z.object({
303
+ complianceReview: z.nativeEnum(ComplianceReview),
304
+ data: z
305
+ .array(fileIdSchema)
306
+ .min(1, 'At least one file is required')
307
+ .max(50, 'Maximum of 50 files allowed')
308
+ .openapi({
309
+ example: [
310
+ 'file_00041061050r3gg28a1c60t3gf',
311
+ 'file_00041061050r3gg28a1c60t3g',
312
+ ],
313
+ }),
314
+ });
315
+ export type reviewFiles = z.infer<typeof reviewFiles>;
316
+
317
+ export const PatchFileMetadata = z.object({
318
+ corrected: z.record(z.string(), z.unknown()),
319
+ });
320
+ export type PatchFileMetadata = z.infer<typeof PatchFileMetadata>;
321
+
322
+ /**
323
+ * Zod schema for file metadata structure
324
+ * Used to validate the metadata field in File entity
325
+ */
326
+ export const FileMetadataSchema = z.object({
327
+ extracted: z
328
+ .record(z.any())
329
+ .refine(
330
+ (data) =>
331
+ data && typeof data === 'object' && Object.keys(data).length > 0,
332
+ {
333
+ message: 'Extracted data is required and cannot be empty',
334
+ },
335
+ ),
336
+ expected: z
337
+ .record(z.any())
338
+ .refine(
339
+ (data) =>
340
+ data && typeof data === 'object' && Object.keys(data).length > 0,
341
+ {
342
+ message: 'Expected data is required and cannot be empty',
343
+ },
344
+ ),
345
+ corrected: z.record(z.any()).optional(),
346
+ });
347
+ export type FileMetadata = z.infer<typeof FileMetadataSchema>;
348
+
349
+ export enum FilePipeLineReviewAction {
350
+ SAVE = 'SAVE',
351
+ REJECTED = 'REJECTED',
352
+ ACCEPTED = 'ACCEPTED',
353
+ }
354
+
355
+ export const FilePipeLineReviewSchemaZod = z.object({
356
+ action: z
357
+ .nativeEnum(FilePipeLineReviewAction)
358
+ .optional()
359
+ .default(FilePipeLineReviewAction.SAVE),
360
+ });
361
+ export type FilePipeLineReviewSchemaZod = z.infer<
362
+ typeof FilePipeLineReviewSchemaZod
363
+ >;
364
+
365
+ export const ADFPipelineResponseZod = z.object({
366
+ extracted: z.record(z.string(), z.unknown()),
367
+ });
368
+ export type ADFPipelineResponseZod = z.infer<typeof ADFPipelineResponseZod>;
369
+
370
+ export const ReprocessAllFilesZod = z.object({
371
+ success: z.boolean().openapi({ example: true }),
372
+ message: z
373
+ .string()
374
+ .openapi({ example: 'All files reprocessed successfully' }),
375
+ });
376
+ export type ReprocessAllFilesZod = z.infer<typeof ReprocessAllFilesZod>;
377
+
378
+ /**
379
+ * Shared constants for BoldSign form field names.
380
+ * These constants ensure consistency between:
381
+ * - BoldSignService.getAllFormFields() - which generates form fields
382
+ * - getRequiredFieldPatterns() - which validates required fields
383
+ *
384
+ * If field names change, update them here to keep both functions in sync.
385
+ */
386
+
387
+ /**
388
+ * Base form field names that are always present
389
+ */
390
+ export const BOLDSIGN_BASE_FORM_FIELDS = {
391
+ LEGAL_NAME: 'LegalName',
392
+ QUANTITY: 'Quantity',
393
+ TOTAL: 'Total',
394
+ INVESTOR_TYPE: 'InvestorType',
395
+ OFFERING_NAME: 'OfferingName',
396
+ ASSET_NAME: 'AssetName',
397
+ PRICE_PER_SHARE: 'PricePerShare',
398
+ ISSUER_NAME: 'IssuerName',
399
+ } as const;
400
+
401
+ /**
402
+ * Entity form field name prefixes and suffixes
403
+ */
404
+ export const ENTITY_FIELD_PREFIX = 'Entity';
405
+ export const ENTITY_FORM_FIELDS = {
406
+ NAME: 'EntityName',
407
+ ADDRESS: 'EntityAddress',
408
+ ADDRESS2: 'EntityAddress2',
409
+ CITY: 'EntityCity',
410
+ STATE: 'EntityState',
411
+ ZIP: 'EntityZip',
412
+ COUNTRY: 'EntityCountry',
413
+ FULL_ADDRESS: 'EntityFullAddress',
414
+ PHONE: 'EntityPhone',
415
+ TYPE: 'EntityType',
416
+ EIN: 'EntityEIN',
417
+ STATE_OF_INCORPORATION: 'EntityStateOfIncorporation',
418
+ DATE_OF_INCORPORATION: 'EntityDateOfIncorporation',
419
+ } as const;
420
+
421
+ /**
422
+ * Signer form field name prefixes and suffixes
423
+ */
424
+ export const SIGNER_FIELD_PREFIX = 'Signer';
425
+ export const SIGNER_FIELD_SUFFIXES = {
426
+ NAME: 'Name',
427
+ EMAIL: 'Email',
428
+ PHONE: 'Phone',
429
+ FULL_ADDRESS: 'FullAddress',
430
+ ADDRESS: 'Address',
431
+ ADDRESS2: 'Address2',
432
+ CITY: 'City',
433
+ STATE: 'State',
434
+ ZIP: 'Zip',
435
+ COUNTRY: 'Country',
436
+ SSN_LAST_4: 'SSNLast4',
437
+ CITIZENSHIP: 'Citizenship',
438
+ ACCREDITATION_TYPE: 'AccreditationType',
439
+ IS_ACCREDITED: 'IsAccredited',
440
+ DOB: 'DOB',
441
+ } as const;
442
+
443
+ /**
444
+ * Helper function to generate signer field name
445
+ * @param signerIndex - The signer index (1-based)
446
+ * @param suffix - The field suffix (e.g., 'Name', 'Email')
447
+ * @returns The complete field name (e.g., 'Signer1Name', 'Signer2Email')
448
+ */
449
+ export function getSignerFieldName(
450
+ signerIndex: number,
451
+ suffix: string,
452
+ ): string {
453
+ return `${SIGNER_FIELD_PREFIX}${signerIndex}${suffix}`;
454
+ }
455
+ /**
456
+ * Type for all form field names
457
+ */
458
+ export type FormFieldName =
459
+ | (typeof BOLDSIGN_BASE_FORM_FIELDS)[keyof typeof BOLDSIGN_BASE_FORM_FIELDS]
460
+ | (typeof ENTITY_FORM_FIELDS)[keyof typeof ENTITY_FORM_FIELDS]
461
+ | ReturnType<typeof getSignerFieldName>;
@@ -0,0 +1,154 @@
1
+ import { typeid } from 'typeid-js';
2
+ import {
3
+ FileZod,
4
+ allowedFileExtension,
5
+ fileIdSchema,
6
+ PostFileQueryParams,
7
+ FileFilters,
8
+ ADFFileProcessingPipeline,
9
+ } from './file.types';
10
+ import { FileLabels, FileStatus } from './common.types';
11
+
12
+ describe('File Validation', () => {
13
+ describe('fileIdSchema', () => {
14
+ it('should validate correct file IDs', () => {
15
+ const validFileId = typeid('file').toString();
16
+ const result = fileIdSchema.safeParse(validFileId);
17
+ expect(result.success).toBe(true);
18
+ });
19
+
20
+ it('should reject invalid file IDs', () => {
21
+ const invalidFileIds = [
22
+ 'user_01j5y5ghx8fvc83dmx3pznq7hv',
23
+ 'file_invalid',
24
+ '',
25
+ ];
26
+
27
+ invalidFileIds.forEach((id) => {
28
+ const result = fileIdSchema.safeParse(id);
29
+ expect(result.success).toBe(false);
30
+ });
31
+ });
32
+ });
33
+
34
+ describe('FileZod', () => {
35
+ it('should validate a correct file object', () => {
36
+ const validFile = {
37
+ id: typeid('file').toString(),
38
+ tid: 'some-tid',
39
+ url: 'https://example.com/file.pdf',
40
+ fileType: 'pdf',
41
+ fileName: 'test.pdf',
42
+ name: 'test',
43
+ category: 'application',
44
+ label: FileLabels.OTHER,
45
+ metadata: null,
46
+ private: false,
47
+ complianceReview: 'PENDING',
48
+ targetTable: 'TRADES',
49
+ targetId: null,
50
+ linkedFrom: null,
51
+ accountId: typeid('account').toString(),
52
+ userId: typeid('user').toString(),
53
+ status: FileStatus.PENDING,
54
+ pipeline: ADFFileProcessingPipeline.NONE,
55
+ errorMessage: null,
56
+ reviewLog: null,
57
+ createdAt: new Date(),
58
+ updatedAt: new Date(),
59
+ deletedAt: null,
60
+ };
61
+
62
+ const result = FileZod.safeParse(validFile);
63
+ expect(result.success).toBe(true);
64
+ });
65
+
66
+ it('should reject invalid file objects', () => {
67
+ const invalidFile = {
68
+ id: 'invalid_id',
69
+ url: 'not-a-url',
70
+ // missing required fields
71
+ };
72
+
73
+ const result = FileZod.safeParse(invalidFile);
74
+ expect(result.success).toBe(false);
75
+ });
76
+ });
77
+
78
+ describe('PostFileQueryParams', () => {
79
+ it('should validate correct query parameters', () => {
80
+ const validQuery = {
81
+ name: 'test_file',
82
+ category: 'application',
83
+ label: FileLabels.OTHER,
84
+ targetId: typeid('trade').toString(),
85
+ };
86
+
87
+ const result = PostFileQueryParams.safeParse(validQuery);
88
+ expect(result.success).toBe(true);
89
+ });
90
+
91
+ it('should reject invalid target IDs', () => {
92
+ const invalidQuery = {
93
+ name: 'test_file',
94
+ category: 'application',
95
+ label: FileLabels.OTHER,
96
+ targetId: 'invalid_id',
97
+ };
98
+
99
+ const result = PostFileQueryParams.safeParse(invalidQuery);
100
+ expect(result.success).toBe(false);
101
+ });
102
+ });
103
+
104
+ describe('FileFilters', () => {
105
+ it('should handle optional includeLinkedFrom parameter', () => {
106
+ const validFilters = [
107
+ { includeLinkedFrom: 'true' },
108
+ { includeLinkedFrom: 'false' },
109
+ { includeLinkedFrom: undefined },
110
+ {},
111
+ ];
112
+
113
+ validFilters.forEach((filter) => {
114
+ const result = FileFilters.safeParse(filter);
115
+ expect(result.success).toBe(true);
116
+ });
117
+ });
118
+
119
+ it('should reject invalid includeLinkedFrom values', () => {
120
+ const invalidFilters = [
121
+ { includeLinkedFrom: 'yes' },
122
+ { includeLinkedFrom: '1' },
123
+ { includeLinkedFrom: 'TRUE' },
124
+ ];
125
+
126
+ invalidFilters.forEach((filter) => {
127
+ const result = FileFilters.safeParse(filter);
128
+ expect(result.success).toBe(false);
129
+ });
130
+ });
131
+ });
132
+
133
+ describe('allowedFileExtension', () => {
134
+ it('should contain all supported extensions', () => {
135
+ const expectedExtensions = [
136
+ '.pdf',
137
+ '.doc',
138
+ '.docx',
139
+ '.xls',
140
+ '.xlsx',
141
+ '.csv',
142
+ '.gif',
143
+ '.png',
144
+ '.jpg',
145
+ '.jpeg',
146
+ '.txt',
147
+ ];
148
+
149
+ expectedExtensions.forEach((ext) => {
150
+ expect(allowedFileExtension).toContain(ext);
151
+ });
152
+ });
153
+ });
154
+ });
@@ -0,0 +1,29 @@
1
+ import { z } from 'zod';
2
+
3
+ export const HealthStatusSchema = z.object({
4
+ status: z.enum(['up', 'down']),
5
+ message: z.string().optional(),
6
+ });
7
+ export type HealthStatus = z.infer<typeof HealthStatusSchema>;
8
+
9
+ export const ServiceHealthSchema = z.record(z.string(), HealthStatusSchema);
10
+ export type ServiceHealth = z.infer<typeof ServiceHealthSchema>;
11
+
12
+ export const HealthCheckResponseSchema = z.object({
13
+ status: z.enum(['ok', 'error']),
14
+ details: z
15
+ .object({
16
+ status: z.enum(['error', 'ok', 'shutting_down']),
17
+ info: z.record(z.string(), HealthStatusSchema).optional(),
18
+ error: z.record(z.string(), HealthStatusSchema).optional(),
19
+ details: z.record(z.string(), z.any()).optional(),
20
+ })
21
+ .optional(),
22
+ });
23
+ export type HealthCheckResponse = z.infer<typeof HealthCheckResponseSchema>;
24
+
25
+ export const HealthErrorSchema = z.object({
26
+ status: z.literal('error'),
27
+ message: z.string().optional(),
28
+ });
29
+ export type HealthError = z.infer<typeof HealthErrorSchema>;
@@ -0,0 +1,48 @@
1
+ export * from './account.types';
2
+ export * from './account-contact.types';
3
+ export * from './api-keys.types';
4
+ export * from './api-key-logs.types';
5
+ export * from './auth.types';
6
+ export * from './common.types';
7
+ export * from './entity.types';
8
+ export * from './error-responses.types';
9
+ export * from './file.types';
10
+ export * from './individuals.types';
11
+ export * from './invite.types';
12
+ export * from './jobs.types';
13
+ export * from './kyb.types';
14
+ export * from './kyc.types';
15
+ export * from './legal-entity.types';
16
+ export * from './note.types';
17
+ export * from './offering.types';
18
+ export * from './review.types';
19
+ export * from './sms.types';
20
+ export * from './trade.types';
21
+ export * from './user.types';
22
+ export * from './file.types';
23
+ export * from './review.types';
24
+ export * from './issuer-offering.types';
25
+ export * from './covered-person.types';
26
+ export * from './issuer.types';
27
+ export * from './page.types';
28
+ export * from './page-revision.types';
29
+ export * from './login-history.types';
30
+ export * from './offering-submission.types';
31
+ export * from './task.types';
32
+ export * from './secure-request.types';
33
+ export * from './issuer-payment-method.types';
34
+ export * from './site-settings.types';
35
+ export * from './typography.types';
36
+ export * from './contact-us.types';
37
+ export * from './role.types';
38
+ export * from './site-link.types';
39
+ export * from './email-theme.types';
40
+ export * from './disbursements.types';
41
+ export * from './domain-filter.types';
42
+ export * from './aic.types';
43
+ export * from './default-theme-config.types';
44
+ export * from './offering-reports.types';
45
+
46
+ export enum Versions {
47
+ V1 = 'v1',
48
+ }