@opensaas/stack-core 0.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 (95) hide show
  1. package/.turbo/turbo-build.log +4 -0
  2. package/README.md +447 -0
  3. package/dist/access/engine.d.ts +73 -0
  4. package/dist/access/engine.d.ts.map +1 -0
  5. package/dist/access/engine.js +244 -0
  6. package/dist/access/engine.js.map +1 -0
  7. package/dist/access/field-transforms.d.ts +47 -0
  8. package/dist/access/field-transforms.d.ts.map +1 -0
  9. package/dist/access/field-transforms.js +2 -0
  10. package/dist/access/field-transforms.js.map +1 -0
  11. package/dist/access/index.d.ts +3 -0
  12. package/dist/access/index.d.ts.map +1 -0
  13. package/dist/access/index.js +2 -0
  14. package/dist/access/index.js.map +1 -0
  15. package/dist/access/types.d.ts +83 -0
  16. package/dist/access/types.d.ts.map +1 -0
  17. package/dist/access/types.js +2 -0
  18. package/dist/access/types.js.map +1 -0
  19. package/dist/config/index.d.ts +39 -0
  20. package/dist/config/index.d.ts.map +1 -0
  21. package/dist/config/index.js +38 -0
  22. package/dist/config/index.js.map +1 -0
  23. package/dist/config/types.d.ts +413 -0
  24. package/dist/config/types.d.ts.map +1 -0
  25. package/dist/config/types.js +2 -0
  26. package/dist/config/types.js.map +1 -0
  27. package/dist/context/index.d.ts +31 -0
  28. package/dist/context/index.d.ts.map +1 -0
  29. package/dist/context/index.js +524 -0
  30. package/dist/context/index.js.map +1 -0
  31. package/dist/context/nested-operations.d.ts +10 -0
  32. package/dist/context/nested-operations.d.ts.map +1 -0
  33. package/dist/context/nested-operations.js +261 -0
  34. package/dist/context/nested-operations.js.map +1 -0
  35. package/dist/fields/index.d.ts +78 -0
  36. package/dist/fields/index.d.ts.map +1 -0
  37. package/dist/fields/index.js +381 -0
  38. package/dist/fields/index.js.map +1 -0
  39. package/dist/hooks/index.d.ts +58 -0
  40. package/dist/hooks/index.d.ts.map +1 -0
  41. package/dist/hooks/index.js +79 -0
  42. package/dist/hooks/index.js.map +1 -0
  43. package/dist/index.d.ts +11 -0
  44. package/dist/index.d.ts.map +1 -0
  45. package/dist/index.js +12 -0
  46. package/dist/index.js.map +1 -0
  47. package/dist/lib/case-utils.d.ts +49 -0
  48. package/dist/lib/case-utils.d.ts.map +1 -0
  49. package/dist/lib/case-utils.js +68 -0
  50. package/dist/lib/case-utils.js.map +1 -0
  51. package/dist/lib/case-utils.test.d.ts +2 -0
  52. package/dist/lib/case-utils.test.d.ts.map +1 -0
  53. package/dist/lib/case-utils.test.js +101 -0
  54. package/dist/lib/case-utils.test.js.map +1 -0
  55. package/dist/utils/password.d.ts +81 -0
  56. package/dist/utils/password.d.ts.map +1 -0
  57. package/dist/utils/password.js +132 -0
  58. package/dist/utils/password.js.map +1 -0
  59. package/dist/validation/schema.d.ts +17 -0
  60. package/dist/validation/schema.d.ts.map +1 -0
  61. package/dist/validation/schema.js +42 -0
  62. package/dist/validation/schema.js.map +1 -0
  63. package/dist/validation/schema.test.d.ts +2 -0
  64. package/dist/validation/schema.test.d.ts.map +1 -0
  65. package/dist/validation/schema.test.js +143 -0
  66. package/dist/validation/schema.test.js.map +1 -0
  67. package/docs/type-distribution-fix.md +136 -0
  68. package/package.json +48 -0
  69. package/src/access/engine.ts +360 -0
  70. package/src/access/field-transforms.ts +99 -0
  71. package/src/access/index.ts +20 -0
  72. package/src/access/types.ts +103 -0
  73. package/src/config/index.ts +71 -0
  74. package/src/config/types.ts +478 -0
  75. package/src/context/index.ts +814 -0
  76. package/src/context/nested-operations.ts +412 -0
  77. package/src/fields/index.ts +438 -0
  78. package/src/hooks/index.ts +132 -0
  79. package/src/index.ts +62 -0
  80. package/src/lib/case-utils.test.ts +127 -0
  81. package/src/lib/case-utils.ts +74 -0
  82. package/src/utils/password.ts +147 -0
  83. package/src/validation/schema.test.ts +171 -0
  84. package/src/validation/schema.ts +59 -0
  85. package/tests/access-relationships.test.ts +613 -0
  86. package/tests/access.test.ts +499 -0
  87. package/tests/config.test.ts +195 -0
  88. package/tests/context.test.ts +248 -0
  89. package/tests/hooks.test.ts +417 -0
  90. package/tests/password-type-distribution.test.ts +155 -0
  91. package/tests/password-types.test.ts +147 -0
  92. package/tests/password.test.ts +249 -0
  93. package/tsconfig.json +12 -0
  94. package/tsconfig.tsbuildinfo +1 -0
  95. package/vitest.config.ts +27 -0
