fragment-ts 1.0.0 → 1.0.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 (297) hide show
  1. package/README.md +401 -0
  2. package/USAGE.md +1439 -0
  3. package/bin/fragment.js +2 -0
  4. package/dist/ai/ai.module.d.ts +27 -0
  5. package/dist/ai/ai.module.d.ts.map +1 -0
  6. package/dist/ai/ai.module.js +77 -0
  7. package/dist/ai/ai.module.js.map +1 -0
  8. package/dist/auth/auth.module.d.ts +18 -0
  9. package/dist/auth/auth.module.d.ts.map +1 -0
  10. package/dist/auth/auth.module.js +89 -0
  11. package/dist/auth/auth.module.js.map +1 -0
  12. package/dist/cli/commands/build.command.d.ts +6 -1
  13. package/dist/cli/commands/build.command.d.ts.map +1 -0
  14. package/dist/cli/commands/build.command.js +131 -10
  15. package/dist/cli/commands/build.command.js.map +1 -0
  16. package/dist/cli/commands/diagnostics.command.d.ts +16 -0
  17. package/dist/cli/commands/diagnostics.command.d.ts.map +1 -0
  18. package/dist/cli/commands/diagnostics.command.js +419 -0
  19. package/dist/cli/commands/diagnostics.command.js.map +1 -0
  20. package/dist/cli/commands/generate.command.d.ts +13 -6
  21. package/dist/cli/commands/generate.command.d.ts.map +1 -0
  22. package/dist/cli/commands/generate.command.js +201 -128
  23. package/dist/cli/commands/generate.command.js.map +1 -0
  24. package/dist/cli/commands/init.command.d.ts +11 -5
  25. package/dist/cli/commands/init.command.d.ts.map +1 -0
  26. package/dist/cli/commands/init.command.js +268 -305
  27. package/dist/cli/commands/init.command.js.map +1 -0
  28. package/dist/cli/commands/migrate.command.d.ts +12 -5
  29. package/dist/cli/commands/migrate.command.d.ts.map +1 -0
  30. package/dist/cli/commands/migrate.command.js +212 -83
  31. package/dist/cli/commands/migrate.command.js.map +1 -0
  32. package/dist/cli/commands/serve.command.d.ts +5 -4
  33. package/dist/cli/commands/serve.command.d.ts.map +1 -0
  34. package/dist/cli/commands/serve.command.js +179 -21
  35. package/dist/cli/commands/serve.command.js.map +1 -0
  36. package/dist/cli/index.d.ts +2 -0
  37. package/dist/cli/index.d.ts.map +1 -0
  38. package/dist/cli/index.js +22 -0
  39. package/dist/cli/index.js.map +1 -0
  40. package/dist/core/container/di-container.d.ts +13 -7
  41. package/dist/core/container/di-container.d.ts.map +1 -0
  42. package/dist/core/container/di-container.js +76 -25
  43. package/dist/core/container/di-container.js.map +1 -0
  44. package/dist/core/decorators/application.decorator.d.ts +8 -0
  45. package/dist/core/decorators/application.decorator.d.ts.map +1 -0
  46. package/dist/core/decorators/application.decorator.js +16 -0
  47. package/dist/core/decorators/application.decorator.js.map +1 -0
  48. package/dist/core/decorators/auto-configuration.decorator.d.ts +2 -0
  49. package/dist/core/decorators/auto-configuration.decorator.d.ts.map +1 -0
  50. package/dist/core/decorators/auto-configuration.decorator.js +19 -0
  51. package/dist/core/decorators/auto-configuration.decorator.js.map +1 -0
  52. package/dist/core/decorators/conditional.decorators.d.ts +4 -0
  53. package/dist/core/decorators/conditional.decorators.d.ts.map +1 -0
  54. package/dist/core/decorators/conditional.decorators.js +22 -0
  55. package/dist/core/decorators/conditional.decorators.js.map +1 -0
  56. package/dist/core/decorators/controller.decorator.d.ts +1 -3
  57. package/dist/core/decorators/controller.decorator.d.ts.map +1 -0
  58. package/dist/core/decorators/controller.decorator.js +13 -9
  59. package/dist/core/decorators/controller.decorator.js.map +1 -0
  60. package/dist/core/decorators/http.decorators.d.ts +12 -0
  61. package/dist/core/decorators/http.decorators.d.ts.map +1 -0
  62. package/dist/core/decorators/http.decorators.js +52 -0
  63. package/dist/core/decorators/http.decorators.js.map +1 -0
  64. package/dist/core/decorators/injectable.decorator.d.ts +3 -3
  65. package/dist/core/decorators/injectable.decorator.d.ts.map +1 -0
  66. package/dist/core/decorators/injectable.decorator.js +12 -8
  67. package/dist/core/decorators/injectable.decorator.js.map +1 -0
  68. package/dist/core/decorators/injection.decorators.d.ts +5 -0
  69. package/dist/core/decorators/injection.decorators.d.ts.map +1 -0
  70. package/dist/core/decorators/injection.decorators.js +29 -0
  71. package/dist/core/decorators/injection.decorators.js.map +1 -0
  72. package/dist/core/decorators/repository.decorator.d.ts +1 -0
  73. package/dist/core/decorators/repository.decorator.d.ts.map +1 -0
  74. package/dist/core/decorators/repository.decorator.js +13 -1
  75. package/dist/core/decorators/repository.decorator.js.map +1 -0
  76. package/dist/core/decorators/service.decorator.d.ts +1 -0
  77. package/dist/core/decorators/service.decorator.d.ts.map +1 -0
  78. package/dist/core/decorators/service.decorator.js +13 -1
  79. package/dist/core/decorators/service.decorator.js.map +1 -0
  80. package/dist/core/index.d.ts +13 -0
  81. package/dist/core/index.d.ts.map +1 -0
  82. package/dist/core/index.js +29 -0
  83. package/dist/core/index.js.map +1 -0
  84. package/dist/core/loader/file-loader.d.ts +5 -0
  85. package/dist/core/loader/file-loader.d.ts.map +1 -0
  86. package/dist/core/loader/file-loader.js +85 -0
  87. package/dist/core/loader/file-loader.js.map +1 -0
  88. package/dist/core/metadata/metadata-keys.d.ts +20 -0
  89. package/dist/core/metadata/metadata-keys.d.ts.map +1 -0
  90. package/dist/core/metadata/metadata-keys.js +23 -0
  91. package/dist/core/metadata/metadata-keys.js.map +1 -0
  92. package/dist/core/metadata/metadata-storage.d.ts +36 -0
  93. package/dist/core/metadata/metadata-storage.d.ts.map +1 -0
  94. package/dist/core/metadata/metadata-storage.js +53 -0
  95. package/dist/core/metadata/metadata-storage.js.map +1 -0
  96. package/dist/core/scanner/component-scanner.d.ts +16 -0
  97. package/dist/core/scanner/component-scanner.d.ts.map +1 -0
  98. package/dist/core/scanner/component-scanner.js +147 -0
  99. package/dist/core/scanner/component-scanner.js.map +1 -0
  100. package/dist/index.d.ts +22 -0
  101. package/dist/index.d.ts.map +1 -0
  102. package/dist/index.js +60 -0
  103. package/dist/index.js.map +1 -0
  104. package/dist/plugins/plugin-manager.d.ts +14 -0
  105. package/dist/plugins/plugin-manager.d.ts.map +1 -0
  106. package/dist/{cli/utils/file-generator.js → plugins/plugin-manager.js} +32 -23
  107. package/dist/plugins/plugin-manager.js.map +1 -0
  108. package/dist/shared/errors.d.ts +18 -0
  109. package/dist/shared/errors.d.ts.map +1 -0
  110. package/dist/shared/errors.js +41 -0
  111. package/dist/shared/errors.js.map +1 -0
  112. package/dist/testing/runner.d.ts +26 -0
  113. package/dist/testing/runner.d.ts.map +1 -0
  114. package/dist/testing/runner.js +143 -0
  115. package/dist/testing/runner.js.map +1 -0
  116. package/dist/typeorm/typeorm-module.d.ts +36 -0
  117. package/dist/typeorm/typeorm-module.d.ts.map +1 -0
  118. package/dist/typeorm/typeorm-module.js +150 -0
  119. package/dist/typeorm/typeorm-module.js.map +1 -0
  120. package/dist/web/application.d.ts +29 -0
  121. package/dist/web/application.d.ts.map +1 -0
  122. package/dist/web/application.js +301 -0
  123. package/dist/web/application.js.map +1 -0
  124. package/dist/web/interfaces.d.ts +14 -0
  125. package/dist/web/interfaces.d.ts.map +1 -0
  126. package/dist/{auth/dto/login.dto.js → web/interfaces.js} +1 -0
  127. package/dist/web/interfaces.js.map +1 -0
  128. package/examples/ai-assistant/fragment.json +18 -0
  129. package/examples/ai-assistant/package-lock.json +2921 -0
  130. package/examples/ai-assistant/package.json +20 -0
  131. package/examples/ai-assistant/src/controllers/app.controller.ts +9 -0
  132. package/examples/ai-assistant/src/controllers/chat.controller.ts +69 -0
  133. package/examples/ai-assistant/src/controllers/conversation.controller.ts +29 -0
  134. package/examples/ai-assistant/src/controllers/message.controller.ts +29 -0
  135. package/examples/ai-assistant/src/controllers/search.controller.ts +16 -0
  136. package/examples/ai-assistant/src/dto/create-conversation.dto.ts +6 -0
  137. package/examples/ai-assistant/src/dto/create-message.dto.ts +6 -0
  138. package/examples/ai-assistant/src/entities/conversation.entity.ts +23 -0
  139. package/examples/ai-assistant/src/entities/message.entity.ts +27 -0
  140. package/examples/ai-assistant/src/entities/user.entity.ts +16 -0
  141. package/examples/ai-assistant/src/main.ts +15 -0
  142. package/examples/ai-assistant/src/migrations/1767738396740-CreateConversationTables.ts +37 -0
  143. package/examples/ai-assistant/src/repositories/conversation.repository.ts +32 -0
  144. package/examples/ai-assistant/src/repositories/message.repository.ts +30 -0
  145. package/examples/ai-assistant/src/services/ai.service.ts +41 -0
  146. package/examples/ai-assistant/src/services/app.service.ts +8 -0
  147. package/examples/ai-assistant/src/services/conversation.service.ts +75 -0
  148. package/examples/ai-assistant/src/services/message.service.ts +27 -0
  149. package/examples/ai-assistant/tsconfig.json +23 -0
  150. package/examples/blog-api/fragment.json +18 -0
  151. package/examples/blog-api/package.json +19 -0
  152. package/examples/blog-api/src/controllers/app.controller.ts +9 -0
  153. package/examples/blog-api/src/controllers/auth.controller.ts +17 -0
  154. package/examples/blog-api/src/controllers/category.controller.ts +29 -0
  155. package/examples/blog-api/src/controllers/comment.controller.ts +27 -0
  156. package/examples/blog-api/src/controllers/post.controller.ts +46 -0
  157. package/examples/blog-api/src/dto/create-category.dto.ts +6 -0
  158. package/examples/blog-api/src/dto/create-comment.dto.ts +6 -0
  159. package/examples/blog-api/src/dto/create-post.dto.ts +6 -0
  160. package/examples/blog-api/src/entities/category.entity.ts +16 -0
  161. package/examples/blog-api/src/entities/comment.entity.ts +29 -0
  162. package/examples/blog-api/src/entities/post.entity.ts +42 -0
  163. package/examples/blog-api/src/entities/user.entity.ts +25 -0
  164. package/examples/blog-api/src/main.ts +16 -0
  165. package/examples/blog-api/src/migrations/1767737463842-InitialSchema.ts +60 -0
  166. package/examples/blog-api/src/repositories/category.repository.ts +25 -0
  167. package/examples/blog-api/src/repositories/comment.repository.ts +25 -0
  168. package/examples/blog-api/src/repositories/post.repository.ts +29 -0
  169. package/examples/blog-api/src/seeds/SampleData.seed.ts +41 -0
  170. package/examples/blog-api/src/services/app.service.ts +8 -0
  171. package/examples/blog-api/src/services/auth.service.ts +15 -0
  172. package/examples/blog-api/src/services/category.service.ts +27 -0
  173. package/examples/blog-api/src/services/comment.service.ts +31 -0
  174. package/examples/blog-api/src/services/post.service.ts +35 -0
  175. package/examples/blog-api/tsconfig.json +23 -0
  176. package/package.json +56 -33
  177. package/src/ai/ai.module.ts +110 -0
  178. package/src/auth/auth.module.ts +77 -0
  179. package/src/cli/commands/build.command.ts +123 -13
  180. package/src/cli/commands/diagnostics.command.ts +438 -0
  181. package/src/cli/commands/generate.command.ts +206 -137
  182. package/src/cli/commands/init.command.ts +337 -349
  183. package/src/cli/commands/migrate.command.ts +203 -88
  184. package/src/cli/commands/serve.command.ts +176 -24
  185. package/src/cli/index.ts +23 -0
  186. package/src/core/container/di-container.ts +83 -26
  187. package/src/core/decorators/application.decorator.ts +26 -0
  188. package/src/core/decorators/auto-configuration.decorator.ts +17 -0
  189. package/src/core/decorators/conditional.decorators.ts +19 -0
  190. package/src/core/decorators/controller.decorator.ts +14 -11
  191. package/src/core/decorators/http.decorators.ts +71 -0
  192. package/src/core/decorators/injectable.decorator.ts +14 -9
  193. package/src/core/decorators/injection.decorators.ts +26 -0
  194. package/src/core/decorators/repository.decorator.ts +13 -2
  195. package/src/core/decorators/service.decorator.ts +13 -2
  196. package/src/core/index.ts +13 -0
  197. package/src/core/loader/file-loader.ts +55 -0
  198. package/src/core/metadata/metadata-keys.ts +19 -0
  199. package/src/core/metadata/metadata-storage.ts +91 -0
  200. package/src/core/scanner/component-scanner.ts +129 -0
  201. package/src/index.ts +45 -0
  202. package/src/plugins/plugin-manager.ts +52 -0
  203. package/src/shared/errors.ts +34 -0
  204. package/src/testing/runner.ts +143 -0
  205. package/src/typeorm/typeorm-module.ts +216 -0
  206. package/src/web/application.ts +348 -0
  207. package/src/web/interfaces.ts +17 -0
  208. package/tsconfig.json +8 -5
  209. package/.env.example +0 -0
  210. package/base.ts +0 -1810
  211. package/base2.ts +0 -968
  212. package/bin/frg.ts +0 -5
  213. package/config/fragment.lock.yaml +0 -0
  214. package/config/fragment.yaml +0 -0
  215. package/dist/app.d.ts +0 -15
  216. package/dist/app.js +0 -90
  217. package/dist/auth/auth.controller.d.ts +0 -10
  218. package/dist/auth/auth.controller.js +0 -87
  219. package/dist/auth/auth.middleware.d.ts +0 -2
  220. package/dist/auth/auth.middleware.js +0 -24
  221. package/dist/auth/auth.service.d.ts +0 -20
  222. package/dist/auth/auth.service.js +0 -143
  223. package/dist/auth/dto/login.dto.d.ts +0 -9
  224. package/dist/cli/cli.d.ts +0 -12
  225. package/dist/cli/cli.js +0 -186
  226. package/dist/cli/commands/config.command.d.ts +0 -6
  227. package/dist/cli/commands/config.command.js +0 -284
  228. package/dist/cli/templates/controller.template.d.ts +0 -1
  229. package/dist/cli/templates/controller.template.js +0 -52
  230. package/dist/cli/templates/entity.template.d.ts +0 -1
  231. package/dist/cli/templates/entity.template.js +0 -23
  232. package/dist/cli/templates/repository.template.d.ts +0 -1
  233. package/dist/cli/templates/repository.template.js +0 -43
  234. package/dist/cli/templates/service.template.d.ts +0 -1
  235. package/dist/cli/templates/service.template.js +0 -43
  236. package/dist/cli/utils/file-generator.d.ts +0 -9
  237. package/dist/cli/utils/logger.d.ts +0 -14
  238. package/dist/cli/utils/logger.js +0 -49
  239. package/dist/controllers/health.controller.d.ts +0 -13
  240. package/dist/controllers/health.controller.js +0 -50
  241. package/dist/core/config/config-loader.d.ts +0 -31
  242. package/dist/core/config/config-loader.js +0 -98
  243. package/dist/core/decorators/auth-guard.decorator.d.ts +0 -3
  244. package/dist/core/decorators/auth-guard.decorator.js +0 -18
  245. package/dist/core/decorators/autowire.decorator.d.ts +0 -3
  246. package/dist/core/decorators/autowire.decorator.js +0 -17
  247. package/dist/core/decorators/middleware.decorator.d.ts +0 -3
  248. package/dist/core/decorators/middleware.decorator.js +0 -20
  249. package/dist/core/decorators/route.decorator.d.ts +0 -14
  250. package/dist/core/decorators/route.decorator.js +0 -32
  251. package/dist/core/openai/openai-client.d.ts +0 -12
  252. package/dist/core/openai/openai-client.js +0 -93
  253. package/dist/database/data-source.d.ts +0 -4
  254. package/dist/database/data-source.js +0 -26
  255. package/dist/entities/session.entity.d.ts +0 -9
  256. package/dist/entities/session.entity.js +0 -45
  257. package/dist/entities/user.entity.d.ts +0 -10
  258. package/dist/entities/user.entity.js +0 -48
  259. package/dist/middlewares/logging.middleware.d.ts +0 -2
  260. package/dist/middlewares/logging.middleware.js +0 -28
  261. package/dist/repositories/session.repository.d.ts +0 -9
  262. package/dist/repositories/session.repository.js +0 -50
  263. package/dist/repositories/user.repository.d.ts +0 -10
  264. package/dist/repositories/user.repository.js +0 -43
  265. package/dist/server.d.ts +0 -1
  266. package/dist/server.js +0 -30
  267. package/dist/services/health.service.d.ts +0 -13
  268. package/dist/services/health.service.js +0 -44
  269. package/readme.md +0 -120
  270. package/src/app.ts +0 -121
  271. package/src/auth/auth.controller.ts +0 -52
  272. package/src/auth/auth.middleware.ts +0 -27
  273. package/src/auth/auth.service.ts +0 -110
  274. package/src/auth/dto/login.dto.ts +0 -11
  275. package/src/cli/cli.ts +0 -212
  276. package/src/cli/commands/config.command.ts +0 -280
  277. package/src/cli/templates/controller.template.ts +0 -51
  278. package/src/cli/templates/entity.template.ts +0 -22
  279. package/src/cli/templates/repository.template.ts +0 -42
  280. package/src/cli/templates/service.template.ts +0 -42
  281. package/src/cli/utils/file-generator.ts +0 -37
  282. package/src/cli/utils/logger.ts +0 -52
  283. package/src/controllers/health.controller.ts +0 -24
  284. package/src/core/config/config-loader.ts +0 -98
  285. package/src/core/decorators/auth-guard.decorator.ts +0 -15
  286. package/src/core/decorators/autowire.decorator.ts +0 -18
  287. package/src/core/decorators/middleware.decorator.ts +0 -18
  288. package/src/core/decorators/route.decorator.ts +0 -33
  289. package/src/core/openai/openai-client.ts +0 -99
  290. package/src/database/data-source.ts +0 -29
  291. package/src/entities/session.entity.ts +0 -25
  292. package/src/entities/user.entity.ts +0 -27
  293. package/src/middlewares/logging.middleware.ts +0 -28
  294. package/src/repositories/session.repository.ts +0 -42
  295. package/src/repositories/user.repository.ts +0 -37
  296. package/src/server.ts +0 -32
  297. package/src/services/health.service.ts +0 -29
