@futdevpro/nts-dynamo 1.15.2 → 1.15.5

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 (192) hide show
  1. package/.cursor/rules/__assistant_guide.mdc +30 -0
  2. package/.cursor/rules/__main.mdc +64 -0
  3. package/.cursor/rules/_ag_backend-structure.mdc +86 -0
  4. package/.cursor/rules/_ag_backend.mdc +16 -0
  5. package/.cursor/rules/_ag_debug.mdc +8 -0
  6. package/.cursor/rules/_ag_documentation_writing_rules.mdc +372 -0
  7. package/.cursor/rules/_ag_file-refactoring.mdc +113 -0
  8. package/.cursor/rules/_ag_fixes_rules.mdc +6 -0
  9. package/.cursor/rules/_ag_frontend-structure.mdc +87 -0
  10. package/.cursor/rules/_ag_frontend.mdc +40 -0
  11. package/.cursor/rules/_ag_import-rules.mdc +45 -0
  12. package/.cursor/rules/_ag_naming.mdc +116 -0
  13. package/.cursor/rules/_ag_running_commands.mdc +5 -0
  14. package/.cursor/rules/_ag_server-controller.mdc +6 -0
  15. package/.cursor/rules/_ag_should-be.mdc +7 -0
  16. package/.cursor/rules/_ag_swearing.mdc +47 -0
  17. package/.cursor/rules/ai_development_guide.md +61 -0
  18. package/.cursor/rules/ai_directives.md +114 -0
  19. package/.cursor/rules/cursor-rules.md +160 -0
  20. package/.cursor/rules/default-command.mdc +465 -0
  21. package/.cursor/rules/error_code_pattern.md +40 -0
  22. package/.cursor/rules/saved rule mcp server use.md +16 -0
  23. package/_specifications/BACKLOG.md +15 -0
  24. package/_specifications/TODO.md +15 -0
  25. package/build/_modules/ai/_models/ai-test-generation-result.interface.d.ts +17 -0
  26. package/build/_modules/ai/_models/ai-test-generation-result.interface.d.ts.map +1 -0
  27. package/build/_modules/ai/_models/ai-test-generation-result.interface.js +3 -0
  28. package/build/_modules/ai/_models/ai-test-generation-result.interface.js.map +1 -0
  29. package/build/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.d.ts +36 -0
  30. package/build/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.d.ts.map +1 -0
  31. package/build/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.js +118 -0
  32. package/build/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.js.map +1 -0
  33. package/build/_modules/ai/_modules/anthropic/index.d.ts +3 -0
  34. package/build/_modules/ai/_modules/anthropic/index.d.ts.map +1 -0
  35. package/build/_modules/ai/_modules/anthropic/index.js +8 -0
  36. package/build/_modules/ai/_modules/anthropic/index.js.map +1 -0
  37. package/build/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.d.ts +35 -0
  38. package/build/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.d.ts.map +1 -0
  39. package/build/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.js +129 -0
  40. package/build/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.js.map +1 -0
  41. package/build/_modules/ai/_modules/fdp-ai/index.d.ts +3 -0
  42. package/build/_modules/ai/_modules/fdp-ai/index.d.ts.map +1 -0
  43. package/build/_modules/ai/_modules/fdp-ai/index.js +8 -0
  44. package/build/_modules/ai/_modules/fdp-ai/index.js.map +1 -0
  45. package/build/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.d.ts +40 -0
  46. package/build/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.d.ts.map +1 -0
  47. package/build/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.js +111 -0
  48. package/build/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.js.map +1 -0
  49. package/build/_modules/ai/_modules/open-ai/index.d.ts +1 -0
  50. package/build/_modules/ai/_modules/open-ai/index.d.ts.map +1 -1
  51. package/build/_modules/ai/_modules/open-ai/index.js +1 -0
  52. package/build/_modules/ai/_modules/open-ai/index.js.map +1 -1
  53. package/build/_modules/ai/_services/ai-user-key.service-base.d.ts +45 -0
  54. package/build/_modules/ai/_services/ai-user-key.service-base.d.ts.map +1 -0
  55. package/build/_modules/ai/_services/ai-user-key.service-base.js +15 -0
  56. package/build/_modules/ai/_services/ai-user-key.service-base.js.map +1 -0
  57. package/build/_modules/ai/index.d.ts +2 -0
  58. package/build/_modules/ai/index.d.ts.map +1 -1
  59. package/build/_modules/ai/index.js +2 -0
  60. package/build/_modules/ai/index.js.map +1 -1
  61. package/build/_modules/custom-data/custom-data.controller.d.ts.map +1 -1
  62. package/build/_modules/custom-data/custom-data.controller.js +1 -2
  63. package/build/_modules/custom-data/custom-data.controller.js.map +1 -1
  64. package/build/_modules/socket/app-extended.server.js +1 -1
  65. package/build/_modules/socket/app-extended.server.js.map +1 -1
  66. package/build/_services/base/data.service.d.ts +1 -1
  67. package/build/_services/base/data.service.js +1 -1
  68. package/build/_services/base/db.service.d.ts +1 -1
  69. package/build/_services/base/db.service.js +1 -1
  70. package/build/_services/core/api.service.d.ts.map +1 -1
  71. package/build/_services/core/api.service.js +1 -0
  72. package/build/_services/core/api.service.js.map +1 -1
  73. package/build/_services/core/auth.service.d.ts +2 -2
  74. package/build/_services/core/auth.service.js +1 -1
  75. package/build/_services/core/email.service.js +1 -1
  76. package/build/_services/core/email.service.js.map +1 -1
  77. package/build/_services/core/global.service.d.ts +1 -1
  78. package/build/_services/core/global.service.js +2 -2
  79. package/build/_services/core/global.service.js.map +1 -1
  80. package/build/_services/server/app.server.d.ts.map +1 -1
  81. package/build/_services/server/app.server.js +11 -1
  82. package/build/_services/server/app.server.js.map +1 -1
  83. package/package.json +18 -4
  84. package/scripts/run-coverage-tests.js +5 -1
  85. package/spec/support/helpers/spec-reporter-loader.js +359 -0
  86. package/spec/support/helpers/ts-node-helper.js +84 -0
  87. package/spec/support/jasmine.coverage.json +2 -1
  88. package/spec/support/jasmine.json +3 -3
  89. package/src/_collections/archive.util.spec.ts +36 -0
  90. package/src/_collections/get-environment-settings.util.spec.ts +210 -0
  91. package/src/_collections/star.controller.spec.ts +224 -0
  92. package/src/_models/control-models/api-call-params.control-model.spec.ts +62 -3
  93. package/src/_models/control-models/app-ext-system-controls.control-model.spec.ts +52 -0
  94. package/src/_models/control-models/app-params.control-model.spec.ts +158 -2
  95. package/src/_models/control-models/endpoint-params.control-model.spec.ts +578 -0
  96. package/src/_modules/ai/_models/ai-test-generation-result.interface.ts +16 -0
  97. package/src/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.ts +138 -0
  98. package/src/_modules/ai/_modules/anthropic/index.ts +5 -0
  99. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.spec.ts +242 -0
  100. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.spec.ts +209 -0
  101. package/src/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.ts +189 -0
  102. package/src/_modules/ai/_modules/fdp-ai/index.ts +5 -0
  103. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-document.data-service.spec.ts +342 -0
  104. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.spec.ts +550 -0
  105. package/src/_modules/ai/_modules/open-ai/_services/oai-embedding.control-service.spec.ts +240 -0
  106. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.spec.ts +462 -0
  107. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.spec.ts +437 -0
  108. package/src/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.ts +157 -0
  109. package/src/_modules/ai/_modules/open-ai/index.ts +1 -0
  110. package/src/_modules/ai/_services/ai-embedding.service-base.spec.ts +98 -0
  111. package/src/_modules/ai/_services/ai-llm-chat.service-base.spec.ts +229 -0
  112. package/src/_modules/ai/_services/ai-llm.service-base.spec.ts +250 -0
  113. package/src/_modules/ai/_services/ai-provider.service-base.spec.ts +79 -0
  114. package/src/_modules/ai/_services/ai-user-key.service-base.ts +59 -0
  115. package/src/_modules/ai/index.ts +2 -0
  116. package/src/_modules/assistant/_collections/ass.util.spec.ts +176 -0
  117. package/src/_modules/assistant/_services/ass-io.control-service.spec.ts +140 -0
  118. package/src/_modules/assistant/_services/ass-main.control-service.spec.ts +192 -0
  119. package/src/_modules/bot/_modules/discord-bot/_services/dib-messaging-provider.control-service.spec.ts +431 -0
  120. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.spec.ts +160 -0
  121. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.spec.ts +374 -0
  122. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.spec.ts +344 -0
  123. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.spec.ts +345 -0
  124. package/src/_modules/bot/_services/bot-commands.control-service.spec.ts +116 -0
  125. package/src/_modules/bot/_services/bot-io.control-service.spec.ts +285 -0
  126. package/src/_modules/bot/_services/bot-main.control-service.spec.ts +208 -0
  127. package/src/_modules/bot/_services/bot-messaging-provider.service-base.spec.ts +349 -0
  128. package/src/_modules/bot/_services/bot-routines.control-service.spec.ts +111 -0
  129. package/src/_modules/custom-data/custom-data.controller.spec.ts +49 -0
  130. package/src/_modules/custom-data/custom-data.controller.ts +1 -3
  131. package/src/_modules/custom-data/custom-data.data-service.spec.ts +54 -0
  132. package/src/_modules/custom-data/get-custom-data-routing-module.util.spec.ts +28 -0
  133. package/src/_modules/defaults/_services/default-auth.service.spec.ts +269 -0
  134. package/src/_modules/defaults/_services/default-socket-events.service.spec.ts +42 -0
  135. package/src/_modules/defaults/_services/default-user.data-service.spec.ts +187 -0
  136. package/src/_modules/discord-assistant/_collections/dias.util.spec.ts +366 -0
  137. package/src/_modules/discord-assistant/_services/dias-io.control-service.spec.ts +108 -0
  138. package/src/_modules/discord-assistant/_services/dias-main.control-service.spec.ts +22 -0
  139. package/src/_modules/discord-assistant/_services/dias.service-base.spec.ts +195 -0
  140. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.spec.ts +34 -0
  141. package/src/_modules/discord-bot/_collections/dibo-operations.util.spec.ts +214 -0
  142. package/src/_modules/discord-bot/_services/dibo-commands.control-service.spec.ts +154 -0
  143. package/src/_modules/discord-bot/_services/dibo-io.control-service.spec.ts +264 -0
  144. package/src/_modules/discord-bot/_services/dibo-main.control-service.spec.ts +408 -0
  145. package/src/_modules/discord-bot/_services/dibo-routines.control-service.spec.ts +105 -0
  146. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.spec.ts +418 -0
  147. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.spec.ts +345 -0
  148. package/src/_modules/messaging/_collections/msg.util.spec.ts +226 -0
  149. package/src/_modules/messaging/_services/msg-events.service.spec.ts +219 -0
  150. package/src/_modules/messaging/_services/msg-main.control-service.spec.ts +147 -0
  151. package/src/_modules/messaging/_services/msg.controller.spec.ts +201 -0
  152. package/src/_modules/mock/data-model.mock.spec.ts +27 -24
  153. package/src/_modules/oauth2/_routes/oauth2.controller.spec.ts +107 -0
  154. package/src/_modules/oauth2/_services/oauth2.auth-service.spec.ts +254 -0
  155. package/src/_modules/oauth2/_services/oauth2.control-service.spec.ts +585 -0
  156. package/src/_modules/server/errors/errors.control-service.spec.ts +230 -0
  157. package/src/_modules/server/errors/errors.controller.spec.ts +165 -0
  158. package/src/_modules/server/errors/errors.data-service.spec.ts +355 -0
  159. package/src/_modules/server/server-status/server-status-snapshot.control-service.spec.ts +70 -0
  160. package/src/_modules/server/server-status/server-status-snapshot.data-service.spec.ts +77 -0
  161. package/src/_modules/server/server-status/server-status.control-service.spec.ts +516 -0
  162. package/src/_modules/server/server-status/server-status.controller.spec.ts +156 -0
  163. package/src/_modules/socket/_models/socket-client-service-params.control-model.spec.ts +6 -3
  164. package/src/_modules/socket/_models/socket-presence.control-model.spec.ts +164 -0
  165. package/src/_modules/socket/_services/socket-client.service.spec.ts +15 -0
  166. package/src/_modules/socket/app-extended.server.ts +1 -1
  167. package/src/_modules/test/get-test-routing-module.util.spec.ts +28 -0
  168. package/src/_modules/test/test.controller.spec.ts +72 -0
  169. package/src/_modules/usage/usage.controller.spec.ts +81 -0
  170. package/src/_modules/usage/usage.data-service.spec.ts +332 -0
  171. package/src/_services/base/api.service-base.spec.ts +125 -0
  172. package/src/_services/base/archive-data.service.spec.ts +196 -0
  173. package/src/_services/base/data.service.spec.ts +493 -0
  174. package/src/_services/base/data.service.ts +1 -1
  175. package/src/_services/base/db.service.spec.ts +59 -18
  176. package/src/_services/base/db.service.ts +1 -1
  177. package/src/_services/base/singleton.service-base.spec.ts +28 -0
  178. package/src/_services/base/singleton.service.spec.ts +114 -0
  179. package/src/_services/core/api.service.ts +1 -0
  180. package/src/_services/core/auth.service.spec.ts +159 -0
  181. package/src/_services/core/auth.service.ts +2 -2
  182. package/src/_services/core/email.service.spec.ts +14 -22
  183. package/src/_services/core/email.service.ts +1 -1
  184. package/src/_services/core/global.service.spec.ts +275 -0
  185. package/src/_services/core/global.service.ts +2 -2
  186. package/src/_services/core/service-collection.service.spec.ts +46 -0
  187. package/src/_services/route/controller.service.ts +1 -1
  188. package/src/_services/route/routing-module.service.spec.ts +8 -6
  189. package/src/_services/server/app.server.ts +17 -1
  190. package/src/_services/shared.static-service.spec.ts +89 -0
  191. package/src/_modules/socket/app-extended.server.spec.ts +0 -227
  192. package/src/_services/server/app.server.spec.ts +0 -138
