nestjs-ddd-cli 2.2.1 → 3.2.1

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 (254) hide show
  1. package/README.md +247 -408
  2. package/ddd.schema.json +111 -0
  3. package/dist/commands/aggregate-validator.d.ts +9 -0
  4. package/dist/commands/aggregate-validator.js +953 -0
  5. package/dist/commands/aggregate-validator.js.map +1 -0
  6. package/dist/commands/ai-assist.d.ts +8 -0
  7. package/dist/commands/ai-assist.js +337 -0
  8. package/dist/commands/ai-assist.js.map +1 -0
  9. package/dist/commands/api-contracts.d.ts +9 -0
  10. package/dist/commands/api-contracts.js +1368 -0
  11. package/dist/commands/api-contracts.js.map +1 -0
  12. package/dist/commands/api-docs.d.ts +8 -0
  13. package/dist/commands/api-docs.js +408 -0
  14. package/dist/commands/api-docs.js.map +1 -0
  15. package/dist/commands/api-versioning.d.ts +11 -0
  16. package/dist/commands/api-versioning.js +643 -0
  17. package/dist/commands/api-versioning.js.map +1 -0
  18. package/dist/commands/audit-logging.d.ts +9 -0
  19. package/dist/commands/audit-logging.js +1129 -0
  20. package/dist/commands/audit-logging.js.map +1 -0
  21. package/dist/commands/batch-generate.d.ts +10 -0
  22. package/dist/commands/batch-generate.js +405 -0
  23. package/dist/commands/batch-generate.js.map +1 -0
  24. package/dist/commands/caching-strategies.d.ts +9 -0
  25. package/dist/commands/caching-strategies.js +874 -0
  26. package/dist/commands/caching-strategies.js.map +1 -0
  27. package/dist/commands/code-analyzer.d.ts +42 -0
  28. package/dist/commands/code-analyzer.js +474 -0
  29. package/dist/commands/code-analyzer.js.map +1 -0
  30. package/dist/commands/database-seeding.d.ts +6 -0
  31. package/dist/commands/database-seeding.js +621 -0
  32. package/dist/commands/database-seeding.js.map +1 -0
  33. package/dist/commands/db-optimization.d.ts +7 -0
  34. package/dist/commands/db-optimization.js +687 -0
  35. package/dist/commands/db-optimization.js.map +1 -0
  36. package/dist/commands/dependency-graph.d.ts +6 -0
  37. package/dist/commands/dependency-graph.js +329 -0
  38. package/dist/commands/dependency-graph.js.map +1 -0
  39. package/dist/commands/doctor-enhanced.d.ts +22 -0
  40. package/dist/commands/doctor-enhanced.js +543 -0
  41. package/dist/commands/doctor-enhanced.js.map +1 -0
  42. package/dist/commands/doctor.d.ts +4 -0
  43. package/dist/commands/doctor.js +151 -0
  44. package/dist/commands/doctor.js.map +1 -0
  45. package/dist/commands/env-manager.d.ts +6 -0
  46. package/dist/commands/env-manager.js +419 -0
  47. package/dist/commands/env-manager.js.map +1 -0
  48. package/dist/commands/event-sourcing-full.d.ts +10 -0
  49. package/dist/commands/event-sourcing-full.js +1107 -0
  50. package/dist/commands/event-sourcing-full.js.map +1 -0
  51. package/dist/commands/feature-flags.d.ts +9 -0
  52. package/dist/commands/feature-flags.js +824 -0
  53. package/dist/commands/feature-flags.js.map +1 -0
  54. package/dist/commands/filter-dsl.d.ts +10 -0
  55. package/dist/commands/filter-dsl.js +1407 -0
  56. package/dist/commands/filter-dsl.js.map +1 -0
  57. package/dist/commands/generate-all.js +485 -32
  58. package/dist/commands/generate-all.js.map +1 -1
  59. package/dist/commands/generate-deployment.d.ts +8 -0
  60. package/dist/commands/generate-deployment.js +746 -0
  61. package/dist/commands/generate-deployment.js.map +1 -0
  62. package/dist/commands/generate-domain-service.d.ts +14 -0
  63. package/dist/commands/generate-domain-service.js +796 -0
  64. package/dist/commands/generate-domain-service.js.map +1 -0
  65. package/dist/commands/generate-entity.js +82 -24
  66. package/dist/commands/generate-entity.js.map +1 -1
  67. package/dist/commands/generate-from-schema.d.ts +56 -0
  68. package/dist/commands/generate-from-schema.js +222 -0
  69. package/dist/commands/generate-from-schema.js.map +1 -0
  70. package/dist/commands/generate-orchestrator.d.ts +14 -0
  71. package/dist/commands/generate-orchestrator.js +887 -0
  72. package/dist/commands/generate-orchestrator.js.map +1 -0
  73. package/dist/commands/generate-repository.d.ts +14 -0
  74. package/dist/commands/generate-repository.js +1019 -0
  75. package/dist/commands/generate-repository.js.map +1 -0
  76. package/dist/commands/generate-shared.d.ts +4 -0
  77. package/dist/commands/generate-shared.js +388 -0
  78. package/dist/commands/generate-shared.js.map +1 -0
  79. package/dist/commands/generate-value-object.d.ts +32 -0
  80. package/dist/commands/generate-value-object.js +700 -0
  81. package/dist/commands/generate-value-object.js.map +1 -0
  82. package/dist/commands/graphql-subscriptions.d.ts +6 -0
  83. package/dist/commands/graphql-subscriptions.js +607 -0
  84. package/dist/commands/graphql-subscriptions.js.map +1 -0
  85. package/dist/commands/graphql-types.d.ts +5 -0
  86. package/dist/commands/graphql-types.js +423 -0
  87. package/dist/commands/graphql-types.js.map +1 -0
  88. package/dist/commands/health-probes-advanced.d.ts +6 -0
  89. package/dist/commands/health-probes-advanced.js +655 -0
  90. package/dist/commands/health-probes-advanced.js.map +1 -0
  91. package/dist/commands/i18n-setup.d.ts +10 -0
  92. package/dist/commands/i18n-setup.js +677 -0
  93. package/dist/commands/i18n-setup.js.map +1 -0
  94. package/dist/commands/init-config.d.ts +6 -0
  95. package/dist/commands/init-config.js +370 -0
  96. package/dist/commands/init-config.js.map +1 -0
  97. package/dist/commands/init-project.js +56 -6
  98. package/dist/commands/init-project.js.map +1 -1
  99. package/dist/commands/interactive-scaffold.d.ts +5 -0
  100. package/dist/commands/interactive-scaffold.js +271 -0
  101. package/dist/commands/interactive-scaffold.js.map +1 -0
  102. package/dist/commands/metrics-prometheus.d.ts +6 -0
  103. package/dist/commands/metrics-prometheus.js +681 -0
  104. package/dist/commands/metrics-prometheus.js.map +1 -0
  105. package/dist/commands/migration-engine.d.ts +6 -0
  106. package/dist/commands/migration-engine.js +446 -0
  107. package/dist/commands/migration-engine.js.map +1 -0
  108. package/dist/commands/migration.d.ts +12 -0
  109. package/dist/commands/migration.js +484 -0
  110. package/dist/commands/migration.js.map +1 -0
  111. package/dist/commands/monorepo.d.ts +8 -0
  112. package/dist/commands/monorepo.js +483 -0
  113. package/dist/commands/monorepo.js.map +1 -0
  114. package/dist/commands/multi-database.d.ts +5 -0
  115. package/dist/commands/multi-database.js +439 -0
  116. package/dist/commands/multi-database.js.map +1 -0
  117. package/dist/commands/observability-tracing.d.ts +10 -0
  118. package/dist/commands/observability-tracing.js +740 -0
  119. package/dist/commands/observability-tracing.js.map +1 -0
  120. package/dist/commands/openapi-export.d.ts +8 -0
  121. package/dist/commands/openapi-export.js +359 -0
  122. package/dist/commands/openapi-export.js.map +1 -0
  123. package/dist/commands/perf-analyzer.d.ts +8 -0
  124. package/dist/commands/perf-analyzer.js +423 -0
  125. package/dist/commands/perf-analyzer.js.map +1 -0
  126. package/dist/commands/rate-limiting.d.ts +10 -0
  127. package/dist/commands/rate-limiting.js +953 -0
  128. package/dist/commands/rate-limiting.js.map +1 -0
  129. package/dist/commands/recipe-plugin.d.ts +56 -0
  130. package/dist/commands/recipe-plugin.js +315 -0
  131. package/dist/commands/recipe-plugin.js.map +1 -0
  132. package/dist/commands/recipe.d.ts +6 -0
  133. package/dist/commands/recipe.js +3941 -0
  134. package/dist/commands/recipe.js.map +1 -0
  135. package/dist/commands/recipes/elasticsearch.recipe.d.ts +1 -0
  136. package/dist/commands/recipes/elasticsearch.recipe.js +761 -0
  137. package/dist/commands/recipes/elasticsearch.recipe.js.map +1 -0
  138. package/dist/commands/recipes/event-sourcing.recipe.d.ts +1 -0
  139. package/dist/commands/recipes/event-sourcing.recipe.js +889 -0
  140. package/dist/commands/recipes/event-sourcing.recipe.js.map +1 -0
  141. package/dist/commands/recipes/index.d.ts +7 -0
  142. package/dist/commands/recipes/index.js +24 -0
  143. package/dist/commands/recipes/index.js.map +1 -0
  144. package/dist/commands/recipes/message-queue.recipe.d.ts +1 -0
  145. package/dist/commands/recipes/message-queue.recipe.js +706 -0
  146. package/dist/commands/recipes/message-queue.recipe.js.map +1 -0
  147. package/dist/commands/recipes/middleware.recipe.d.ts +1 -0
  148. package/dist/commands/recipes/middleware.recipe.js +383 -0
  149. package/dist/commands/recipes/middleware.recipe.js.map +1 -0
  150. package/dist/commands/recipes/multi-tenancy.recipe.d.ts +1 -0
  151. package/dist/commands/recipes/multi-tenancy.recipe.js +520 -0
  152. package/dist/commands/recipes/multi-tenancy.recipe.js.map +1 -0
  153. package/dist/commands/recipes/oauth2.recipe.d.ts +1 -0
  154. package/dist/commands/recipes/oauth2.recipe.js +472 -0
  155. package/dist/commands/recipes/oauth2.recipe.js.map +1 -0
  156. package/dist/commands/recipes/websocket.recipe.d.ts +1 -0
  157. package/dist/commands/recipes/websocket.recipe.js +453 -0
  158. package/dist/commands/recipes/websocket.recipe.js.map +1 -0
  159. package/dist/commands/resilience-patterns.d.ts +13 -0
  160. package/dist/commands/resilience-patterns.js +1029 -0
  161. package/dist/commands/resilience-patterns.js.map +1 -0
  162. package/dist/commands/security-patterns.d.ts +11 -0
  163. package/dist/commands/security-patterns.js +2233 -0
  164. package/dist/commands/security-patterns.js.map +1 -0
  165. package/dist/commands/template-debug.d.ts +27 -0
  166. package/dist/commands/template-debug.js +388 -0
  167. package/dist/commands/template-debug.js.map +1 -0
  168. package/dist/commands/test-factory-full.d.ts +9 -0
  169. package/dist/commands/test-factory-full.js +1570 -0
  170. package/dist/commands/test-factory-full.js.map +1 -0
  171. package/dist/commands/test-scaffold.d.ts +7 -0
  172. package/dist/commands/test-scaffold.js +621 -0
  173. package/dist/commands/test-scaffold.js.map +1 -0
  174. package/dist/index.js +1088 -0
  175. package/dist/index.js.map +1 -1
  176. package/dist/templates/ai-context/CLAUDE.md.hbs +158 -0
  177. package/dist/templates/ai-context/conventions.md.hbs +154 -0
  178. package/dist/templates/command/create-command.hbs +6 -14
  179. package/dist/templates/command/delete-command.hbs +19 -0
  180. package/dist/templates/command/update-command.hbs +24 -0
  181. package/dist/templates/controller/controller.hbs +64 -17
  182. package/dist/templates/dto/create-dto.hbs +29 -5
  183. package/dist/templates/dto/filter-dto.hbs +52 -0
  184. package/dist/templates/dto/filter-query.dto.hbs +148 -0
  185. package/dist/templates/dto/paginated-response.dto.hbs +29 -0
  186. package/dist/templates/dto/pagination-query.dto.hbs +30 -0
  187. package/dist/templates/dto/response-dto.hbs +38 -0
  188. package/dist/templates/dto/update-dto.hbs +11 -0
  189. package/dist/templates/entity/entity.hbs +32 -1
  190. package/dist/templates/event/domain-event.hbs +33 -7
  191. package/dist/templates/event/event-handler.hbs +40 -0
  192. package/dist/templates/exception/base-exceptions.hbs +69 -0
  193. package/dist/templates/exception/entity-not-found.exception.hbs +7 -0
  194. package/dist/templates/mapper/mapper.hbs +49 -24
  195. package/dist/templates/module/module.hbs +34 -10
  196. package/dist/templates/orm-entity/orm-entity.hbs +63 -12
  197. package/dist/templates/prisma/prisma-mapper.hbs +71 -0
  198. package/dist/templates/prisma/prisma-repository.hbs +114 -0
  199. package/dist/templates/prisma/prisma-schema.hbs +20 -0
  200. package/dist/templates/prisma/prisma-service.hbs +51 -0
  201. package/dist/templates/query/get-all.query.hbs +50 -0
  202. package/dist/templates/query/get-by-id.query.hbs +31 -0
  203. package/dist/templates/repository/repository.hbs +55 -13
  204. package/dist/templates/resolver/graphql-input.hbs +54 -0
  205. package/dist/templates/resolver/graphql-type.hbs +58 -0
  206. package/dist/templates/resolver/pagination-args.hbs +33 -0
  207. package/dist/templates/resolver/resolver.hbs +62 -0
  208. package/dist/templates/shared/prisma-query-builder.util.hbs +189 -0
  209. package/dist/templates/shared/query-builder.util.hbs +218 -0
  210. package/dist/templates/test/controller.spec.hbs +124 -0
  211. package/dist/templates/test/repository.spec.hbs +158 -0
  212. package/dist/templates/test/usecase.spec.hbs +116 -0
  213. package/dist/templates/usecase/create-usecase.hbs +19 -7
  214. package/dist/templates/usecase/delete-usecase.hbs +17 -0
  215. package/dist/templates/usecase/update-usecase.hbs +31 -0
  216. package/dist/utils/config.utils.d.ts +45 -0
  217. package/dist/utils/config.utils.js +211 -0
  218. package/dist/utils/config.utils.js.map +1 -0
  219. package/dist/utils/error.utils.d.ts +145 -0
  220. package/dist/utils/error.utils.js +422 -0
  221. package/dist/utils/error.utils.js.map +1 -0
  222. package/dist/utils/field.utils.d.ts +54 -0
  223. package/dist/utils/field.utils.js +389 -0
  224. package/dist/utils/field.utils.js.map +1 -0
  225. package/dist/utils/file.utils.d.ts +19 -8
  226. package/dist/utils/file.utils.js +135 -4
  227. package/dist/utils/file.utils.js.map +1 -1
  228. package/dist/utils/idempotency.utils.d.ts +123 -0
  229. package/dist/utils/idempotency.utils.js +444 -0
  230. package/dist/utils/idempotency.utils.js.map +1 -0
  231. package/dist/utils/naming.utils.js +24 -5
  232. package/dist/utils/naming.utils.js.map +1 -1
  233. package/dist/utils/performance.utils.d.ts +37 -0
  234. package/dist/utils/performance.utils.js +158 -0
  235. package/dist/utils/performance.utils.js.map +1 -0
  236. package/dist/utils/relation.utils.d.ts +92 -0
  237. package/dist/utils/relation.utils.js +388 -0
  238. package/dist/utils/relation.utils.js.map +1 -0
  239. package/dist/utils/rollback.utils.d.ts +49 -0
  240. package/dist/utils/rollback.utils.js +306 -0
  241. package/dist/utils/rollback.utils.js.map +1 -0
  242. package/dist/utils/schema.utils.d.ts +123 -0
  243. package/dist/utils/schema.utils.js +419 -0
  244. package/dist/utils/schema.utils.js.map +1 -0
  245. package/dist/utils/security.utils.d.ts +57 -0
  246. package/dist/utils/security.utils.js +315 -0
  247. package/dist/utils/security.utils.js.map +1 -0
  248. package/dist/utils/template-engine.utils.d.ts +80 -0
  249. package/dist/utils/template-engine.utils.js +463 -0
  250. package/dist/utils/template-engine.utils.js.map +1 -0
  251. package/dist/utils/validation-registry.utils.d.ts +160 -0
  252. package/dist/utils/validation-registry.utils.js +526 -0
  253. package/dist/utils/validation-registry.utils.js.map +1 -0
  254. package/package.json +3 -1