@@ -1,43 +1,100 @@
1
-
2
- import 'reflect-metadata';
3
- import { AUTOWIRE_METADATA } from '../decorators/autowire.decorator';
1
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
2
+ import { Scope } from '../decorators/injectable.decorator';
4
3
 
5
4
  export class DIContainer {
6
- private static instances = new Map<any, any>();
7
- private static classes = new Set<any>();
5
+ private static instance: DIContainer;
6
+ private singletons: Map<any, any> = new Map();
7
+ private transients: Map<any, any> = new Map();
8
+ private factories: Map<any, () => any> = new Map();
8
9
 
9
- static register(target: any): void {
10
- this.classes.add(target);
10
+ static getInstance(): DIContainer {
11
+ if (!DIContainer.instance) {
12
+ DIContainer.instance = new DIContainer();
13
+ }
14
+ return DIContainer.instance;
11
15
  }
12
16
 
13
- static resolve<T>(target: any): T {
14
- // Return existing singleton if available
15
- if (this.instances.has(target)) {
16
- return this.instances.get(target);
17
+ register(token: any, instance?: any, factory?: () => any): void {
18
+ if (instance) {
19
+ this.singletons.set(token, instance);
20
+ } else if (factory) {
21
+ this.factories.set(token, factory);
17
22
  }
23
+ }
18
24
 
19
- // Get constructor parameter metadata
20
- const params = Reflect.getOwnMetadata(AUTOWIRE_METADATA, target) || [];
21
- const sortedParams = params.sort((a: any, b: any) => a.index - b.index);
25
+ resolve<T>(token: any): T {
26
+ if (this.singletons.has(token)) {
27
+ return this.singletons.get(token);
28
+ }
22
29
 
23
- // Resolve dependencies recursively
24
- const dependencies = sortedParams.map((param: any) => {
25
- return this.resolve(param.type);
26
- });
30
+ if (this.factories.has(token)) {
31
+ const factory = this.factories.get(token)!;
32
+ const instance = factory();
33
+ const scope = Reflect.getMetadata(METADATA_KEYS.SCOPE, token) || 'singleton';
34
+
35
+ if (scope === 'singleton') {
36
+ this.singletons.set(token, instance);
37
+ }
38
+
39
+ return instance;
40
+ }
27
41
 
28
- // Create instance
29
- const instance = new target(...dependencies);
30
- this.instances.set(target, instance);
42
+ if (typeof token === 'function') {
43
+ const instance = this.createInstance(token);
44
+ const scope = Reflect.getMetadata(METADATA_KEYS.SCOPE, token) || 'singleton';
45
+
46
+ if (scope === 'singleton') {
47
+ this.singletons.set(token, instance);
48
+ }
49
+
50
+ return instance;
51
+ }
31
52
 
53
+ throw new Error(`Cannot resolve dependency: ${token}`);
54
+ }
55
+
56
+ private createInstance(target: any): any {
57
+ const paramTypes = Reflect.getMetadata('design:paramtypes', target) || [];
58
+ const params = paramTypes.map((type: any) => this.resolve(type));
59
+ const instance = new target(...params);
60
+
61
+ this.injectProperties(instance);
62
+
32
63
  return instance;
33
64
  }
34
65
 
35
- static getRegisteredClasses(): any[] {
36
- return Array.from(this.classes);
66
+ private injectProperties(instance: any): void {
67
+ const properties = Object.getOwnPropertyNames(instance.constructor.prototype);
68
+
69
+ properties.forEach(prop => {
70
+ const autowired = Reflect.getMetadata(METADATA_KEYS.AUTOWIRED, instance, prop);
71
+ const inject = Reflect.getMetadata(METADATA_KEYS.INJECT, instance, prop);
72
+ const value = Reflect.getMetadata(METADATA_KEYS.VALUE, instance, prop);
73
+
74
+ if (autowired) {
75
+ instance[prop] = this.resolve(autowired);
76
+ } else if (inject) {
77
+ instance[prop] = this.resolve(inject);
78
+ } else if (value) {
79
+ instance[prop] = this.resolveValue(value);
80
+ }
81
+ });
82
+ }
83
+
84
+ private resolveValue(expression: string): any {
85
+ const match = expression.match(/\$\{(.+?)\}/);
86
+ if (match) {
87
+ const key = match[1];
88
+ return process.env[key] || '';
89
+ }
90
+ return expression;
91
+ }
92
+
93
+ has(token: any): boolean {
94
+ return this.singletons.has(token) || this.factories.has(token);
37
95
  }
38
96
 
39
- static clear(): void {
40
- this.instances.clear();
41
- this.classes.clear();
97
+ getAllInstances(): any[] {
98
+ return Array.from(this.singletons.values());
42
99
  }
43
100
  }
@@ -0,0 +1,26 @@
1
+ import { METADATA_KEYS } from "../metadata/metadata-keys";
2
+
3
+ export interface ApplicationOptions {
4
+ port?: number;
5
+ host?: string;
6
+ configPath?: string;
7
+ autoScan?: boolean;
8
+ }
9
+
10
+ export function FragmentApplication(
11
+ options: ApplicationOptions = {},
12
+ ): ClassDecorator {
13
+ return (target: any) => {
14
+ Reflect.defineMetadata(
15
+ METADATA_KEYS.APPLICATION,
16
+ {
17
+ ...options,
18
+ port: options.port || 3000,
19
+ host: options.host || "0.0.0.0",
20
+ configPath: options.configPath || "fragment.json",
21
+ autoScan: options.autoScan || true,
22
+ },
23
+ target,
24
+ );
25
+ };
26
+ }
@@ -0,0 +1,17 @@
1
+ import { Injectable } from './injectable.decorator';
2
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
3
+ import { MetadataStorage } from '../metadata/metadata-storage';
4
+
5
+ export function AutoConfiguration(): ClassDecorator {
6
+ return (target: any) => {
7
+ Injectable('singleton')(target);
8
+ Reflect.defineMetadata(METADATA_KEYS.AUTO_CONFIGURATION, true, target);
9
+
10
+ const storage = MetadataStorage.getInstance();
11
+ storage.addClass({
12
+ target,
13
+ type: 'auto-configuration',
14
+ scope: 'singleton'
15
+ });
16
+ };
17
+ }
@@ -0,0 +1,19 @@
1
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
2
+
3
+ export function ConditionalOnClass(classRef: any): ClassDecorator {
4
+ return (target: any) => {
5
+ Reflect.defineMetadata(METADATA_KEYS.CONDITIONAL_ON_CLASS, classRef, target);
6
+ };
7
+ }
8
+
9
+ export function ConditionalOnMissingBean(token: any): ClassDecorator {
10
+ return (target: any) => {
11
+ Reflect.defineMetadata(METADATA_KEYS.CONDITIONAL_ON_MISSING_BEAN, token, target);
12
+ };
13
+ }
14
+
15
+ export function ConditionalOnProperty(key: string, expectedValue?: any): ClassDecorator {
16
+ return (target: any) => {
17
+ Reflect.defineMetadata(METADATA_KEYS.CONDITIONAL_ON_PROPERTY, { key, expectedValue }, target);
18
+ };
19
+ }
@@ -1,15 +1,18 @@
1
-
2
- import 'reflect-metadata';
3
- import { DIContainer } from '../container/di-container';
4
-
5
- export const CONTROLLER_METADATA = 'controller:metadata';
6
- export const CONTROLLER_PATH = 'controller:path';
1
+ import { Injectable } from './injectable.decorator';
2
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
3
+ import { MetadataStorage } from '../metadata/metadata-storage';
7
4
 
8
5
  export function Controller(path: string = ''): ClassDecorator {
9
6
  return (target: any) => {
10
- Reflect.defineMetadata(CONTROLLER_METADATA, true, target);
11
- Reflect.defineMetadata(CONTROLLER_PATH, path, target);
12
- DIContainer.register(target);
13
- return target;
7
+ Injectable('singleton')(target);
8
+ Reflect.defineMetadata(METADATA_KEYS.CONTROLLER, path, target);
9
+
10
+ const storage = MetadataStorage.getInstance();
11
+ storage.addClass({
12
+ target,
13
+ type: 'controller',
14
+ scope: 'singleton',
15
+ path
16
+ });
14
17
  };
15
- }
18
+ }
@@ -0,0 +1,71 @@
1
+ import { METADATA_KEYS } from "../metadata/metadata-keys";
2
+ import { MetadataStorage } from "../metadata/metadata-storage";
3
+
4
+ function createHttpMethodDecorator(method: string) {
5
+ return (path: string = ""): MethodDecorator => {
6
+ return (
7
+ target: any,
8
+ propertyKey: string | symbol,
9
+ descriptor: PropertyDescriptor,
10
+ ) => {
11
+ Reflect.defineMetadata(
12
+ METADATA_KEYS.HTTP_METHOD,
13
+ method,
14
+ target,
15
+ propertyKey,
16
+ );
17
+ Reflect.defineMetadata(
18
+ METADATA_KEYS.ROUTE_PATH,
19
+ path,
20
+ target,
21
+ propertyKey,
22
+ );
23
+
24
+ const storage = MetadataStorage.getInstance();
25
+ storage.addMethod({
26
+ target: target.constructor,
27
+ propertyKey: propertyKey as string,
28
+ method,
29
+ path,
30
+ paramMetadata: storage.getParams(target, propertyKey as string),
31
+ });
32
+ };
33
+ };
34
+ }
35
+
36
+ export const Get = createHttpMethodDecorator("GET");
37
+ export const Post = createHttpMethodDecorator("POST");
38
+ export const Put = createHttpMethodDecorator("PUT");
39
+ export const Delete = createHttpMethodDecorator("DELETE");
40
+ export const Patch = createHttpMethodDecorator("PATCH");
41
+
42
+ function createParamDecorator(type: string) {
43
+ return (paramName?: string): ParameterDecorator => {
44
+ return (
45
+ target: Object,
46
+ propertyKey: string | symbol | undefined,
47
+ parameterIndex: number,
48
+ ): void => {
49
+ if (!propertyKey) {
50
+ throw new Error(`@${type}() cannot be used on constructor parameters`);
51
+ }
52
+
53
+ const storage = MetadataStorage.getInstance();
54
+
55
+ storage.addParam({
56
+ target,
57
+ propertyKey: propertyKey.toString(),
58
+ index: parameterIndex,
59
+ type: type as any,
60
+ paramName,
61
+ });
62
+ };
63
+ };
64
+ }
65
+
66
+ export const Body = createParamDecorator("body");
67
+ export const Param = createParamDecorator("param");
68
+ export const Query = createParamDecorator("query");
69
+ export const Header = createParamDecorator("header");
70
+ export const Req = (): ParameterDecorator => createParamDecorator("req")();
71
+ export const Res = (): ParameterDecorator => createParamDecorator("res")();
@@ -1,13 +1,18 @@
1
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
2
+ import { MetadataStorage } from '../metadata/metadata-storage';
1
3
 
2
- import 'reflect-metadata';
3
- import { DIContainer } from '../container/di-container';
4
+ export type Scope = 'singleton' | 'request' | 'transient';
4
5
 
5
- export const INJECTABLE_METADATA = 'injectable:metadata';
6
-
7
- export function Injectable(): ClassDecorator {
6
+ export function Injectable(scope: Scope = 'singleton'): ClassDecorator {
8
7
  return (target: any) => {
9
- Reflect.defineMetadata(INJECTABLE_METADATA, true, target);
10
- DIContainer.register(target);
11
- return target;
8
+ Reflect.defineMetadata(METADATA_KEYS.INJECTABLE, true, target);
9
+ Reflect.defineMetadata(METADATA_KEYS.SCOPE, scope, target);
10
+
11
+ const storage = MetadataStorage.getInstance();
12
+ storage.addClass({
13
+ target,
14
+ type: 'injectable',
15
+ scope
16
+ });
12
17
  };
13
- }
18
+ }
@@ -0,0 +1,26 @@
1
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
2
+
3
+ export function Autowired(): PropertyDecorator {
4
+ return (target: any, propertyKey: string | symbol) => {
5
+ const type = Reflect.getMetadata('design:type', target, propertyKey);
6
+ Reflect.defineMetadata(METADATA_KEYS.AUTOWIRED, type, target, propertyKey);
7
+ };
8
+ }
9
+
10
+ export function Inject(token: string): PropertyDecorator {
11
+ return (target: any, propertyKey: string | symbol) => {
12
+ Reflect.defineMetadata(METADATA_KEYS.INJECT, token, target, propertyKey);
13
+ };
14
+ }
15
+
16
+ export function Qualifier(name: string): PropertyDecorator {
17
+ return (target: any, propertyKey: string | symbol) => {
18
+ Reflect.defineMetadata(METADATA_KEYS.QUALIFIER, name, target, propertyKey);
19
+ };
20
+ }
21
+
22
+ export function Value(expression: string): PropertyDecorator {
23
+ return (target: any, propertyKey: string | symbol) => {
24
+ Reflect.defineMetadata(METADATA_KEYS.VALUE, expression, target, propertyKey);
25
+ };
26
+ }
@@ -1,6 +1,17 @@
1
-
2
1
  import { Injectable } from './injectable.decorator';
2
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
3
+ import { MetadataStorage } from '../metadata/metadata-storage';
3
4
 
4
5
  export function Repository(): ClassDecorator {
5
- return Injectable();
6
+ return (target: any) => {
7
+ Injectable('singleton')(target);
8
+ Reflect.defineMetadata(METADATA_KEYS.REPOSITORY, true, target);
9
+
10
+ const storage = MetadataStorage.getInstance();
11
+ storage.addClass({
12
+ target,
13
+ type: 'repository',
14
+ scope: 'singleton'
15
+ });
16
+ };
6
17
  }
@@ -1,6 +1,17 @@
1
-
2
1
  import { Injectable } from './injectable.decorator';
2
+ import { METADATA_KEYS } from '../metadata/metadata-keys';
3
+ import { MetadataStorage } from '../metadata/metadata-storage';
3
4
 
4
5
  export function Service(): ClassDecorator {
5
- return Injectable();
6
+ return (target: any) => {
7
+ Injectable('singleton')(target);
8
+ Reflect.defineMetadata(METADATA_KEYS.SERVICE, true, target);
9
+
10
+ const storage = MetadataStorage.getInstance();
11
+ storage.addClass({
12
+ target,
13
+ type: 'service',
14
+ scope: 'singleton'
15
+ });
16
+ };
6
17
  }
@@ -0,0 +1,13 @@
1
+ import 'reflect-metadata';
2
+
3
+ export * from './decorators/application.decorator';
4
+ export * from './decorators/injectable.decorator';
5
+ export * from './decorators/service.decorator';
6
+ export * from './decorators/controller.decorator';
7
+ export * from './decorators/repository.decorator';
8
+ export * from './decorators/auto-configuration.decorator';
9
+ export * from './decorators/conditional.decorators';
10
+ export * from './decorators/injection.decorators';
11
+ export * from './decorators/http.decorators';
12
+ export * from './container/di-container';
13
+ export * from './metadata/metadata-storage';
@@ -0,0 +1,55 @@
1
+ import * as fs from "fs";
2
+ import * as path from "path";
3
+ import { glob } from "glob";
4
+
5
+ export class FileLoader {
6
+ static async loadFiles(
7
+ pattern: string,
8
+ cwd: string = process.cwd(),
9
+ ): Promise<void> {
10
+ try {
11
+ const files = await glob(pattern, {
12
+ cwd,
13
+ ignore: [
14
+ "**/*.spec.ts",
15
+ "**/*.test.ts",
16
+ "**/*.d.ts",
17
+ "**/node_modules/**",
18
+ ],
19
+ });
20
+
21
+ for (const file of files) {
22
+ const absolutePath = path.join(cwd, file);
23
+
24
+ // Check if it's a TypeScript file
25
+ if (file.endsWith(".ts")) {
26
+ try {
27
+ // Use require to execute the file (decorators run on import)
28
+ delete require.cache[absolutePath];
29
+ require(absolutePath);
30
+ } catch (error) {
31
+ console.warn(`Failed to load ${file}:`, (error as any)?.message);
32
+ }
33
+ }
34
+ }
35
+ } catch (error) {
36
+ console.error("Error loading files:", error);
37
+ }
38
+ }
39
+
40
+ static async loadProjectFiles(): Promise<void> {
41
+ const isProduction = process.env.NODE_ENV === "production";
42
+ const srcDir = isProduction ? "dist" : "src";
43
+ const pattern = isProduction ? "**/*.js" : "**/*.ts";
44
+
45
+ const projectRoot = process.cwd();
46
+ const sourceDir = path.join(projectRoot, srcDir);
47
+
48
+ if (!fs.existsSync(sourceDir)) {
49
+ console.warn(`Source directory not found: ${sourceDir}`);
50
+ return;
51
+ }
52
+
53
+ await this.loadFiles(pattern, sourceDir);
54
+ }
55
+ }
@@ -0,0 +1,19 @@
1
+ export const METADATA_KEYS = {
2
+ INJECTABLE: 'fragment:injectable',
3
+ SCOPE: 'fragment:scope',
4
+ SERVICE: 'fragment:service',
5
+ CONTROLLER: 'fragment:controller',
6
+ REPOSITORY: 'fragment:repository',
7
+ AUTO_CONFIGURATION: 'fragment:auto-configuration',
8
+ CONDITIONAL_ON_CLASS: 'fragment:conditional-class',
9
+ CONDITIONAL_ON_MISSING_BEAN: 'fragment:conditional-missing-bean',
10
+ CONDITIONAL_ON_PROPERTY: 'fragment:conditional-property',
11
+ AUTOWIRED: 'fragment:autowired',
12
+ INJECT: 'fragment:inject',
13
+ QUALIFIER: 'fragment:qualifier',
14
+ VALUE: 'fragment:value',
15
+ HTTP_METHOD: 'fragment:http-method',
16
+ ROUTE_PATH: 'fragment:route-path',
17
+ PARAM_METADATA: 'fragment:param-metadata',
18
+ APPLICATION: 'fragment:application'
19
+ };
@@ -0,0 +1,91 @@
1
+ import { METADATA_KEYS } from "./metadata-keys";
2
+
3
+ export interface ClassMetadata {
4
+ target: any;
5
+ type:
6
+ | "injectable"
7
+ | "service"
8
+ | "controller"
9
+ | "repository"
10
+ | "auto-configuration";
11
+ scope?: "singleton" | "request" | "transient";
12
+ path?: string;
13
+ }
14
+
15
+ export interface MethodMetadata {
16
+ target: any;
17
+ propertyKey: string;
18
+ method: string;
19
+ path: string;
20
+ paramMetadata: ParamMetadata[];
21
+ }
22
+
23
+ export interface ParamMetadata {
24
+ target: any;
25
+ propertyKey: string;
26
+ index: number;
27
+ type: "body" | "param" | "query" | "header" | "req" | "res";
28
+ paramName?: string;
29
+ }
30
+
31
+ export class MetadataStorage {
32
+ private static instance: MetadataStorage;
33
+ private classes: Map<any, ClassMetadata> = new Map();
34
+ private methods: Map<string, MethodMetadata> = new Map();
35
+ private params: Map<string, ParamMetadata[]> = new Map();
36
+
37
+ static getInstance(): MetadataStorage {
38
+ if (!MetadataStorage.instance) {
39
+ MetadataStorage.instance = new MetadataStorage();
40
+ }
41
+ return MetadataStorage.instance;
42
+ }
43
+
44
+ addClass(metadata: ClassMetadata): void {
45
+ this.classes.set(metadata.target, metadata);
46
+ }
47
+
48
+ getClass(target: any): ClassMetadata | undefined {
49
+ return this.classes.get(target);
50
+ }
51
+
52
+ getAllClasses(): ClassMetadata[] {
53
+ return Array.from(this.classes.values());
54
+ }
55
+
56
+ addMethod(metadata: MethodMetadata): void {
57
+ const key = `${metadata.target.name}.${metadata.propertyKey}`;
58
+
59
+ const existingParams = this.getParams(
60
+ metadata.target,
61
+ metadata.propertyKey,
62
+ );
63
+ metadata.paramMetadata = existingParams.sort((a, b) => a.index - b.index);
64
+
65
+ this.methods.set(key, metadata);
66
+ }
67
+
68
+ getMethod(target: any, propertyKey: string): MethodMetadata | undefined {
69
+ const key = `${target.name}.${propertyKey}`;
70
+ return this.methods.get(key);
71
+ }
72
+
73
+ getAllMethods(): MethodMetadata[] {
74
+ return Array.from(this.methods.values());
75
+ }
76
+
77
+ addParam(metadata: ParamMetadata): void {
78
+ const key = `${metadata.target.constructor.name}.${metadata.propertyKey}`;
79
+ const existing = this.params.get(key) || [];
80
+ existing.push(metadata);
81
+ this.params.set(key, existing);
82
+ }
83
+
84
+ getParams(target: any, propertyKey: string): ParamMetadata[] {
85
+ const targetName = target.name || target.constructor?.name;
86
+ if (!targetName) return [];
87
+
88
+ const key = `${targetName}.${propertyKey}`;
89
+ return this.params.get(key) || [];
90
+ }
91
+ }