@@ -0,0 +1,332 @@
1
+
2
+ import { DyNTS_Usage_DataService } from './usage.data-service';
3
+ import { DyFM_RelativeDate } from '@futdevpro/fsm-dynamo';
4
+ import { DyFM_UsageSession, DyFM_UsageData } from '@futdevpro/fsm-dynamo/usage';
5
+ import { DyNTS_GlobalService } from '../../_services/core/global.service';
6
+ import { DyNTS_DBService } from '../../_services/base/db.service';
7
+ import { DyNTS_Shared } from '../../_services/shared.static-service';
8
+ import { Request } from 'express';
9
+
10
+ describe('| DyNTS_Usage_DataService', () => {
11
+ let service: DyNTS_Usage_DataService;
12
+ let mockDBService: jasmine.SpyObj<DyNTS_DBService<DyFM_UsageSession>>;
13
+ let mockRequest: Partial<Request>;
14
+
15
+ beforeEach(() => {
16
+ mockDBService = jasmine.createSpyObj('DyNTS_DBService', [
17
+ 'getAll',
18
+ 'getDataById',
19
+ 'find',
20
+ 'createData',
21
+ 'modifyData',
22
+ ]);
23
+
24
+ spyOn(DyNTS_GlobalService, 'getDBService').and.returnValue(mockDBService);
25
+
26
+ mockRequest = {
27
+ ip: '127.0.0.1',
28
+ headers: {
29
+ 'x-forwarded-for': '192.168.1.1',
30
+ },
31
+ };
32
+
33
+ service = new DyNTS_Usage_DataService({
34
+ usageSession: new DyFM_UsageSession({
35
+ _id: 'session-id',
36
+ sessionStart: new Date(),
37
+ }),
38
+ usageData: [],
39
+ issuer: 'test-issuer',
40
+ });
41
+ });
42
+
43
+ it('| should create service instance', () => {
44
+ expect(service).toBeDefined();
45
+ expect(service.data).toBeDefined();
46
+ expect(service.usageData).toEqual([]);
47
+ expect(service.simplifiedDailyUsage).toEqual([]);
48
+ });
49
+
50
+ it('| should initialize with usage data', () => {
51
+ const usageData: DyFM_UsageData[] = [
52
+ {
53
+ page: '/test',
54
+ opened: new Date(),
55
+ timeSpentOnPage: 1000,
56
+ },
57
+ ];
58
+
59
+ const serviceWithData = new DyNTS_Usage_DataService({
60
+ usageData,
61
+ issuer: 'test-issuer',
62
+ });
63
+
64
+ expect(serviceWithData.usageData).toEqual(usageData);
65
+ });
66
+
67
+ describe('| getLocationDataFromRequest', () => {
68
+ it('| should get location data from request', () => {
69
+ spyOn(DyNTS_Shared, 'getIpFromRequest').and.returnValue('192.168.1.1');
70
+ spyOn(DyNTS_Shared, 'getLocationDataByRequest').and.returnValue({
71
+ range: [0, 0],
72
+ country: 'US',
73
+ region: 'NY',
74
+ eu: '0',
75
+ timezone: 'America/New_York',
76
+ city: 'New York',
77
+ ll: [40.7128, -74.0060],
78
+ metro: 0,
79
+ area: 0,
80
+ });
81
+
82
+ service.getLocationDataFromRequest(mockRequest as Request);
83
+
84
+ expect(service.data.address).toBe('192.168.1.1');
85
+ expect(service.data.locationData).toBeDefined();
86
+ expect(service.data.locationData?.country).toBe('US');
87
+ });
88
+
89
+ it('| should throw error when location data retrieval fails', () => {
90
+ spyOn(DyNTS_Shared, 'getIpFromRequest').and.throwError('IP retrieval failed');
91
+
92
+ expect(() => {
93
+ service.getLocationDataFromRequest(mockRequest as Request);
94
+ }).toThrow();
95
+ });
96
+ });
97
+
98
+ describe('| getUsage', () => {
99
+ it('| should get usage data for range', async () => {
100
+ const mockSessions = [
101
+ new DyFM_UsageSession({
102
+ _id: 'session-1',
103
+ date: '2024-01-01',
104
+ sessionStart: new Date('2024-01-01'),
105
+ totalSessionTime: 5000,
106
+ locationData: {
107
+ range: [0, 0],
108
+ country: 'HU',
109
+ region: 'BU',
110
+ eu: '1',
111
+ timezone: 'Europe/Budapest',
112
+ city: 'Budapest',
113
+ ll: [47.4979, 19.0402],
114
+ metro: 0,
115
+ area: 0,
116
+ },
117
+ }),
118
+ new DyFM_UsageSession({
119
+ _id: 'session-2',
120
+ date: '2024-01-01',
121
+ sessionStart: new Date('2024-01-01'),
122
+ totalSessionTime: 3000,
123
+ locationData: {
124
+ range: [0, 0],
125
+ country: 'US',
126
+ region: 'NY',
127
+ eu: '0',
128
+ timezone: 'America/New_York',
129
+ city: 'New York',
130
+ ll: [40.7128, -74.0060],
131
+ metro: 0,
132
+ area: 0,
133
+ },
134
+ }),
135
+ ];
136
+ mockDBService.find.and.returnValue(Promise.resolve(mockSessions));
137
+
138
+ await service.getUsage(DyFM_RelativeDate.lastWeek);
139
+
140
+ expect(service.dataList.length).toBe(2);
141
+ expect(service.simplifiedDailyUsage.length).toBe(1);
142
+ expect(service.simplifiedDailyUsage[0].date).toBe('2024-01-01');
143
+ expect(service.simplifiedDailyUsage[0].visitorsHun).toBe(1);
144
+ expect(service.simplifiedDailyUsage[0].visitorsElse).toBe(1);
145
+ expect(service.simplifiedDailyUsage[0].totalVisitTime).toBe(8000);
146
+ });
147
+
148
+ it('| should group sessions by date', async () => {
149
+ const mockSessions = [
150
+ new DyFM_UsageSession({
151
+ _id: 'session-1',
152
+ date: '2024-01-01',
153
+ sessionStart: new Date('2024-01-01'),
154
+ totalSessionTime: 5000,
155
+ }),
156
+ new DyFM_UsageSession({
157
+ _id: 'session-2',
158
+ date: '2024-01-02',
159
+ sessionStart: new Date('2024-01-02'),
160
+ totalSessionTime: 3000,
161
+ }),
162
+ ];
163
+ mockDBService.find.and.returnValue(Promise.resolve(mockSessions));
164
+
165
+ await service.getUsage(DyFM_RelativeDate.lastWeek);
166
+
167
+ expect(service.simplifiedDailyUsage.length).toBe(2);
168
+ });
169
+
170
+ it('| should calculate average visit time', async () => {
171
+ const mockSessions = [
172
+ new DyFM_UsageSession({
173
+ _id: 'session-1',
174
+ date: '2024-01-01',
175
+ sessionStart: new Date('2024-01-01'),
176
+ totalSessionTime: 5000,
177
+ }),
178
+ new DyFM_UsageSession({
179
+ _id: 'session-2',
180
+ date: '2024-01-01',
181
+ sessionStart: new Date('2024-01-01'),
182
+ totalSessionTime: 3000,
183
+ }),
184
+ ];
185
+ mockDBService.find.and.returnValue(Promise.resolve(mockSessions));
186
+
187
+ await service.getUsage(DyFM_RelativeDate.lastWeek);
188
+
189
+ // Average is (5000 + 3000) / 2 = 4000 ms, then Math.round(4000 * 100) / 100 = 4000
190
+ expect(service.simplifiedDailyUsage[0].averageVisitTime).toBe(4000);
191
+ });
192
+
193
+ it('| should sort daily usage by date descending', async () => {
194
+ const mockSessions = [
195
+ new DyFM_UsageSession({
196
+ _id: 'session-1',
197
+ date: '2024-01-01',
198
+ sessionStart: new Date('2024-01-01'),
199
+ totalSessionTime: 5000,
200
+ }),
201
+ new DyFM_UsageSession({
202
+ _id: 'session-2',
203
+ date: '2024-01-02',
204
+ sessionStart: new Date('2024-01-02'),
205
+ totalSessionTime: 3000,
206
+ }),
207
+ ];
208
+ mockDBService.find.and.returnValue(Promise.resolve(mockSessions));
209
+
210
+ await service.getUsage(DyFM_RelativeDate.lastWeek);
211
+
212
+ expect(service.simplifiedDailyUsage[0].date).toBe('2024-01-02');
213
+ expect(service.simplifiedDailyUsage[1].date).toBe('2024-01-01');
214
+ });
215
+ });
216
+
217
+ describe('| updateUsageData', () => {
218
+ it('| should update usage data', async () => {
219
+ const existingSession = new DyFM_UsageSession({
220
+ _id: 'session-id',
221
+ sessionStart: new Date(),
222
+ usageData: [],
223
+ });
224
+ mockDBService.getDataById.and.returnValue(Promise.resolve(existingSession));
225
+ spyOn(service, 'saveData').and.returnValue(Promise.resolve(existingSession));
226
+ spyOn(service, 'getLocationDataFromRequest');
227
+
228
+ service.usageData = [
229
+ {
230
+ page: '/test',
231
+ opened: new Date(),
232
+ timeSpentOnPage: 1000,
233
+ },
234
+ {
235
+ page: '/test2',
236
+ opened: new Date(),
237
+ timeSpentOnPage: 2000,
238
+ },
239
+ ];
240
+
241
+ await service.updateUsageData(mockRequest as Request);
242
+
243
+ expect(service.saveData).toHaveBeenCalled();
244
+ expect(existingSession.totalSessionTime).toBe(3000);
245
+ });
246
+
247
+ it('| should throw error when session not found', async () => {
248
+ mockDBService.getDataById.and.returnValue(Promise.resolve(null));
249
+
250
+ await expectAsync(
251
+ service.updateUsageData(mockRequest as Request)
252
+ ).toBeRejected();
253
+ });
254
+
255
+ it('| should get location data if not present', async () => {
256
+ const existingSession = new DyFM_UsageSession({
257
+ _id: 'session-id',
258
+ sessionStart: new Date(),
259
+ usageData: [],
260
+ });
261
+ mockDBService.getDataById.and.returnValue(Promise.resolve(existingSession));
262
+ spyOn(service, 'saveData').and.returnValue(Promise.resolve(existingSession));
263
+ const getLocationSpy = spyOn(service, 'getLocationDataFromRequest');
264
+
265
+ await service.updateUsageData(mockRequest as Request);
266
+
267
+ expect(getLocationSpy).toHaveBeenCalled();
268
+ });
269
+ });
270
+
271
+ describe('| closeSession', () => {
272
+ it('| should close session', async () => {
273
+ const sessionStart = new Date('2024-01-01T10:00:00');
274
+ const existingSession = new DyFM_UsageSession({
275
+ _id: 'session-id',
276
+ sessionStart,
277
+ usageData: [
278
+ {
279
+ page: '/test',
280
+ opened: sessionStart,
281
+ timeSpentOnPage: 1000,
282
+ },
283
+ ],
284
+ totalSessionTime: 1000,
285
+ });
286
+ mockDBService.getDataById.and.returnValue(Promise.resolve(existingSession));
287
+ spyOn(service, 'saveData').and.returnValue(Promise.resolve(existingSession));
288
+ spyOn(service, 'getLocationDataFromRequest');
289
+
290
+ await service.closeSession(mockRequest as Request);
291
+
292
+ expect(existingSession.sessionEnd).toBeDefined();
293
+ expect(service.saveData).toHaveBeenCalled();
294
+ });
295
+
296
+ it('| should calculate total session time', async () => {
297
+ const sessionStart = new Date('2024-01-01T10:00:00');
298
+ const sessionEnd = new Date('2024-01-01T10:00:10');
299
+ const existingSession = new DyFM_UsageSession({
300
+ _id: 'session-id',
301
+ sessionStart,
302
+ usageData: [
303
+ {
304
+ page: '/test',
305
+ opened: sessionStart,
306
+ timeSpentOnPage: 1000,
307
+ },
308
+ ],
309
+ totalSessionTime: 1000,
310
+ });
311
+ mockDBService.getDataById.and.returnValue(Promise.resolve(existingSession));
312
+ spyOn(service, 'saveData').and.returnValue(Promise.resolve(existingSession));
313
+ spyOn(service, 'getLocationDataFromRequest');
314
+ jasmine.clock().install();
315
+ jasmine.clock().mockDate(sessionEnd);
316
+
317
+ await service.closeSession(mockRequest as Request);
318
+
319
+ expect(existingSession.totalSessionTime).toBe(10000);
320
+ jasmine.clock().uninstall();
321
+ });
322
+
323
+ it('| should throw error when session not found', async () => {
324
+ mockDBService.getDataById.and.returnValue(Promise.resolve(null));
325
+
326
+ await expectAsync(
327
+ service.closeSession(mockRequest as Request)
328
+ ).toBeRejected();
329
+ });
330
+ });
331
+ });
332
+
@@ -0,0 +1,125 @@
1
+
2
+ import { DyNTS_ApiService_Base } from './api.service-base';
3
+ import { DyNTS_ApiService } from '../core/api.service';
4
+ import { DyFM_Error, DyFM_HttpCallType, DyFM_Log } from '@futdevpro/fsm-dynamo';
5
+ import { DyNTS_ApiCall_Params } from '../../_models/control-models/api-call-params.control-model';
6
+
7
+ class TestApiService extends DyNTS_ApiService_Base {
8
+ get baseUrl(): string {
9
+ return 'https://api.example.com';
10
+ }
11
+
12
+ get testEndpoint(): string {
13
+ return '/test';
14
+ }
15
+
16
+ get connectingSystemName(): string {
17
+ return 'TestSystem';
18
+ }
19
+
20
+ static getInstance(): TestApiService {
21
+ return TestApiService.getSingletonInstance();
22
+ }
23
+ }
24
+
25
+ describe('| DyNTS_ApiService_Base', () => {
26
+ let service: TestApiService;
27
+
28
+ beforeEach(() => {
29
+ service = TestApiService.getInstance();
30
+ });
31
+
32
+ it('| should be a singleton instance', () => {
33
+ const instance1 = TestApiService.getInstance();
34
+ const instance2 = TestApiService.getInstance();
35
+
36
+ expect(instance1).toBe(instance2);
37
+ expect(instance1).toBeInstanceOf(TestApiService);
38
+ });
39
+
40
+ describe('| runApiCallWithAvailabilityCheck', () => {
41
+ it('| should return result when api call succeeds', async () => {
42
+ const apiCall = jasmine.createSpy('apiCall').and.returnValue(Promise.resolve('success'));
43
+
44
+ const result = await service.runApiCallWithAvailabilityCheck(apiCall, 'issuer-123');
45
+
46
+ expect(result).toBe('success');
47
+ expect(apiCall).toHaveBeenCalled();
48
+ });
49
+
50
+ it('| should check server availability when api call fails', async () => {
51
+ const apiCall = jasmine.createSpy('apiCall').and.returnValue(
52
+ Promise.reject(new Error('API call failed'))
53
+ );
54
+ spyOn(service, 'checkServerAvailability').and.returnValue(Promise.resolve());
55
+
56
+ await expectAsync(
57
+ service.runApiCallWithAvailabilityCheck(apiCall, 'issuer-123')
58
+ ).toBeRejectedWithError('API call failed');
59
+
60
+ expect(service.checkServerAvailability).toHaveBeenCalledWith('issuer-123');
61
+ });
62
+ });
63
+
64
+ describe('| checkServerAvailability', () => {
65
+ it('| should succeed when testGet succeeds', async () => {
66
+ spyOn(service, 'testGet').and.returnValue(Promise.resolve());
67
+ const logSpy = spyOn(DyFM_Log, 'success');
68
+
69
+ await service.checkServerAvailability('issuer-123');
70
+
71
+ expect(service.testGet).toHaveBeenCalledWith('issuer-123');
72
+ });
73
+
74
+ it('| should throw error when testGet fails', async () => {
75
+ spyOn(service, 'testGet').and.returnValue(
76
+ Promise.reject(new Error('Test failed'))
77
+ );
78
+
79
+ try {
80
+ await service.checkServerAvailability('issuer-123');
81
+ fail('Should have thrown an error');
82
+ } catch (error) {
83
+ expect(error).toBeInstanceOf(DyFM_Error);
84
+ expect((error as DyFM_Error)._message).toContain('TestSystem server is not available');
85
+ expect((error as DyFM_Error)._message).toContain('baseUrl: https://api.example.com');
86
+ expect((error as DyFM_Error).__userMessage).toContain('TestSystem server is not available');
87
+ expect((error as DyFM_Error)._errorCode).toContain('DyNTS-ASB-CSA0');
88
+ }
89
+ });
90
+ });
91
+
92
+ describe('| testGet', () => {
93
+ it('| should call ApiService.startApiCall with correct parameters', async () => {
94
+ const startApiCallSpy = spyOn(DyNTS_ApiService, 'startApiCall').and.returnValue(
95
+ Promise.resolve()
96
+ );
97
+ const logSpy = spyOn(DyFM_Log, 'success');
98
+
99
+ await service.testGet('issuer-123');
100
+
101
+ expect(startApiCallSpy).toHaveBeenCalledWith(
102
+ jasmine.any(DyNTS_ApiCall_Params)
103
+ );
104
+ const callParams = startApiCallSpy.calls.mostRecent().args[0] as DyNTS_ApiCall_Params;
105
+ expect(callParams.type).toBe(DyFM_HttpCallType.get);
106
+ expect(callParams.baseUrl).toBe('https://api.example.com');
107
+ expect(callParams.endpoint).toBe('/test');
108
+ expect(logSpy).toHaveBeenCalled();
109
+ });
110
+
111
+ it('| should throw error when testEndpoint is not set', async () => {
112
+ const serviceWithoutEndpoint = new (class extends DyNTS_ApiService_Base {
113
+ get baseUrl(): string { return 'https://api.example.com'; }
114
+ get testEndpoint(): string { return ''; }
115
+ get connectingSystemName(): string { return 'TestSystem'; }
116
+ static getInstance() { return this.getSingletonInstance(); }
117
+ } as any)();
118
+
119
+ await expectAsync(
120
+ serviceWithoutEndpoint.testGet('issuer-123')
121
+ ).toBeRejected();
122
+ });
123
+ });
124
+ });
125
+
@@ -0,0 +1,196 @@
1
+
2
+ import { DyNTS_ArchiveDataService } from './archive-data.service';
3
+ import { DyFM_DataModel_Params, DyFM_Metadata, DyFM_Error } from '@futdevpro/fsm-dynamo';
4
+ import { DyNTS_GlobalService } from '../core/global.service';
5
+ import { DyNTS_DBService } from './db.service';
6
+ import { DyNTS_getArchivedDBName } from '../../_collections/archive.util';
7
+
8
+ class TestMetadata extends DyFM_Metadata {
9
+ name: string;
10
+ }
11
+
12
+ describe('| DyNTS_ArchiveDataService', () => {
13
+ let mockDataParams: DyFM_DataModel_Params<TestMetadata>;
14
+ let mockOriginalDBService: jasmine.SpyObj<DyNTS_DBService<TestMetadata>>;
15
+ let mockArchivedDBService: jasmine.SpyObj<DyNTS_DBService<TestMetadata>>;
16
+
17
+ beforeEach(() => {
18
+ mockDataParams = {
19
+ dataName: 'test_data',
20
+ addArchive: true,
21
+ constructed: () => true,
22
+ properties: {},
23
+ } as any;
24
+
25
+ mockOriginalDBService = jasmine.createSpyObj<DyNTS_DBService<TestMetadata>>('OriginalDBService', [
26
+ 'getDataById',
27
+ 'trueDeleteDataById',
28
+ ]);
29
+
30
+ mockArchivedDBService = jasmine.createSpyObj<DyNTS_DBService<TestMetadata>>('ArchivedDBService', [
31
+ 'createData',
32
+ 'getDataById',
33
+ ]);
34
+
35
+ spyOn(DyNTS_GlobalService, 'getDBService').and.returnValue(mockOriginalDBService);
36
+ });
37
+
38
+ it('| should throw error when addArchive is missing from dataParams', () => {
39
+ const dataParamsWithoutArchive = {
40
+ ...mockDataParams,
41
+ addArchive: false,
42
+ } as any;
43
+
44
+ const data = new TestMetadata();
45
+
46
+ expect(() => {
47
+ new DyNTS_ArchiveDataService(data, dataParamsWithoutArchive, 'issuer-123');
48
+ }).toThrowError('addArchive is missing from dataParams!');
49
+ });
50
+
51
+ it('| should create service with archived data name', () => {
52
+ const data = new TestMetadata();
53
+
54
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
55
+
56
+ expect(service).toBeInstanceOf(DyNTS_ArchiveDataService);
57
+ expect(service.originalDBService).toBe(mockOriginalDBService);
58
+ expect(service.dataParams.dataName).toBe('test_data_archived');
59
+ });
60
+
61
+ describe('| archiveDataById', () => {
62
+ it('| should throw error when id is missing', async () => {
63
+ const data = new TestMetadata();
64
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
65
+
66
+ await expectAsync(
67
+ service.archiveDataById('')
68
+ ).toBeRejected();
69
+ });
70
+
71
+ it('| should throw error when data not found', async () => {
72
+ const data = new TestMetadata();
73
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
74
+ mockOriginalDBService.getDataById.and.returnValue(Promise.resolve(null));
75
+
76
+ await expectAsync(
77
+ service.archiveDataById('non-existent-id')
78
+ ).toBeRejected();
79
+ });
80
+
81
+ it('| should archive data when found', async () => {
82
+ const data = new TestMetadata({ _id: 'test-id' });
83
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
84
+ mockOriginalDBService.getDataById.and.returnValue(Promise.resolve(data));
85
+ mockOriginalDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
86
+ spyOn(service, 'archiveData').and.returnValue(Promise.resolve(data));
87
+
88
+ const result = await service.archiveDataById('test-id');
89
+
90
+ expect(mockOriginalDBService.getDataById).toHaveBeenCalledWith('test-id');
91
+ expect(service.archiveData).toHaveBeenCalledWith(data);
92
+ expect(result).toBe(data);
93
+ });
94
+ });
95
+
96
+ describe('| getDataByOriginalId', () => {
97
+ it('| should find data by original id', async () => {
98
+ const data = new TestMetadata();
99
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
100
+ const archivedData = new TestMetadata({ _originalId: 'original-id' });
101
+ spyOn(service, 'findData').and.returnValue(Promise.resolve(archivedData));
102
+
103
+ const result = await service.getDataByOriginalId('original-id');
104
+
105
+ expect(service.findData).toHaveBeenCalled();
106
+ expect(result).toBe(archivedData);
107
+ });
108
+
109
+ it('| should throw error on find failure', async () => {
110
+ const data = new TestMetadata();
111
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
112
+ spyOn(service, 'findData').and.returnValue(Promise.reject(new Error('Find failed')));
113
+
114
+ await expectAsync(
115
+ service.getDataByOriginalId('original-id')
116
+ ).toBeRejected();
117
+ });
118
+ });
119
+
120
+ describe('| archiveData', () => {
121
+ it('| should throw error when originalId already exists', async () => {
122
+ const data = new TestMetadata({ _id: 'test-id', _originalId: 'existing-original-id' } as TestMetadata);
123
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
124
+
125
+ await expectAsync(
126
+ service.archiveData(data)
127
+ ).toBeRejected();
128
+ });
129
+
130
+ it('| should throw error when data is already archived', async () => {
131
+ const data = new TestMetadata({ _id: 'test-id', _archived: new Date() } as TestMetadata);
132
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
133
+
134
+ await expectAsync(
135
+ service.archiveData(data)
136
+ ).toBeRejected();
137
+ });
138
+
139
+ it('| should throw error when id is missing', async () => {
140
+ const data = new TestMetadata();
141
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
142
+
143
+ await expectAsync(
144
+ service.archiveData(data)
145
+ ).toBeRejected();
146
+ });
147
+
148
+ it('| should archive data successfully', async () => {
149
+ const data = new TestMetadata({ _id: 'test-id' });
150
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
151
+ mockOriginalDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
152
+ spyOn(service, 'saveData').and.returnValue(Promise.resolve(data));
153
+
154
+ const result = await service.archiveData(data);
155
+
156
+ expect(mockOriginalDBService.trueDeleteDataById).toHaveBeenCalledWith('test-id');
157
+ expect(data._originalId).toBe('test-id');
158
+ expect(data._archived).toBeInstanceOf(Date);
159
+ expect(data._id).toBeUndefined();
160
+ expect(service.saveData).toHaveBeenCalled();
161
+ expect(result).toBe(data);
162
+ });
163
+ });
164
+
165
+ describe('| saveData', () => {
166
+ it('| should throw error when originalId is missing', async () => {
167
+ const data = new TestMetadata();
168
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
169
+
170
+ await expectAsync(
171
+ service.saveData(data)
172
+ ).toBeRejected();
173
+ });
174
+
175
+ it('| should throw error when data is not archived', async () => {
176
+ const data = new TestMetadata({ _originalId: 'original-id' } as TestMetadata);
177
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
178
+
179
+ await expectAsync(
180
+ service.saveData(data)
181
+ ).toBeRejected();
182
+ });
183
+
184
+ it('| should save archived data successfully', async () => {
185
+ const data = new TestMetadata({ _originalId: 'original-id', _archived: new Date() } as TestMetadata);
186
+ const service = new DyNTS_ArchiveDataService(data, mockDataParams, 'issuer-123');
187
+ const parentSaveDataSpy = spyOn(Object.getPrototypeOf(DyNTS_ArchiveDataService.prototype), 'saveData').and.returnValue(Promise.resolve(data));
188
+
189
+ const result = await service.saveData(data);
190
+
191
+ expect(parentSaveDataSpy).toHaveBeenCalled();
192
+ expect(result).toBe(data);
193
+ });
194
+ });
195
+ });
196
+