@@ -0,0 +1,68 @@
1
+ /**
2
+ * Case conversion utilities for consistent naming across the stack
3
+ *
4
+ * - Config list names: PascalCase (e.g., "AuthUser", "BlogPost")
5
+ * - Prisma models: PascalCase (e.g., "AuthUser", "BlogPost")
6
+ * - Prisma client properties: camelCase (e.g., "authUser", "blogPost")
7
+ * - Context db properties: camelCase (e.g., "authUser", "blogPost")
8
+ * - URLs: kebab-case (e.g., "auth-user", "blog-post")
9
+ */
10
+ /**
11
+ * Convert PascalCase to camelCase
12
+ * AuthUser -> authUser
13
+ * BlogPost -> blogPost
14
+ */
15
+ export function pascalToCamel(str) {
16
+ return str.charAt(0).toLowerCase() + str.slice(1);
17
+ }
18
+ /**
19
+ * Convert PascalCase to kebab-case
20
+ * AuthUser -> auth-user
21
+ * BlogPost -> blog-post
22
+ */
23
+ export function pascalToKebab(str) {
24
+ return str.replace(/([A-Z])/g, (match, p1, offset) => {
25
+ return offset > 0 ? `-${p1.toLowerCase()}` : p1.toLowerCase();
26
+ });
27
+ }
28
+ /**
29
+ * Convert kebab-case to PascalCase
30
+ * auth-user -> AuthUser
31
+ * blog-post -> BlogPost
32
+ */
33
+ export function kebabToPascal(str) {
34
+ return str
35
+ .split('-')
36
+ .map((word) => word.charAt(0).toUpperCase() + word.slice(1))
37
+ .join('');
38
+ }
39
+ /**
40
+ * Convert kebab-case to camelCase
41
+ * auth-user -> authUser
42
+ * blog-post -> blogPost
43
+ */
44
+ export function kebabToCamel(str) {
45
+ return str.replace(/-([a-z])/g, (match, p1) => p1.toUpperCase());
46
+ }
47
+ /**
48
+ * Get the database key for a list (camelCase)
49
+ * Used for accessing context.db and prisma client
50
+ */
51
+ export function getDbKey(listKey) {
52
+ return pascalToCamel(listKey);
53
+ }
54
+ /**
55
+ * Get the URL segment for a list (kebab-case)
56
+ * Used for constructing admin URLs
57
+ */
58
+ export function getUrlKey(listKey) {
59
+ return pascalToKebab(listKey);
60
+ }
61
+ /**
62
+ * Get the list key from a URL segment (PascalCase)
63
+ * Used for parsing admin URLs
64
+ */
65
+ export function getListKeyFromUrl(urlSegment) {
66
+ return kebabToPascal(urlSegment);
67
+ }
68
+ //# sourceMappingURL=case-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"case-utils.js","sourceRoot":"","sources":["../../src/lib/case-utils.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH;;;;GAIG;AACH,MAAM,UAAU,aAAa,CAAC,GAAW;IACvC,OAAO,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAA;AACnD,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,aAAa,CAAC,GAAW;IACvC,OAAO,GAAG,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE;QACnD,OAAO,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,EAAE,CAAA;IAC/D,CAAC,CAAC,CAAA;AACJ,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,aAAa,CAAC,GAAW;IACvC,OAAO,GAAG;SACP,KAAK,CAAC,GAAG,CAAC;SACV,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SAC3D,IAAI,CAAC,EAAE,CAAC,CAAA;AACb,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,YAAY,CAAC,GAAW;IACtC,OAAO,GAAG,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC,CAAA;AAClE,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,QAAQ,CAAC,OAAe;IACtC,OAAO,aAAa,CAAC,OAAO,CAAC,CAAA;AAC/B,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,SAAS,CAAC,OAAe;IACvC,OAAO,aAAa,CAAC,OAAO,CAAC,CAAA;AAC/B,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,iBAAiB,CAAC,UAAkB;IAClD,OAAO,aAAa,CAAC,UAAU,CAAC,CAAA;AAClC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=case-utils.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"case-utils.test.d.ts","sourceRoot":"","sources":["../../src/lib/case-utils.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,101 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { pascalToCamel, pascalToKebab, kebabToPascal, kebabToCamel, getDbKey, getUrlKey, getListKeyFromUrl, } from './case-utils.js';
3
+ describe('Case Conversion Utilities', () => {
4
+ describe('pascalToCamel', () => {
5
+ it('should convert single word PascalCase to camelCase', () => {
6
+ expect(pascalToCamel('User')).toBe('user');
7
+ });
8
+ it('should convert multi-word PascalCase to camelCase', () => {
9
+ expect(pascalToCamel('AuthUser')).toBe('authUser');
10
+ expect(pascalToCamel('BlogPost')).toBe('blogPost');
11
+ expect(pascalToCamel('UserProfile')).toBe('userProfile');
12
+ });
13
+ it('should handle already camelCase strings', () => {
14
+ expect(pascalToCamel('user')).toBe('user');
15
+ expect(pascalToCamel('authUser')).toBe('authUser');
16
+ });
17
+ });
18
+ describe('pascalToKebab', () => {
19
+ it('should convert single word PascalCase to kebab-case', () => {
20
+ expect(pascalToKebab('User')).toBe('user');
21
+ });
22
+ it('should convert multi-word PascalCase to kebab-case', () => {
23
+ expect(pascalToKebab('AuthUser')).toBe('auth-user');
24
+ expect(pascalToKebab('BlogPost')).toBe('blog-post');
25
+ expect(pascalToKebab('UserProfile')).toBe('user-profile');
26
+ });
27
+ it('should handle consecutive capital letters', () => {
28
+ expect(pascalToKebab('HTMLElement')).toBe('h-t-m-l-element');
29
+ });
30
+ });
31
+ describe('kebabToPascal', () => {
32
+ it('should convert single word kebab-case to PascalCase', () => {
33
+ expect(kebabToPascal('user')).toBe('User');
34
+ });
35
+ it('should convert multi-word kebab-case to PascalCase', () => {
36
+ expect(kebabToPascal('auth-user')).toBe('AuthUser');
37
+ expect(kebabToPascal('blog-post')).toBe('BlogPost');
38
+ expect(kebabToPascal('user-profile')).toBe('UserProfile');
39
+ });
40
+ it('should handle already PascalCase strings without hyphens', () => {
41
+ expect(kebabToPascal('User')).toBe('User');
42
+ // Note: "AuthUser" without hyphens is treated as a single word
43
+ expect(kebabToPascal('AuthUser')).toBe('AuthUser');
44
+ });
45
+ });
46
+ describe('kebabToCamel', () => {
47
+ it('should convert single word kebab-case to camelCase', () => {
48
+ expect(kebabToCamel('user')).toBe('user');
49
+ });
50
+ it('should convert multi-word kebab-case to camelCase', () => {
51
+ expect(kebabToCamel('auth-user')).toBe('authUser');
52
+ expect(kebabToCamel('blog-post')).toBe('blogPost');
53
+ expect(kebabToCamel('user-profile')).toBe('userProfile');
54
+ });
55
+ });
56
+ describe('getDbKey', () => {
57
+ it('should convert list key to database key format', () => {
58
+ expect(getDbKey('User')).toBe('user');
59
+ expect(getDbKey('AuthUser')).toBe('authUser');
60
+ expect(getDbKey('BlogPost')).toBe('blogPost');
61
+ });
62
+ });
63
+ describe('getUrlKey', () => {
64
+ it('should convert list key to URL key format', () => {
65
+ expect(getUrlKey('User')).toBe('user');
66
+ expect(getUrlKey('AuthUser')).toBe('auth-user');
67
+ expect(getUrlKey('BlogPost')).toBe('blog-post');
68
+ });
69
+ });
70
+ describe('getListKeyFromUrl', () => {
71
+ it('should convert URL key to list key format', () => {
72
+ expect(getListKeyFromUrl('user')).toBe('User');
73
+ expect(getListKeyFromUrl('auth-user')).toBe('AuthUser');
74
+ expect(getListKeyFromUrl('blog-post')).toBe('BlogPost');
75
+ });
76
+ });
77
+ describe('Round-trip conversions', () => {
78
+ it('should maintain consistency: PascalCase -> kebab-case -> PascalCase', () => {
79
+ const original = 'BlogPost';
80
+ const kebab = getUrlKey(original);
81
+ const backToPascal = getListKeyFromUrl(kebab);
82
+ expect(backToPascal).toBe(original);
83
+ });
84
+ it('should maintain consistency: PascalCase -> camelCase -> operations', () => {
85
+ const original = 'AuthUser';
86
+ const camel = getDbKey(original);
87
+ expect(camel).toBe('authUser');
88
+ });
89
+ it('should handle multi-word conversions correctly', () => {
90
+ const testCases = ['User', 'AuthUser', 'BlogPost', 'UserProfile'];
91
+ testCases.forEach((original) => {
92
+ const url = getUrlKey(original);
93
+ const db = getDbKey(original);
94
+ const fromUrl = getListKeyFromUrl(url);
95
+ expect(fromUrl).toBe(original);
96
+ expect(db.charAt(0)).toBe(original.charAt(0).toLowerCase());
97
+ });
98
+ });
99
+ });
100
+ });
101
+ //# sourceMappingURL=case-utils.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"case-utils.test.js","sourceRoot":"","sources":["../../src/lib/case-utils.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,81 @@
1
+ /**
2
+ * Hash a plain text password using bcrypt
3
+ *
4
+ * @param plainPassword - The plain text password to hash
5
+ * @param costFactor - The bcrypt cost factor (default: 10)
6
+ * @returns Promise resolving to the hashed password
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * const hashed = await hashPassword('mypassword')
11
+ * // Returns: $2a$10$...
12
+ * ```
13
+ */
14
+ export declare function hashPassword(plainPassword: string, costFactor?: number): Promise<string>;
15
+ /**
16
+ * Compare a plain text password with a hashed password
17
+ *
18
+ * @param plainPassword - The plain text password to compare
19
+ * @param hashedPassword - The hashed password to compare against
20
+ * @returns Promise resolving to true if passwords match, false otherwise
21
+ *
22
+ * @example
23
+ * ```typescript
24
+ * const isValid = await comparePassword('mypassword', hashedPassword)
25
+ * if (isValid) {
26
+ * // Password is correct
27
+ * }
28
+ * ```
29
+ */
30
+ export declare function comparePassword(plainPassword: string, hashedPassword: string): Promise<boolean>;
31
+ /**
32
+ * Check if a string appears to be a bcrypt hash
33
+ * Bcrypt hashes follow the format: $2a$10$...
34
+ *
35
+ * @param value - The string to check
36
+ * @returns True if the string looks like a bcrypt hash
37
+ */
38
+ export declare function isHashedPassword(value: string): boolean;
39
+ /**
40
+ * HashedPassword class wraps a bcrypt hash and provides a compare method
41
+ * This allows password field values to be used as strings while also
42
+ * providing a convenient compare() method for authentication
43
+ *
44
+ * @example
45
+ * ```typescript
46
+ * const user = await context.db.user.findUnique({ where: { id: '1' } })
47
+ * const isValid = await user.password.compare('plaintextPassword')
48
+ * ```
49
+ */
50
+ export declare class HashedPassword {
51
+ private readonly hash;
52
+ constructor(hash: string);
53
+ /**
54
+ * Compare a plain text password with this hashed password
55
+ *
56
+ * @param plainPassword - The plain text password to compare
57
+ * @returns Promise resolving to true if passwords match
58
+ */
59
+ compare(plainPassword: string): Promise<boolean>;
60
+ /**
61
+ * Get the underlying hash string
62
+ * This allows the HashedPassword to be used anywhere a string is expected
63
+ */
64
+ toString(): string;
65
+ /**
66
+ * Get the underlying hash when used in string contexts
67
+ * This allows the HashedPassword to be coerced to a string automatically
68
+ */
69
+ [Symbol.toPrimitive](hint: string): string;
70
+ /**
71
+ * Return the hash for JSON serialization
72
+ * This ensures the hash is properly serialized when converting to JSON
73
+ */
74
+ toJSON(): string;
75
+ /**
76
+ * Get the underlying hash value
77
+ * This allows accessing the raw hash string
78
+ */
79
+ valueOf(): string;
80
+ }
81
+ //# sourceMappingURL=password.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password.d.ts","sourceRoot":"","sources":["../../src/utils/password.ts"],"names":[],"mappings":"AASA;;;;;;;;;;;;GAYG;AACH,wBAAsB,YAAY,CAChC,aAAa,EAAE,MAAM,EACrB,UAAU,GAAE,MAA4B,GACvC,OAAO,CAAC,MAAM,CAAC,CAMjB;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAsB,eAAe,CACnC,aAAa,EAAE,MAAM,EACrB,cAAc,EAAE,MAAM,GACrB,OAAO,CAAC,OAAO,CAAC,CAgBlB;AAED;;;;;;GAMG;AACH,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAMvD;AAED;;;;;;;;;;GAUG;AACH,qBAAa,cAAc;IACb,OAAO,CAAC,QAAQ,CAAC,IAAI;gBAAJ,IAAI,EAAE,MAAM;IAMzC;;;;;OAKG;IACG,OAAO,CAAC,aAAa,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAItD;;;OAGG;IACH,QAAQ,IAAI,MAAM;IAIlB;;;OAGG;IACH,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAO1C;;;OAGG;IACH,MAAM,IAAI,MAAM;IAIhB;;;OAGG;IACH,OAAO,IAAI,MAAM;CAGlB"}
@@ -0,0 +1,132 @@
1
+ import bcrypt from 'bcryptjs';
2
+ /**
3
+ * Default bcrypt cost factor (rounds)
4
+ * Higher values = more secure but slower
5
+ * 10 is a good balance for production use
6
+ */
7
+ const DEFAULT_COST_FACTOR = 10;
8
+ /**
9
+ * Hash a plain text password using bcrypt
10
+ *
11
+ * @param plainPassword - The plain text password to hash
12
+ * @param costFactor - The bcrypt cost factor (default: 10)
13
+ * @returns Promise resolving to the hashed password
14
+ *
15
+ * @example
16
+ * ```typescript
17
+ * const hashed = await hashPassword('mypassword')
18
+ * // Returns: $2a$10$...
19
+ * ```
20
+ */
21
+ export async function hashPassword(plainPassword, costFactor = DEFAULT_COST_FACTOR) {
22
+ if (typeof plainPassword !== 'string' || plainPassword.length === 0) {
23
+ throw new Error('Password must be a non-empty string');
24
+ }
25
+ return bcrypt.hash(plainPassword, costFactor);
26
+ }
27
+ /**
28
+ * Compare a plain text password with a hashed password
29
+ *
30
+ * @param plainPassword - The plain text password to compare
31
+ * @param hashedPassword - The hashed password to compare against
32
+ * @returns Promise resolving to true if passwords match, false otherwise
33
+ *
34
+ * @example
35
+ * ```typescript
36
+ * const isValid = await comparePassword('mypassword', hashedPassword)
37
+ * if (isValid) {
38
+ * // Password is correct
39
+ * }
40
+ * ```
41
+ */
42
+ export async function comparePassword(plainPassword, hashedPassword) {
43
+ if (typeof plainPassword !== 'string' || typeof hashedPassword !== 'string') {
44
+ return false;
45
+ }
46
+ if (plainPassword.length === 0 || hashedPassword.length === 0) {
47
+ return false;
48
+ }
49
+ try {
50
+ return await bcrypt.compare(plainPassword, hashedPassword);
51
+ }
52
+ catch (error) {
53
+ // Invalid hash format or other bcrypt error
54
+ console.error('Password comparison failed:', error);
55
+ return false;
56
+ }
57
+ }
58
+ /**
59
+ * Check if a string appears to be a bcrypt hash
60
+ * Bcrypt hashes follow the format: $2a$10$...
61
+ *
62
+ * @param value - The string to check
63
+ * @returns True if the string looks like a bcrypt hash
64
+ */
65
+ export function isHashedPassword(value) {
66
+ if (typeof value !== 'string')
67
+ return false;
68
+ // Bcrypt hashes start with $2a$, $2b$, or $2y$ followed by cost factor
69
+ // and are typically 60 characters long
70
+ return /^\$2[aby]\$\d{2}\$.{53}$/.test(value);
71
+ }
72
+ /**
73
+ * HashedPassword class wraps a bcrypt hash and provides a compare method
74
+ * This allows password field values to be used as strings while also
75
+ * providing a convenient compare() method for authentication
76
+ *
77
+ * @example
78
+ * ```typescript
79
+ * const user = await context.db.user.findUnique({ where: { id: '1' } })
80
+ * const isValid = await user.password.compare('plaintextPassword')
81
+ * ```
82
+ */
83
+ export class HashedPassword {
84
+ hash;
85
+ constructor(hash) {
86
+ this.hash = hash;
87
+ if (!hash || typeof hash !== 'string') {
88
+ throw new Error('HashedPassword requires a non-empty hash string');
89
+ }
90
+ }
91
+ /**
92
+ * Compare a plain text password with this hashed password
93
+ *
94
+ * @param plainPassword - The plain text password to compare
95
+ * @returns Promise resolving to true if passwords match
96
+ */
97
+ async compare(plainPassword) {
98
+ return comparePassword(plainPassword, this.hash);
99
+ }
100
+ /**
101
+ * Get the underlying hash string
102
+ * This allows the HashedPassword to be used anywhere a string is expected
103
+ */
104
+ toString() {
105
+ return this.hash;
106
+ }
107
+ /**
108
+ * Get the underlying hash when used in string contexts
109
+ * This allows the HashedPassword to be coerced to a string automatically
110
+ */
111
+ [Symbol.toPrimitive](hint) {
112
+ if (hint === 'string' || hint === 'default') {
113
+ return this.hash;
114
+ }
115
+ return this.hash;
116
+ }
117
+ /**
118
+ * Return the hash for JSON serialization
119
+ * This ensures the hash is properly serialized when converting to JSON
120
+ */
121
+ toJSON() {
122
+ return this.hash;
123
+ }
124
+ /**
125
+ * Get the underlying hash value
126
+ * This allows accessing the raw hash string
127
+ */
128
+ valueOf() {
129
+ return this.hash;
130
+ }
131
+ }
132
+ //# sourceMappingURL=password.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password.js","sourceRoot":"","sources":["../../src/utils/password.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,UAAU,CAAA;AAE7B;;;;GAIG;AACH,MAAM,mBAAmB,GAAG,EAAE,CAAA;AAE9B;;;;;;;;;;;;GAYG;AACH,MAAM,CAAC,KAAK,UAAU,YAAY,CAChC,aAAqB,EACrB,aAAqB,mBAAmB;IAExC,IAAI,OAAO,aAAa,KAAK,QAAQ,IAAI,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACpE,MAAM,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAA;IACxD,CAAC;IAED,OAAO,MAAM,CAAC,IAAI,CAAC,aAAa,EAAE,UAAU,CAAC,CAAA;AAC/C,CAAC;AAED;;;;;;;;;;;;;;GAcG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CACnC,aAAqB,EACrB,cAAsB;IAEtB,IAAI,OAAO,aAAa,KAAK,QAAQ,IAAI,OAAO,cAAc,KAAK,QAAQ,EAAE,CAAC;QAC5E,OAAO,KAAK,CAAA;IACd,CAAC;IAED,IAAI,aAAa,CAAC,MAAM,KAAK,CAAC,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC9D,OAAO,KAAK,CAAA;IACd,CAAC;IAED,IAAI,CAAC;QACH,OAAO,MAAM,MAAM,CAAC,OAAO,CAAC,aAAa,EAAE,cAAc,CAAC,CAAA;IAC5D,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,4CAA4C;QAC5C,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAA;QACnD,OAAO,KAAK,CAAA;IACd,CAAC;AACH,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,gBAAgB,CAAC,KAAa;IAC5C,IAAI,OAAO,KAAK,KAAK,QAAQ;QAAE,OAAO,KAAK,CAAA;IAE3C,uEAAuE;IACvE,uCAAuC;IACvC,OAAO,0BAA0B,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;AAC/C,CAAC;AAED;;;;;;;;;;GAUG;AACH,MAAM,OAAO,cAAc;IACI;IAA7B,YAA6B,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;QACvC,IAAI,CAAC,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;YACtC,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAA;QACpE,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,OAAO,CAAC,aAAqB;QACjC,OAAO,eAAe,CAAC,aAAa,EAAE,IAAI,CAAC,IAAI,CAAC,CAAA;IAClD,CAAC;IAED;;;OAGG;IACH,QAAQ;QACN,OAAO,IAAI,CAAC,IAAI,CAAA;IAClB,CAAC;IAED;;;OAGG;IACH,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,IAAY;QAC/B,IAAI,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,SAAS,EAAE,CAAC;YAC5C,OAAO,IAAI,CAAC,IAAI,CAAA;QAClB,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAA;IAClB,CAAC;IAED;;;OAGG;IACH,MAAM;QACJ,OAAO,IAAI,CAAC,IAAI,CAAA;IAClB,CAAC;IAED;;;OAGG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,IAAI,CAAA;IAClB,CAAC;CACF"}
@@ -0,0 +1,17 @@
1
+ import { z } from 'zod';
2
+ import type { FieldConfig } from '../config/types.js';
3
+ /**
4
+ * Generate Zod schema from field configurations
5
+ */
6
+ export declare function generateZodSchema(fieldConfigs: Record<string, FieldConfig>, operation?: 'create' | 'update'): z.ZodObject;
7
+ /**
8
+ * Validate data against field configurations using Zod
9
+ * Returns structured errors by field
10
+ */
11
+ export declare function validateWithZod(data: Record<string, unknown>, fieldConfigs: Record<string, FieldConfig>, operation?: 'create' | 'update'): {
12
+ success: true;
13
+ } | {
14
+ success: false;
15
+ errors: Record<string, string>;
16
+ };
17
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/validation/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AACvB,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAA;AAErD;;GAEG;AACH,wBAAgB,iBAAiB,CAC/B,YAAY,EAAE,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,EACzC,SAAS,GAAE,QAAQ,GAAG,QAAmB,GACxC,CAAC,CAAC,SAAS,CAsBb;AAED;;;GAGG;AACH,wBAAgB,eAAe,CAC7B,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC7B,YAAY,EAAE,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,EACzC,SAAS,GAAE,QAAQ,GAAG,QAAmB,GACxC;IAAE,OAAO,EAAE,IAAI,CAAA;CAAE,GAAG;IAAE,OAAO,EAAE,KAAK,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,CAiBxE"}
@@ -0,0 +1,42 @@
1
+ import { z } from 'zod';
2
+ /**
3
+ * Generate Zod schema from field configurations
4
+ */
5
+ export function generateZodSchema(fieldConfigs, operation = 'create') {
6
+ const shape = {};
7
+ for (const [fieldName, fieldConfig] of Object.entries(fieldConfigs)) {
8
+ // Skip system fields and relationships
9
+ if (['id', 'createdAt', 'updatedAt'].includes(fieldName) ||
10
+ fieldConfig.type === 'relationship') {
11
+ continue;
12
+ }
13
+ // Use the field's schema generator
14
+ if (fieldConfig.getZodSchema) {
15
+ shape[fieldName] = fieldConfig.getZodSchema(fieldName, operation);
16
+ }
17
+ else {
18
+ // Fallback for custom field types without schema generators
19
+ shape[fieldName] = z.any().optional();
20
+ }
21
+ }
22
+ return z.object(shape);
23
+ }
24
+ /**
25
+ * Validate data against field configurations using Zod
26
+ * Returns structured errors by field
27
+ */
28
+ export function validateWithZod(data, fieldConfigs, operation = 'create') {
29
+ const schema = generateZodSchema(fieldConfigs, operation);
30
+ const result = schema.safeParse(data);
31
+ if (result.success) {
32
+ return { success: true };
33
+ }
34
+ // Convert Zod errors to field-specific error messages
35
+ const errors = {};
36
+ for (const issue of result.error.issues) {
37
+ const fieldPath = issue.path.join('.');
38
+ errors[fieldPath] = issue.message;
39
+ }
40
+ return { success: false, errors };
41
+ }
42
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/validation/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AAGvB;;GAEG;AACH,MAAM,UAAU,iBAAiB,CAC/B,YAAyC,EACzC,YAAiC,QAAQ;IAEzC,MAAM,KAAK,GAAiC,EAAE,CAAA;IAE9C,KAAK,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;QACpE,uCAAuC;QACvC,IACE,CAAC,IAAI,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC,QAAQ,CAAC,SAAS,CAAC;YACpD,WAAW,CAAC,IAAI,KAAK,cAAc,EACnC,CAAC;YACD,SAAQ;QACV,CAAC;QAED,mCAAmC;QACnC,IAAI,WAAW,CAAC,YAAY,EAAE,CAAC;YAC7B,KAAK,CAAC,SAAS,CAAC,GAAG,WAAW,CAAC,YAAY,CAAC,SAAS,EAAE,SAAS,CAAC,CAAA;QACnE,CAAC;aAAM,CAAC;YACN,4DAA4D;YAC5D,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,CAAA;QACvC,CAAC;IACH,CAAC;IAED,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;AACxB,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,eAAe,CAC7B,IAA6B,EAC7B,YAAyC,EACzC,YAAiC,QAAQ;IAEzC,MAAM,MAAM,GAAG,iBAAiB,CAAC,YAAY,EAAE,SAAS,CAAC,CAAA;IAEzD,MAAM,MAAM,GAAG,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,CAAA;IAErC,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QACnB,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAA;IAC1B,CAAC;IAED,sDAAsD;IACtD,MAAM,MAAM,GAA2B,EAAE,CAAA;IACzC,KAAK,MAAM,KAAK,IAAI,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;QACxC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;QACtC,MAAM,CAAC,SAAS,CAAC,GAAG,KAAK,CAAC,OAAO,CAAA;IACnC,CAAC;IAED,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAA;AACnC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=schema.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.test.d.ts","sourceRoot":"","sources":["../../src/validation/schema.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,143 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { generateZodSchema, validateWithZod } from './schema.js';
3
+ import { text, integer, select } from '../fields/index.js';
4
+ describe('Zod Schema Generation', () => {
5
+ describe('generateZodSchema', () => {
6
+ it('should generate schema for text field with required validation', () => {
7
+ const fields = {
8
+ name: text({ validation: { isRequired: true } }),
9
+ };
10
+ const schema = generateZodSchema(fields, 'create');
11
+ expect(schema).toBeDefined();
12
+ });
13
+ it('should generate schema for text field with length validation', () => {
14
+ const fields = {
15
+ title: text({
16
+ validation: { isRequired: true, length: { min: 3, max: 100 } },
17
+ }),
18
+ };
19
+ const schema = generateZodSchema(fields, 'create');
20
+ expect(schema).toBeDefined();
21
+ });
22
+ it('should generate schema for integer field with min/max validation', () => {
23
+ const fields = {
24
+ age: integer({ validation: { isRequired: true, min: 0, max: 120 } }),
25
+ };
26
+ const schema = generateZodSchema(fields, 'create');
27
+ expect(schema).toBeDefined();
28
+ });
29
+ it('should generate schema for select field', () => {
30
+ const fields = {
31
+ status: select({
32
+ options: [
33
+ { label: 'Active', value: 'active' },
34
+ { label: 'Inactive', value: 'inactive' },
35
+ ],
36
+ validation: { isRequired: true },
37
+ }),
38
+ };
39
+ const schema = generateZodSchema(fields, 'create');
40
+ expect(schema).toBeDefined();
41
+ });
42
+ it('should make fields optional in update mode', () => {
43
+ const fields = {
44
+ name: text({ validation: { isRequired: true } }),
45
+ };
46
+ const schema = generateZodSchema(fields, 'update');
47
+ expect(schema).toBeDefined();
48
+ });
49
+ });
50
+ describe('validateWithZod', () => {
51
+ it('should pass validation for valid text field', () => {
52
+ const fields = {
53
+ name: text({ validation: { isRequired: true } }),
54
+ };
55
+ const result = validateWithZod({ name: 'John Doe' }, fields, 'create');
56
+ expect(result.success).toBe(true);
57
+ });
58
+ it('should fail validation for missing required field', () => {
59
+ const fields = {
60
+ name: text({ validation: { isRequired: true } }),
61
+ };
62
+ const result = validateWithZod({}, fields, 'create');
63
+ expect(result.success).toBe(false);
64
+ if (!result.success) {
65
+ expect(result.errors).toHaveProperty('name');
66
+ }
67
+ });
68
+ it('should fail validation for text too short', () => {
69
+ const fields = {
70
+ title: text({
71
+ validation: { isRequired: true, length: { min: 5 } },
72
+ }),
73
+ };
74
+ const result = validateWithZod({ title: 'Hi' }, fields, 'create');
75
+ expect(result.success).toBe(false);
76
+ if (!result.success) {
77
+ expect(result.errors.title).toContain('at least 5 characters');
78
+ }
79
+ });
80
+ it('should fail validation for text too long', () => {
81
+ const fields = {
82
+ title: text({ validation: { length: { max: 10 } } }),
83
+ };
84
+ const result = validateWithZod({ title: 'This is a very long title' }, fields, 'create');
85
+ expect(result.success).toBe(false);
86
+ if (!result.success) {
87
+ expect(result.errors.title).toContain('at most 10 characters');
88
+ }
89
+ });
90
+ it('should fail validation for integer below min', () => {
91
+ const fields = {
92
+ age: integer({ validation: { min: 18 } }),
93
+ };
94
+ const result = validateWithZod({ age: 15 }, fields, 'create');
95
+ expect(result.success).toBe(false);
96
+ if (!result.success) {
97
+ expect(result.errors.age).toContain('at least 18');
98
+ }
99
+ });
100
+ it('should fail validation for integer above max', () => {
101
+ const fields = {
102
+ age: integer({ validation: { max: 120 } }),
103
+ };
104
+ const result = validateWithZod({ age: 150 }, fields, 'create');
105
+ expect(result.success).toBe(false);
106
+ if (!result.success) {
107
+ expect(result.errors.age).toContain('at most 120');
108
+ }
109
+ });
110
+ it('should fail validation for invalid select value', () => {
111
+ const fields = {
112
+ status: select({
113
+ options: [
114
+ { label: 'Active', value: 'active' },
115
+ { label: 'Inactive', value: 'inactive' },
116
+ ],
117
+ validation: { isRequired: true },
118
+ }),
119
+ };
120
+ const result = validateWithZod({ status: 'invalid' }, fields, 'create');
121
+ expect(result.success).toBe(false);
122
+ if (!result.success) {
123
+ expect(result.errors.status).toBeDefined();
124
+ }
125
+ });
126
+ it('should skip system fields in validation', () => {
127
+ const fields = {
128
+ id: text(),
129
+ name: text({ validation: { isRequired: true } }),
130
+ };
131
+ const result = validateWithZod({ name: 'John' }, fields, 'create');
132
+ expect(result.success).toBe(true);
133
+ });
134
+ it('should allow required fields to be missing in update mode', () => {
135
+ const fields = {
136
+ name: text({ validation: { isRequired: true } }),
137
+ };
138
+ const result = validateWithZod({}, fields, 'update');
139
+ expect(result.success).toBe(true);
140
+ });
141
+ });
142
+ });
143
+ //# sourceMappingURL=schema.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.test.js","sourceRoot":"","sources":["../../src/validation/schema.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAA;AAC7C,OAAO,EAAE,iBAAiB,EAAE,eAAe,EAAE,MAAM,aAAa,CAAA;AAEhE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAA;AAE1D,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;IACrC,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;QACjC,EAAE,CAAC,gEAAgE,EAAE,GAAG,EAAE;YACxE,MAAM,MAAM,GAAgC;gBAC1C,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClD,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,8DAA8D,EAAE,GAAG,EAAE;YACtE,MAAM,MAAM,GAAgC;gBAC1C,KAAK,EAAE,IAAI,CAAC;oBACV,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;iBAC/D,CAAC;aACH,CAAA;YAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClD,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,kEAAkE,EAAE,GAAG,EAAE;YAC1E,MAAM,MAAM,GAAgC;gBAC1C,GAAG,EAAE,OAAO,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC;aACrE,CAAA;YAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClD,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;YACjD,MAAM,MAAM,GAAgC;gBAC1C,MAAM,EAAE,MAAM,CAAC;oBACb,OAAO,EAAE;wBACP,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;wBACpC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;qBACzC;oBACD,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE;iBACjC,CAAC;aACH,CAAA;YAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClD,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACpD,MAAM,MAAM,GAAgC;gBAC1C,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClD,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;YACrD,MAAM,MAAM,GAAgC;gBAC1C,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACtE,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACnC,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;YAC3D,MAAM,MAAM,GAAgC;gBAC1C,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACpD,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;YAC9C,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;YACnD,MAAM,MAAM,GAAgC;gBAC1C,KAAK,EAAE,IAAI,CAAC;oBACV,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE,EAAE;iBACrD,CAAC;aACH,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACjE,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAA;YAChE,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,MAAM,MAAM,GAAgC;gBAC1C,KAAK,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,MAAM,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC;aACrD,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,KAAK,EAAE,2BAA2B,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACxF,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAA;YAChE,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,8CAA8C,EAAE,GAAG,EAAE;YACtD,MAAM,MAAM,GAAgC;gBAC1C,GAAG,EAAE,OAAO,CAAC,EAAE,UAAU,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,CAAC;aAC1C,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YAC7D,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAA;YACpD,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,8CAA8C,EAAE,GAAG,EAAE;YACtD,MAAM,MAAM,GAAgC;gBAC1C,GAAG,EAAE,OAAO,CAAC,EAAE,UAAU,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC;aAC3C,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YAC9D,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAA;YACpD,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;YACzD,MAAM,MAAM,GAAgC;gBAC1C,MAAM,EAAE,MAAM,CAAC;oBACb,OAAO,EAAE;wBACP,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;wBACpC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;qBACzC;oBACD,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE;iBACjC,CAAC;aACH,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACvE,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YAClC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;YAC5C,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;YACjD,MAAM,MAAM,GAAgC;gBAC1C,EAAE,EAAE,IAAI,EAAE;gBACV,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YAClE,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACnC,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,2DAA2D,EAAE,GAAG,EAAE;YACnE,MAAM,MAAM,GAAgC;gBAC1C,IAAI,EAAE,IAAI,CAAC,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,CAAC;aACjD,CAAA;YAED,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAA;YACpD,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACnC,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}