@@ -0,0 +1,389 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.parseFields = parseFields;
4
+ exports.generateFieldsTemplateData = generateFieldsTemplateData;
5
+ const naming_utils_1 = require("./naming.utils");
6
+ /**
7
+ * Parse field definitions from CLI --fields flag
8
+ * Format: "name:type:modifier1:modifier2"
9
+ *
10
+ * Examples:
11
+ * "name:string" -> required string
12
+ * "email:string:unique" -> required unique string with IsEmail
13
+ * "age:number:optional" -> optional number
14
+ * "status:enum:active,inactive" -> enum with values
15
+ * "tags:string[]" -> string array
16
+ * "userId:uuid:relation" -> foreign key relation
17
+ */
18
+ function parseFields(fieldsString) {
19
+ if (!fieldsString || fieldsString.trim() === '') {
20
+ return { fields: [], hasEmail: false, hasDate: false, hasEnum: false, hasRelation: false };
21
+ }
22
+ const fieldDefs = fieldsString.split(' ').filter(f => f.trim() !== '');
23
+ let hasEmail = false;
24
+ let hasDate = false;
25
+ let hasEnum = false;
26
+ let hasRelation = false;
27
+ const fields = fieldDefs.map(fieldDef => {
28
+ const parts = fieldDef.split(':');
29
+ const name = parts[0];
30
+ const typeWithArray = parts[1] || 'string';
31
+ const modifiers = parts.slice(2);
32
+ const isArray = typeWithArray.endsWith('[]');
33
+ const baseType = isArray ? typeWithArray.slice(0, -2) : typeWithArray;
34
+ const isOptional = modifiers.includes('optional');
35
+ const isUnique = modifiers.includes('unique');
36
+ // Check for relation type
37
+ const relationTypes = ['OneToOne', 'OneToMany', 'ManyToOne', 'ManyToMany'];
38
+ const relationTypeModifier = modifiers.find(m => relationTypes.includes(m));
39
+ const isRelation = baseType === 'relation' || modifiers.includes('relation') || !!relationTypeModifier;
40
+ // Parse relation details
41
+ let relationTarget;
42
+ let relationType;
43
+ let relationInverse;
44
+ if (isRelation) {
45
+ if (baseType === 'relation') {
46
+ // Format: fieldName:relation:TargetEntity:RelationType:inverseSide
47
+ relationTarget = modifiers[0]; // First modifier is target entity
48
+ relationType = (modifiers.find(m => relationTypes.includes(m)) || 'ManyToOne');
49
+ // Find inverse (anything that's not a relation type or target)
50
+ relationInverse = modifiers.find(m => !relationTypes.includes(m) && m !== relationTarget && m !== 'optional' && m !== 'unique');
51
+ }
52
+ else {
53
+ // Legacy format or uuid:relation
54
+ relationTarget = (0, naming_utils_1.toPascalCase)(name.replace(/Id$/, ''));
55
+ relationType = 'ManyToOne';
56
+ }
57
+ }
58
+ // Find enum values if present
59
+ const enumModifier = modifiers.find(m => m.includes(',') || (baseType === 'enum' && m && !relationTypes.includes(m)));
60
+ const enumValues = baseType === 'enum' && enumModifier ? enumModifier.split(',') : undefined;
61
+ // Determine TypeScript type
62
+ let tsType = mapToTsType(baseType, enumValues);
63
+ if (isArray)
64
+ tsType = `${tsType}[]`;
65
+ // Determine database type
66
+ const dbType = mapToDbType(baseType, isArray);
67
+ // Determine Prisma type
68
+ const prismaType = mapToPrismaType(baseType, isArray);
69
+ // Generate validators
70
+ const validators = generateValidators(name, baseType, isOptional, isUnique, isArray, enumValues);
71
+ // Track special types
72
+ if (name.toLowerCase().includes('email'))
73
+ hasEmail = true;
74
+ if (baseType === 'date' || baseType === 'datetime')
75
+ hasDate = true;
76
+ if (baseType === 'enum')
77
+ hasEnum = true;
78
+ if (isRelation)
79
+ hasRelation = true;
80
+ const camelCase = (0, naming_utils_1.toCamelCase)(name);
81
+ const snakeCase = (0, naming_utils_1.toSnakeCase)(name);
82
+ const pascalCase = name.charAt(0).toUpperCase() + (0, naming_utils_1.toCamelCase)(name).slice(1);
83
+ return {
84
+ name,
85
+ camelCase,
86
+ snakeCase,
87
+ pascalCase,
88
+ type: baseType,
89
+ tsType: isRelation ? (relationType === 'OneToMany' || relationType === 'ManyToMany' ? `${relationTarget}[]` : relationTarget || 'string') : tsType,
90
+ dbType,
91
+ prismaType,
92
+ isRequired: !isOptional,
93
+ isOptional,
94
+ isUnique,
95
+ isArray,
96
+ validators,
97
+ description: generateDescription(name, baseType),
98
+ example: generateExample(name, baseType),
99
+ enumValues,
100
+ isRelation,
101
+ relationType,
102
+ relationTarget,
103
+ relationInverse,
104
+ };
105
+ });
106
+ return { fields, hasEmail, hasDate, hasEnum, hasRelation };
107
+ }
108
+ function mapToTsType(type, enumValues) {
109
+ const typeMap = {
110
+ string: 'string',
111
+ number: 'number',
112
+ int: 'number',
113
+ integer: 'number',
114
+ float: 'number',
115
+ decimal: 'number',
116
+ boolean: 'boolean',
117
+ bool: 'boolean',
118
+ date: 'Date',
119
+ datetime: 'Date',
120
+ timestamp: 'Date',
121
+ uuid: 'string',
122
+ json: 'Record<string, any>',
123
+ text: 'string',
124
+ enum: enumValues ? enumValues.map(v => `'${v}'`).join(' | ') : 'string',
125
+ };
126
+ return typeMap[type.toLowerCase()] || 'string';
127
+ }
128
+ function mapToDbType(type, isArray) {
129
+ const typeMap = {
130
+ string: 'varchar',
131
+ number: 'int',
132
+ int: 'int',
133
+ integer: 'int',
134
+ float: 'float',
135
+ decimal: 'decimal',
136
+ boolean: 'boolean',
137
+ bool: 'boolean',
138
+ date: 'date',
139
+ datetime: 'timestamp',
140
+ timestamp: 'timestamp',
141
+ uuid: 'uuid',
142
+ json: 'jsonb',
143
+ text: 'text',
144
+ enum: 'varchar',
145
+ };
146
+ const dbType = typeMap[type.toLowerCase()] || 'varchar';
147
+ return isArray ? 'jsonb' : dbType;
148
+ }
149
+ function mapToPrismaType(type, isArray) {
150
+ const typeMap = {
151
+ string: 'String',
152
+ number: 'Int',
153
+ int: 'Int',
154
+ integer: 'Int',
155
+ float: 'Float',
156
+ decimal: 'Decimal',
157
+ boolean: 'Boolean',
158
+ bool: 'Boolean',
159
+ date: 'DateTime',
160
+ datetime: 'DateTime',
161
+ timestamp: 'DateTime',
162
+ uuid: 'String',
163
+ json: 'Json',
164
+ text: 'String',
165
+ enum: 'String',
166
+ };
167
+ const prismaType = typeMap[type.toLowerCase()] || 'String';
168
+ return isArray ? `${prismaType}[]` : prismaType;
169
+ }
170
+ function generateValidators(name, type, isOptional, isUnique, isArray, enumValues) {
171
+ const validators = [];
172
+ const nameLower = name.toLowerCase();
173
+ if (isOptional) {
174
+ validators.push('@IsOptional()');
175
+ }
176
+ else {
177
+ validators.push('@IsNotEmpty()');
178
+ validators.push('@IsDefined()');
179
+ }
180
+ // Type-specific validators with security-focused length limits
181
+ switch (type.toLowerCase()) {
182
+ case 'string':
183
+ case 'text':
184
+ validators.push('@IsString()');
185
+ // Field-specific validation with length limits
186
+ if (nameLower.includes('email')) {
187
+ validators.push('@IsEmail()');
188
+ validators.push('@MaxLength(254)'); // RFC 5321
189
+ }
190
+ else if (nameLower.includes('url') || nameLower.includes('link')) {
191
+ validators.push('@IsUrl()');
192
+ validators.push('@MaxLength(2048)'); // Common browser limit
193
+ }
194
+ else if (nameLower.includes('phone') || nameLower.includes('mobile')) {
195
+ validators.push("@Matches(/^[+]?[0-9\\\\s\\\\-().]+$/, { message: 'Invalid phone format' })");
196
+ validators.push('@MaxLength(20)'); // E.164 max
197
+ }
198
+ else if (nameLower.includes('name') && !nameLower.includes('username')) {
199
+ validators.push('@MaxLength(100)');
200
+ validators.push('@MinLength(1)');
201
+ }
202
+ else if (nameLower.includes('username') || nameLower.includes('login')) {
203
+ validators.push("@Matches(/^[a-zA-Z0-9_-]+$/, { message: 'Username can only contain letters, numbers, underscores, and hyphens' })");
204
+ validators.push('@MaxLength(50)');
205
+ validators.push('@MinLength(3)');
206
+ }
207
+ else if (nameLower.includes('password')) {
208
+ validators.push('@MinLength(8)');
209
+ validators.push('@MaxLength(128)');
210
+ }
211
+ else if (nameLower.includes('description') || nameLower.includes('content') || type.toLowerCase() === 'text') {
212
+ validators.push('@MaxLength(10000)'); // Long text fields
213
+ }
214
+ else if (nameLower.includes('title') || nameLower.includes('subject')) {
215
+ validators.push('@MaxLength(255)');
216
+ }
217
+ else if (nameLower.includes('code') || nameLower.includes('slug')) {
218
+ validators.push("@Matches(/^[a-zA-Z0-9_-]+$/, { message: 'Invalid format - use only letters, numbers, underscores, and hyphens' })");
219
+ validators.push('@MaxLength(100)');
220
+ }
221
+ else if (nameLower.includes('ip') || nameLower.includes('address')) {
222
+ validators.push('@MaxLength(45)'); // IPv6 max
223
+ }
224
+ else {
225
+ // Default string length limit to prevent DoS
226
+ validators.push('@MaxLength(1000)');
227
+ }
228
+ break;
229
+ case 'number':
230
+ case 'int':
231
+ case 'integer':
232
+ validators.push('@IsNumber()');
233
+ validators.push('@IsInt()');
234
+ // Add safe integer bounds
235
+ validators.push('@Min(-2147483648)');
236
+ validators.push('@Max(2147483647)');
237
+ break;
238
+ case 'float':
239
+ case 'decimal':
240
+ validators.push('@IsNumber()');
241
+ // Prevent extremely large numbers
242
+ validators.push('@Min(-1e15)');
243
+ validators.push('@Max(1e15)');
244
+ break;
245
+ case 'boolean':
246
+ case 'bool':
247
+ validators.push('@IsBoolean()');
248
+ break;
249
+ case 'date':
250
+ case 'datetime':
251
+ case 'timestamp':
252
+ validators.push('@IsDate()');
253
+ validators.push('@Type(() => Date)');
254
+ break;
255
+ case 'uuid':
256
+ validators.push('@IsUUID()');
257
+ break;
258
+ case 'enum':
259
+ if (enumValues) {
260
+ validators.push(`@IsIn([${enumValues.map(v => `'${v}'`).join(', ')}])`);
261
+ }
262
+ break;
263
+ case 'json':
264
+ validators.push('@IsObject()');
265
+ break;
266
+ }
267
+ if (isArray) {
268
+ validators.push('@IsArray()');
269
+ validators.push('@ArrayMaxSize(100)'); // Limit array size by default
270
+ }
271
+ return validators;
272
+ }
273
+ function generateRelationDecorator(f) {
274
+ const { camelCase, relationType, relationTarget, relationInverse } = f;
275
+ switch (relationType) {
276
+ case 'OneToOne':
277
+ return ` @OneToOne(() => ${relationTarget}${relationInverse ? `, (${relationInverse.toLowerCase()}) => ${relationInverse.toLowerCase()}.${camelCase}` : ''})\n @JoinColumn()\n ${camelCase}${f.isOptional ? '?' : ''}: ${relationTarget};`;
278
+ case 'OneToMany':
279
+ return ` @OneToMany(() => ${relationTarget}, (${relationTarget.toLowerCase()}) => ${relationTarget.toLowerCase()}.${relationInverse || camelCase})\n ${camelCase}: ${relationTarget}[];`;
280
+ case 'ManyToOne':
281
+ return ` @ManyToOne(() => ${relationTarget}${relationInverse ? `, (${relationTarget.toLowerCase()}) => ${relationTarget.toLowerCase()}.${relationInverse}` : ''})\n @JoinColumn({ name: '${f.snakeCase}_id' })\n ${camelCase}${f.isOptional ? '?' : ''}: ${relationTarget};`;
282
+ case 'ManyToMany':
283
+ return ` @ManyToMany(() => ${relationTarget})\n @JoinTable()\n ${camelCase}: ${relationTarget}[];`;
284
+ default:
285
+ return ` // Unknown relation type for ${camelCase}`;
286
+ }
287
+ }
288
+ function generateDescription(name, type) {
289
+ const readableName = name
290
+ .replace(/([A-Z])/g, ' $1')
291
+ .replace(/[_-]/g, ' ')
292
+ .trim()
293
+ .toLowerCase();
294
+ return `The ${readableName} of the entity`;
295
+ }
296
+ function generateExample(name, type) {
297
+ const lowerName = name.toLowerCase();
298
+ if (lowerName.includes('email'))
299
+ return 'user@example.com';
300
+ if (lowerName.includes('name'))
301
+ return 'John Doe';
302
+ if (lowerName.includes('phone'))
303
+ return '+1234567890';
304
+ if (lowerName.includes('url') || lowerName.includes('link'))
305
+ return 'https://example.com';
306
+ if (lowerName.includes('price') || lowerName.includes('amount'))
307
+ return '99.99';
308
+ if (lowerName.includes('count') || lowerName.includes('quantity'))
309
+ return '10';
310
+ switch (type.toLowerCase()) {
311
+ case 'string':
312
+ case 'text':
313
+ return 'Sample text';
314
+ case 'number':
315
+ case 'int':
316
+ case 'integer':
317
+ return '42';
318
+ case 'float':
319
+ case 'decimal':
320
+ return '3.14';
321
+ case 'boolean':
322
+ case 'bool':
323
+ return 'true';
324
+ case 'uuid':
325
+ return '550e8400-e29b-41d4-a716-446655440000';
326
+ default:
327
+ return undefined;
328
+ }
329
+ }
330
+ /**
331
+ * Generate field-aware template data
332
+ */
333
+ function generateFieldsTemplateData(fields) {
334
+ const entityProperties = fields
335
+ .map(f => ` public readonly ${f.camelCase}${f.isOptional ? '?' : ''}: ${f.tsType};`)
336
+ .join('\n');
337
+ const entityPropsInterface = fields
338
+ .map(f => ` ${f.camelCase}${f.isOptional ? '?' : ''}: ${f.tsType};`)
339
+ .join('\n');
340
+ const dtoProperties = fields
341
+ .map(f => {
342
+ const decorators = [
343
+ ` @ApiProperty({ description: "${f.description}"${f.example ? `, example: "${f.example}"` : ''} })`,
344
+ ...f.validators.map(v => ` ${v}`),
345
+ ].join('\n');
346
+ return `${decorators}\n ${f.snakeCase}${f.isOptional ? '?' : ''}: ${f.tsType};`;
347
+ })
348
+ .join('\n\n');
349
+ const ormColumns = fields
350
+ .map(f => {
351
+ if (f.isRelation && f.relationType && f.relationTarget) {
352
+ // Generate relation decorator
353
+ const relationDecorator = generateRelationDecorator(f);
354
+ return relationDecorator;
355
+ }
356
+ const columnOptions = [];
357
+ if (f.isUnique)
358
+ columnOptions.push('unique: true');
359
+ if (f.isOptional)
360
+ columnOptions.push('nullable: true');
361
+ if (f.dbType !== 'varchar')
362
+ columnOptions.push(`type: "${f.dbType}"`);
363
+ const optionsStr = columnOptions.length > 0 ? `{ ${columnOptions.join(', ')} }` : '';
364
+ return ` @Column(${optionsStr})\n ${f.snakeCase}${f.isOptional ? '?' : ''}: ${f.tsType};`;
365
+ })
366
+ .join('\n\n');
367
+ const migrationColumns = fields
368
+ .map(f => {
369
+ return ` {
370
+ name: "${f.snakeCase}",
371
+ type: "${f.dbType}",
372
+ isNullable: ${f.isOptional},
373
+ ${f.isUnique ? 'isUnique: true,' : ''}
374
+ },`;
375
+ })
376
+ .join('\n');
377
+ const responseProperties = fields
378
+ .map(f => ` @ApiProperty({ description: "${f.description}" })\n @Expose()\n ${f.camelCase}: ${f.tsType};`)
379
+ .join('\n\n');
380
+ return {
381
+ entityProperties,
382
+ entityPropsInterface,
383
+ dtoProperties,
384
+ ormColumns,
385
+ migrationColumns,
386
+ responseProperties,
387
+ };
388
+ }
389
+ //# sourceMappingURL=field.utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"field.utils.js","sourceRoot":"","sources":["../../src/utils/field.utils.ts"],"names":[],"mappings":"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"}
@@ -1,3 +1,4 @@
1
+ import { FieldDefinition } from './field.utils';
1
2
  export interface TemplateData {
2
3
  entityName: string;
3
4
  entityNamePascal: string;
@@ -5,22 +6,32 @@ export interface TemplateData {
5
6
  entityNameKebab: string;
6
7
  entityNameSnake: string;
7
8
  entityNamePlural: string;
9
+ entityNamePluralPascal: string;
10
+ entityNamePluralCamel: string;
8
11
  entityNamePluralKebab: string;
9
12
  tableName: string;
10
13
  moduleName: string;
11
14
  moduleNameKebab: string;
12
- properties?: Array<{
13
- name: string;
14
- type: string;
15
- isRequired: boolean;
16
- isRelation: boolean;
17
- }>;
15
+ moduleNamePascal: string;
16
+ fields?: FieldDefinition[];
17
+ hasFields: boolean;
18
+ entityProperties?: string;
19
+ entityPropsInterface?: string;
20
+ dtoProperties?: string;
21
+ ormColumns?: string;
22
+ migrationColumns?: string;
23
+ responseProperties?: string;
24
+ hasRelations: boolean;
25
+ relationImports?: string;
18
26
  }
19
- export declare function prepareTemplateData(entityName: string, moduleName: string): TemplateData;
27
+ export declare function prepareTemplateData(entityName: string, moduleName: string, fieldsString?: string): TemplateData;
20
28
  export declare function ensureDir(dirPath: string): Promise<void>;
21
29
  export declare function fileExists(filePath: string): Promise<boolean>;
22
30
  export declare function writeFile(filePath: string, content: string): Promise<void>;
23
31
  export declare function readTemplate(templatePath: string): Promise<string>;
24
32
  export declare function compileTemplate(template: string, data: TemplateData): string;
25
- export declare function generateFromTemplate(templatePath: string, outputPath: string, data: TemplateData): Promise<void>;
33
+ export declare function generateFromTemplate(templatePath: string, outputPath: string, data: TemplateData, dryRun?: boolean): Promise<string>;
34
+ export declare function resetDryRunFiles(): void;
35
+ export declare function getDryRunFiles(): string[];
36
+ export declare function addDryRunFile(filePath: string): void;
26
37
  export declare function getModulePath(basePath: string, moduleName: string): string;
@@ -43,12 +43,47 @@ exports.writeFile = writeFile;
43
43
  exports.readTemplate = readTemplate;
44
44
  exports.compileTemplate = compileTemplate;
45
45
  exports.generateFromTemplate = generateFromTemplate;
46
+ exports.resetDryRunFiles = resetDryRunFiles;
47
+ exports.getDryRunFiles = getDryRunFiles;
48
+ exports.addDryRunFile = addDryRunFile;
46
49
  exports.getModulePath = getModulePath;
47
50
  const fs = __importStar(require("fs-extra"));
48
51
  const path = __importStar(require("path"));
49
52
  const handlebars_1 = __importDefault(require("handlebars"));
50
53
  const naming_utils_1 = require("./naming.utils");
51
- function prepareTemplateData(entityName, moduleName) {
54
+ const field_utils_1 = require("./field.utils");
55
+ // Register Handlebars helpers
56
+ handlebars_1.default.registerHelper('eq', function (a, b) {
57
+ return a === b;
58
+ });
59
+ handlebars_1.default.registerHelper('lowercase', function (str) {
60
+ return typeof str === 'string' ? str.toLowerCase() : str;
61
+ });
62
+ handlebars_1.default.registerHelper('uppercase', function (str) {
63
+ return typeof str === 'string' ? str.toUpperCase() : str;
64
+ });
65
+ handlebars_1.default.registerHelper('pascalCase', function (str) {
66
+ return typeof str === 'string' ? (0, naming_utils_1.toPascalCase)(str) : str;
67
+ });
68
+ handlebars_1.default.registerHelper('camelCase', function (str) {
69
+ return typeof str === 'string' ? (0, naming_utils_1.toCamelCase)(str) : str;
70
+ });
71
+ handlebars_1.default.registerHelper('kebabCase', function (str) {
72
+ return typeof str === 'string' ? (0, naming_utils_1.toKebabCase)(str) : str;
73
+ });
74
+ function prepareTemplateData(entityName, moduleName, fieldsString) {
75
+ const parsedFields = fieldsString ? (0, field_utils_1.parseFields)(fieldsString) : null;
76
+ const fieldsTemplateData = parsedFields?.fields.length
77
+ ? (0, field_utils_1.generateFieldsTemplateData)(parsedFields.fields)
78
+ : null;
79
+ // Check for relations and generate imports
80
+ const relationFields = parsedFields?.fields.filter(f => f.isRelation) || [];
81
+ const hasRelations = relationFields.length > 0;
82
+ // Generate unique relation imports
83
+ const relationTargets = [...new Set(relationFields.map(f => f.relationTarget).filter(Boolean))];
84
+ const relationImports = relationTargets.length > 0
85
+ ? relationTargets.map(target => `import { ${target}OrmEntity } from "./${(0, naming_utils_1.toKebabCase)(target)}.orm-entity";`).join('\n')
86
+ : '';
52
87
  return {
53
88
  entityName,
54
89
  entityNamePascal: (0, naming_utils_1.toPascalCase)(entityName),
@@ -56,17 +91,78 @@ function prepareTemplateData(entityName, moduleName) {
56
91
  entityNameKebab: (0, naming_utils_1.toKebabCase)(entityName),
57
92
  entityNameSnake: (0, naming_utils_1.toSnakeCase)(entityName),
58
93
  entityNamePlural: (0, naming_utils_1.toPlural)((0, naming_utils_1.toPascalCase)(entityName)),
94
+ entityNamePluralPascal: (0, naming_utils_1.toPlural)((0, naming_utils_1.toPascalCase)(entityName)),
95
+ entityNamePluralCamel: (0, naming_utils_1.toCamelCase)((0, naming_utils_1.toPlural)(entityName)),
59
96
  entityNamePluralKebab: (0, naming_utils_1.toKebabCase)((0, naming_utils_1.toPlural)(entityName)),
60
97
  tableName: (0, naming_utils_1.toTableName)(entityName),
61
- moduleName,
98
+ moduleName: (0, naming_utils_1.toPascalCase)(moduleName),
62
99
  moduleNameKebab: (0, naming_utils_1.toKebabCase)(moduleName),
100
+ moduleNamePascal: (0, naming_utils_1.toPascalCase)(moduleName),
101
+ // Field-aware properties
102
+ fields: parsedFields?.fields || [],
103
+ hasFields: (parsedFields?.fields.length || 0) > 0,
104
+ entityProperties: fieldsTemplateData?.entityProperties || '',
105
+ entityPropsInterface: fieldsTemplateData?.entityPropsInterface || '',
106
+ dtoProperties: fieldsTemplateData?.dtoProperties || '',
107
+ ormColumns: fieldsTemplateData?.ormColumns || '',
108
+ migrationColumns: fieldsTemplateData?.migrationColumns || '',
109
+ responseProperties: fieldsTemplateData?.responseProperties || '',
110
+ // Relation properties
111
+ hasRelations,
112
+ relationImports,
63
113
  };
64
114
  }
115
+ /**
116
+ * Validate that a path is safe and within allowed bounds
117
+ * Prevents path traversal attacks (OWASP A01:2021)
118
+ */
119
+ function validatePath(targetPath, basePath) {
120
+ if (!targetPath || typeof targetPath !== 'string') {
121
+ throw new Error('Path must be a non-empty string');
122
+ }
123
+ // Check for null bytes (can bypass security checks)
124
+ if (targetPath.includes('\0')) {
125
+ throw new Error('Path contains null byte');
126
+ }
127
+ // Check for path traversal patterns
128
+ const normalizedPath = path.normalize(targetPath);
129
+ if (normalizedPath.includes('..')) {
130
+ // After normalization, if .. still exists, it's going above root
131
+ const resolvedPath = path.resolve(targetPath);
132
+ const cwdPath = basePath || process.cwd();
133
+ if (!resolvedPath.startsWith(cwdPath)) {
134
+ throw new Error('Path traversal detected: path escapes base directory');
135
+ }
136
+ }
137
+ // Block access to sensitive system directories
138
+ const blockedPatterns = [
139
+ /^\/etc\//i,
140
+ /^\/proc\//i,
141
+ /^\/sys\//i,
142
+ /^\/dev\//i,
143
+ /^\/root\//i,
144
+ /^C:\\Windows/i,
145
+ /^C:\\Program Files/i,
146
+ /\.env$/i,
147
+ /\.git\//i,
148
+ /\.ssh\//i,
149
+ /id_rsa/i,
150
+ /\.pem$/i,
151
+ /\.key$/i,
152
+ ];
153
+ for (const pattern of blockedPatterns) {
154
+ if (pattern.test(normalizedPath)) {
155
+ throw new Error(`Access to path is blocked: ${normalizedPath}`);
156
+ }
157
+ }
158
+ }
65
159
  async function ensureDir(dirPath) {
160
+ validatePath(dirPath);
66
161
  await fs.ensureDir(dirPath);
67
162
  }
68
163
  async function fileExists(filePath) {
69
164
  try {
165
+ validatePath(filePath);
70
166
  await fs.access(filePath);
71
167
  return true;
72
168
  }
@@ -75,20 +171,55 @@ async function fileExists(filePath) {
75
171
  }
76
172
  }
77
173
  async function writeFile(filePath, content) {
174
+ validatePath(filePath);
175
+ // Validate content size (prevent DoS via huge files)
176
+ const maxContentSize = 10 * 1024 * 1024; // 10MB max
177
+ if (content && content.length > maxContentSize) {
178
+ throw new Error(`Content exceeds maximum size of ${maxContentSize} bytes`);
179
+ }
78
180
  await ensureDir(path.dirname(filePath));
79
181
  await fs.writeFile(filePath, content, 'utf-8');
80
182
  }
81
183
  async function readTemplate(templatePath) {
184
+ validatePath(templatePath);
185
+ // Get file stats to check size before reading
186
+ try {
187
+ const stats = await fs.stat(templatePath);
188
+ const maxTemplateSize = 1024 * 1024; // 1MB max for templates
189
+ if (stats.size > maxTemplateSize) {
190
+ throw new Error(`Template file exceeds maximum size of ${maxTemplateSize} bytes`);
191
+ }
192
+ }
193
+ catch (error) {
194
+ if (error.code === 'ENOENT') {
195
+ throw new Error(`Template file not found: ${templatePath}`);
196
+ }
197
+ throw error;
198
+ }
82
199
  return fs.readFile(templatePath, 'utf-8');
83
200
  }
84
201
  function compileTemplate(template, data) {
85
202
  const compiledTemplate = handlebars_1.default.compile(template);
86
203
  return compiledTemplate(data);
87
204
  }
88
- async function generateFromTemplate(templatePath, outputPath, data) {
205
+ async function generateFromTemplate(templatePath, outputPath, data, dryRun = false) {
89
206
  const template = await readTemplate(templatePath);
90
207
  const content = compileTemplate(template, data);
91
- await writeFile(outputPath, content);
208
+ if (!dryRun) {
209
+ await writeFile(outputPath, content);
210
+ }
211
+ return outputPath;
212
+ }
213
+ // Dry run tracking
214
+ let dryRunFiles = [];
215
+ function resetDryRunFiles() {
216
+ dryRunFiles = [];
217
+ }
218
+ function getDryRunFiles() {
219
+ return [...dryRunFiles];
220
+ }
221
+ function addDryRunFile(filePath) {
222
+ dryRunFiles.push(filePath);
92
223
  }
93
224
  function getModulePath(basePath, moduleName) {
94
225
  return path.join(basePath, 'src', 'modules', (0, naming_utils_1.toKebabCase)(moduleName));
@@ -1 +1 @@
1
- {"version":3,"file":"file.utils.js","sourceRoot":"","sources":["../../src/utils/file.utils.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAwBA,kDAaC;AAED,8BAEC;AAED,gCAOC;AAED,8BAGC;AAED,oCAEC;AAED,0CAGC;AAED,oDAQC;AAED,sCAEC;AA9ED,6CAA+B;AAC/B,2CAA6B;AAC7B,4DAAoC;AACpC,iDAA4G;AAqB5G,SAAgB,mBAAmB,CAAC,UAAkB,EAAE,UAAkB;IACxE,OAAO;QACL,UAAU;QACV,gBAAgB,EAAE,IAAA,2BAAY,EAAC,UAAU,CAAC;QAC1C,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,gBAAgB,EAAE,IAAA,uBAAQ,EAAC,IAAA,2BAAY,EAAC,UAAU,CAAC,CAAC;QACpD,qBAAqB,EAAE,IAAA,0BAAW,EAAC,IAAA,uBAAQ,EAAC,UAAU,CAAC,CAAC;QACxD,SAAS,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QAClC,UAAU;QACV,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;KACzC,CAAC;AACJ,CAAC;AAEM,KAAK,UAAU,SAAS,CAAC,OAAe;IAC7C,MAAM,EAAE,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;AAC9B,CAAC;AAEM,KAAK,UAAU,UAAU,CAAC,QAAgB;IAC/C,IAAI,CAAC;QACH,MAAM,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAEM,KAAK,UAAU,SAAS,CAAC,QAAgB,EAAE,OAAe;IAC/D,MAAM,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC;IACxC,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AACjD,CAAC;AAEM,KAAK,UAAU,YAAY,CAAC,YAAoB;IACrD,OAAO,EAAE,CAAC,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;AAC5C,CAAC;AAED,SAAgB,eAAe,CAAC,QAAgB,EAAE,IAAkB;IAClE,MAAM,gBAAgB,GAAG,oBAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACtD,OAAO,gBAAgB,CAAC,IAAI,CAAC,CAAC;AAChC,CAAC;AAEM,KAAK,UAAU,oBAAoB,CACxC,YAAoB,EACpB,UAAkB,EAClB,IAAkB;IAElB,MAAM,QAAQ,GAAG,MAAM,YAAY,CAAC,YAAY,CAAC,CAAC;IAClD,MAAM,OAAO,GAAG,eAAe,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;IAChD,MAAM,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;AACvC,CAAC;AAED,SAAgB,aAAa,CAAC,QAAgB,EAAE,UAAkB;IAChE,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,EAAE,SAAS,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC,CAAC,CAAC;AACxE,CAAC"}
1
+ {"version":3,"file":"file.utils.js","sourceRoot":"","sources":["../../src/utils/file.utils.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA2DA,kDA2CC;AAoDD,8BAGC;AAED,gCAQC;AAED,8BAWC;AAED,oCAkBC;AAED,0CAGC;AAED,oDAcC;AAKD,4CAEC;AAED,wCAEC;AAED,sCAEC;AAED,sCAEC;AAhPD,6CAA+B;AAC/B,2CAA6B;AAC7B,4DAAoC;AACpC,iDAA4G;AAC5G,+CAAyF;AAEzF,8BAA8B;AAC9B,oBAAU,CAAC,cAAc,CAAC,IAAI,EAAE,UAAU,CAAC,EAAE,CAAC;IAC5C,OAAO,CAAC,KAAK,CAAC,CAAC;AACjB,CAAC,CAAC,CAAC;AAEH,oBAAU,CAAC,cAAc,CAAC,WAAW,EAAE,UAAU,GAAG;IAClD,OAAO,OAAO,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;AAC3D,CAAC,CAAC,CAAC;AAEH,oBAAU,CAAC,cAAc,CAAC,WAAW,EAAE,UAAU,GAAG;IAClD,OAAO,OAAO,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;AAC3D,CAAC,CAAC,CAAC;AAEH,oBAAU,CAAC,cAAc,CAAC,YAAY,EAAE,UAAU,GAAG;IACnD,OAAO,OAAO,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAA,2BAAY,EAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;AAC3D,CAAC,CAAC,CAAC;AAEH,oBAAU,CAAC,cAAc,CAAC,WAAW,EAAE,UAAU,GAAG;IAClD,OAAO,OAAO,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAA,0BAAW,EAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;AAC1D,CAAC,CAAC,CAAC;AAEH,oBAAU,CAAC,cAAc,CAAC,WAAW,EAAE,UAAU,GAAG;IAClD,OAAO,OAAO,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAA,0BAAW,EAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;AAC1D,CAAC,CAAC,CAAC;AA8BH,SAAgB,mBAAmB,CAAC,UAAkB,EAAE,UAAkB,EAAE,YAAqB;IAC/F,MAAM,YAAY,GAAG,YAAY,CAAC,CAAC,CAAC,IAAA,yBAAW,EAAC,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IACrE,MAAM,kBAAkB,GAAG,YAAY,EAAE,MAAM,CAAC,MAAM;QACpD,CAAC,CAAC,IAAA,wCAA0B,EAAC,YAAY,CAAC,MAAM,CAAC;QACjD,CAAC,CAAC,IAAI,CAAC;IAET,2CAA2C;IAC3C,MAAM,cAAc,GAAG,YAAY,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;IAC5E,MAAM,YAAY,GAAG,cAAc,CAAC,MAAM,GAAG,CAAC,CAAC;IAE/C,mCAAmC;IACnC,MAAM,eAAe,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IAChG,MAAM,eAAe,GAAG,eAAe,CAAC,MAAM,GAAG,CAAC;QAChD,CAAC,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,MAAM,uBAAuB,IAAA,0BAAW,EAAC,MAAO,CAAC,eAAe,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QACxH,CAAC,CAAC,EAAE,CAAC;IAEP,OAAO;QACL,UAAU;QACV,gBAAgB,EAAE,IAAA,2BAAY,EAAC,UAAU,CAAC;QAC1C,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,gBAAgB,EAAE,IAAA,uBAAQ,EAAC,IAAA,2BAAY,EAAC,UAAU,CAAC,CAAC;QACpD,sBAAsB,EAAE,IAAA,uBAAQ,EAAC,IAAA,2BAAY,EAAC,UAAU,CAAC,CAAC;QAC1D,qBAAqB,EAAE,IAAA,0BAAW,EAAC,IAAA,uBAAQ,EAAC,UAAU,CAAC,CAAC;QACxD,qBAAqB,EAAE,IAAA,0BAAW,EAAC,IAAA,uBAAQ,EAAC,UAAU,CAAC,CAAC;QACxD,SAAS,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QAClC,UAAU,EAAE,IAAA,2BAAY,EAAC,UAAU,CAAC;QACpC,eAAe,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC;QACxC,gBAAgB,EAAE,IAAA,2BAAY,EAAC,UAAU,CAAC;QAC1C,yBAAyB;QACzB,MAAM,EAAE,YAAY,EAAE,MAAM,IAAI,EAAE;QAClC,SAAS,EAAE,CAAC,YAAY,EAAE,MAAM,CAAC,MAAM,IAAI,CAAC,CAAC,GAAG,CAAC;QACjD,gBAAgB,EAAE,kBAAkB,EAAE,gBAAgB,IAAI,EAAE;QAC5D,oBAAoB,EAAE,kBAAkB,EAAE,oBAAoB,IAAI,EAAE;QACpE,aAAa,EAAE,kBAAkB,EAAE,aAAa,IAAI,EAAE;QACtD,UAAU,EAAE,kBAAkB,EAAE,UAAU,IAAI,EAAE;QAChD,gBAAgB,EAAE,kBAAkB,EAAE,gBAAgB,IAAI,EAAE;QAC5D,kBAAkB,EAAE,kBAAkB,EAAE,kBAAkB,IAAI,EAAE;QAChE,sBAAsB;QACtB,YAAY;QACZ,eAAe;KAChB,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,SAAS,YAAY,CAAC,UAAkB,EAAE,QAAiB;IACzD,IAAI,CAAC,UAAU,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE,CAAC;QAClD,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;IACrD,CAAC;IAED,oDAAoD;IACpD,IAAI,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC7C,CAAC;IAED,oCAAoC;IACpC,MAAM,cAAc,GAAG,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;IAClD,IAAI,cAAc,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QAClC,iEAAiE;QACjE,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAC9C,MAAM,OAAO,GAAG,QAAQ,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;QAE1C,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;YACtC,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;QAC1E,CAAC;IACH,CAAC;IAED,+CAA+C;IAC/C,MAAM,eAAe,GAAG;QACtB,WAAW;QACX,YAAY;QACZ,WAAW;QACX,WAAW;QACX,YAAY;QACZ,eAAe;QACf,qBAAqB;QACrB,SAAS;QACT,UAAU;QACV,UAAU;QACV,SAAS;QACT,SAAS;QACT,SAAS;KACV,CAAC;IAEF,KAAK,MAAM,OAAO,IAAI,eAAe,EAAE,CAAC;QACtC,IAAI,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CAAC,8BAA8B,cAAc,EAAE,CAAC,CAAC;QAClE,CAAC;IACH,CAAC;AACH,CAAC;AAEM,KAAK,UAAU,SAAS,CAAC,OAAe;IAC7C,YAAY,CAAC,OAAO,CAAC,CAAC;IACtB,MAAM,EAAE,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;AAC9B,CAAC;AAEM,KAAK,UAAU,UAAU,CAAC,QAAgB;IAC/C,IAAI,CAAC;QACH,YAAY,CAAC,QAAQ,CAAC,CAAC;QACvB,MAAM,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAEM,KAAK,UAAU,SAAS,CAAC,QAAgB,EAAE,OAAe;IAC/D,YAAY,CAAC,QAAQ,CAAC,CAAC;IAEvB,qDAAqD;IACrD,MAAM,cAAc,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,CAAC,WAAW;IACpD,IAAI,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,cAAc,EAAE,CAAC;QAC/C,MAAM,IAAI,KAAK,CAAC,mCAAmC,cAAc,QAAQ,CAAC,CAAC;IAC7E,CAAC;IAED,MAAM,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC;IACxC,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AACjD,CAAC;AAEM,KAAK,UAAU,YAAY,CAAC,YAAoB;IACrD,YAAY,CAAC,YAAY,CAAC,CAAC;IAE3B,8CAA8C;IAC9C,IAAI,CAAC;QACH,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC1C,MAAM,eAAe,GAAG,IAAI,GAAG,IAAI,CAAC,CAAC,wBAAwB;QAC7D,IAAI,KAAK,CAAC,IAAI,GAAG,eAAe,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CAAC,yCAAyC,eAAe,QAAQ,CAAC,CAAC;QACpF,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAK,KAA+B,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YACvD,MAAM,IAAI,KAAK,CAAC,4BAA4B,YAAY,EAAE,CAAC,CAAC;QAC9D,CAAC;QACD,MAAM,KAAK,CAAC;IACd,CAAC;IAED,OAAO,EAAE,CAAC,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;AAC5C,CAAC;AAED,SAAgB,eAAe,CAAC,QAAgB,EAAE,IAAkB;IAClE,MAAM,gBAAgB,GAAG,oBAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACtD,OAAO,gBAAgB,CAAC,IAAI,CAAC,CAAC;AAChC,CAAC;AAEM,KAAK,UAAU,oBAAoB,CACxC,YAAoB,EACpB,UAAkB,EAClB,IAAkB,EAClB,MAAM,GAAG,KAAK;IAEd,MAAM,QAAQ,GAAG,MAAM,YAAY,CAAC,YAAY,CAAC,CAAC;IAClD,MAAM,OAAO,GAAG,eAAe,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;IAEhD,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;IACvC,CAAC;IAED,OAAO,UAAU,CAAC;AACpB,CAAC;AAED,mBAAmB;AACnB,IAAI,WAAW,GAAa,EAAE,CAAC;AAE/B,SAAgB,gBAAgB;IAC9B,WAAW,GAAG,EAAE,CAAC;AACnB,CAAC;AAED,SAAgB,cAAc;IAC5B,OAAO,CAAC,GAAG,WAAW,CAAC,CAAC;AAC1B,CAAC;AAED,SAAgB,aAAa,CAAC,QAAgB;IAC5C,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;AAC7B,CAAC;AAED,SAAgB,aAAa,CAAC,QAAgB,EAAE,UAAkB;IAChE,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,EAAE,SAAS,EAAE,IAAA,0BAAW,EAAC,UAAU,CAAC,CAAC,CAAC;AACxE,CAAC"}