@gadmin2n/schematics 0.0.87 → 0.0.89

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 (99) hide show
  1. package/dist/lib/application/files/gadmin2-game-angle-demo/.dockerignore +16 -2
  2. package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.codegen +40 -0
  3. package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.server +76 -0
  4. package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.web +53 -0
  5. package/dist/lib/application/files/gadmin2-game-angle-demo/Jenkinsfile +219 -33
  6. package/dist/lib/application/files/gadmin2-game-angle-demo/compose-ctl.sh +250 -0
  7. package/dist/lib/application/files/gadmin2-game-angle-demo/config/prisma/workflow.prisma +4 -1
  8. package/dist/lib/application/files/gadmin2-game-angle-demo/dev/postgres/init.sql +12 -0
  9. package/dist/lib/application/files/gadmin2-game-angle-demo/docker-compose.md +170 -0
  10. package/dist/lib/application/files/gadmin2-game-angle-demo/docker-compose.yml +254 -0
  11. package/dist/lib/application/files/gadmin2-game-angle-demo/server/package.json +8 -7
  12. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/lib/page-helpers.ts +1 -1
  13. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/prismaModels.ts +1 -1
  14. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/agenda.seed.ts +39 -0
  15. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/audit.seed.ts +40 -0
  16. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/bootstrap.ts +56 -0
  17. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/canvas.seed.ts +39 -0
  18. package/dist/lib/application/files/gadmin2-game-angle-demo/server/{scripts/sync-data-mngt-pages.ts → seed/data-mngt.seed.ts} +36 -20
  19. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/game.seed.ts +44 -0
  20. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/index.ts +30 -6
  21. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/permission.seed.ts +130 -0
  22. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow-event-trigger.ts +60 -0
  23. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow-node-types.ts +11 -25
  24. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow.seed.ts +108 -0
  25. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/main.ts +1 -0
  26. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/agendaJob/agendaJob.controller.spec.ts +31 -2
  27. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/audit/audit.controller.spec.ts +31 -2
  28. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/audit/audit.service.spec.ts +41 -57
  29. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/game/game.controller.spec.ts +31 -2
  30. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/game/game.service.spec.ts +309 -1
  31. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/page/page.controller.spec.ts +31 -2
  32. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/page/page.service.spec.ts +315 -1
  33. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/pageResource/pageResource.controller.spec.ts +31 -2
  34. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/pageResource/pageResource.service.spec.ts +312 -2
  35. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/resource/resource.controller.spec.ts +31 -2
  36. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/resource/resource.service.spec.ts +317 -1
  37. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/role/role.controller.spec.ts +31 -2
  38. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/role/role.service.spec.ts +309 -1
  39. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/rolePages/rolePages.controller.spec.ts +31 -2
  40. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/rolePages/rolePages.service.spec.ts +299 -1
  41. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/roleResource/roleResource.controller.spec.ts +31 -2
  42. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/roleResource/roleResource.service.spec.ts +307 -1
  43. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/user/user.controller.spec.ts +31 -2
  44. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/user/user.service.spec.ts +309 -1
  45. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/dsl-validate.util.spec.ts +205 -0
  46. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/dsl-validate.util.ts +116 -0
  47. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/temporal.service.spec.ts +158 -0
  48. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/temporal.service.ts +110 -1
  49. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/webhook-signature.util.spec.ts +79 -0
  50. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/webhook-signature.util.ts +54 -0
  51. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.controller.ts +34 -0
  52. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.service.spec.ts +457 -0
  53. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.service.ts +241 -4
  54. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowEventOutbox/workflowEventOutbox.controller.spec.ts +34 -2
  55. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowEventOutbox/workflowEventOutbox.service.spec.ts +24 -30
  56. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeInstance/workflowNodeInstance.controller.spec.ts +34 -2
  57. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeInstance/workflowNodeInstance.service.spec.ts +36 -36
  58. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeType/workflowNodeType.controller.spec.ts +34 -2
  59. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeType/workflowNodeType.service.spec.ts +48 -24
  60. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/README.md +312 -3
  61. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/TODO.md +152 -0
  62. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/.dockerignore +12 -0
  63. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/Dockerfile +79 -0
  64. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/GRACEFUL-DEPLOYMENT.md +270 -0
  65. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/activities/index.ts +1 -1
  66. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/activities/reporting.ts +23 -0
  67. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/index.ts +70 -5
  68. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/outbox-poller.ts +246 -90
  69. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/tests/cron-trigger-workflow.test.ts +20 -0
  70. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/workflows/dsl-workflow.ts +96 -8
  71. package/dist/lib/application/files/gadmin2-game-angle-demo/web/nginx.conf +74 -0
  72. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/ElementInspector.tsx +18 -0
  73. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/promptGenerator.ts +1 -1
  74. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/form.tsx +1 -1
  75. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/en/common.json +3 -3
  76. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/zh_CN/common.json +3 -3
  77. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/plugins/devShellPlugin.ts +4 -1
  78. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasEditPage.tsx +9 -0
  79. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasListPage.tsx +156 -139
  80. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasPage.tsx +14 -2
  81. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasToolbar.tsx +62 -0
  82. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/PublishModal.tsx +4 -6
  83. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasApi.ts +18 -27
  84. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasDefaults.ts +32 -11
  85. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/demos.ts +48 -61
  86. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas-page/index.tsx +3 -6
  87. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/components/DslView.tsx +16 -16
  88. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/editor.tsx +28 -35
  89. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/instance-detail.tsx +34 -3
  90. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/show.tsx +1 -1
  91. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/types.ts +1 -1
  92. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/styles/antd.css +6 -0
  93. package/package.json +1 -1
  94. package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile +0 -63
  95. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/sync-resources.ts +0 -100
  96. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/permissions.ts +0 -302
  97. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.controller.spec.ts +0 -20
  98. package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/sql/create-event-trigger.sql +0 -87
  99. /package/dist/lib/application/files/gadmin2-game-angle-demo/{GRACEFUL-DEPLOYMENT.md → server/GRACEFUL-DEPLOYMENT.md} +0 -0
@@ -1,18 +1,326 @@
1
+ /// <reference types="jest" />
2
+ import { ConfigService } from '@nestjs/config';
1
3
  import { Test, TestingModule } from '@nestjs/testing';
4
+ import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
5
+
6
+ // Mock nestjs-prisma 模块
7
+ jest.mock('nestjs-prisma', () => ({
8
+ PrismaService: jest.fn(),
9
+ }));
10
+
11
+ import { PrismaService } from 'nestjs-prisma';
2
12
  import { RoleService } from './role.service';
3
13
 
14
+ // Mock PrismaService
15
+ const mockPrismaService = {
16
+ role: {
17
+ create: jest.fn(),
18
+ createMany: jest.fn(),
19
+ findMany: jest.fn(),
20
+ findUnique: jest.fn(),
21
+ update: jest.fn(),
22
+ updateMany: jest.fn(),
23
+ delete: jest.fn(),
24
+ deleteMany: jest.fn(),
25
+ count: jest.fn(),
26
+ groupBy: jest.fn(),
27
+ aggregate: jest.fn(),
28
+ },
29
+ };
30
+
31
+ // Mock Logger
32
+ const mockLogger = {
33
+ child: jest.fn().mockReturnThis(),
34
+ info: jest.fn(),
35
+ error: jest.fn(),
36
+ warn: jest.fn(),
37
+ debug: jest.fn(),
38
+ };
39
+
40
+ // Mock ConfigService
41
+ const mockConfigService = {
42
+ get: jest.fn(),
43
+ };
44
+
4
45
  describe('RoleService', () => {
5
46
  let service: RoleService;
47
+ let prisma: typeof mockPrismaService;
6
48
 
7
49
  beforeEach(async () => {
50
+ // 每个测试前重置所有 mock
51
+ jest.clearAllMocks();
52
+
8
53
  const module: TestingModule = await Test.createTestingModule({
9
- providers: [RoleService],
54
+ providers: [
55
+ RoleService,
56
+ { provide: PrismaService, useValue: mockPrismaService },
57
+ { provide: ConfigService, useValue: mockConfigService },
58
+ { provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
59
+ ],
10
60
  }).compile();
11
61
 
12
62
  service = module.get<RoleService>(RoleService);
63
+ prisma = mockPrismaService;
13
64
  });
14
65
 
15
66
  it('should be defined', () => {
16
67
  expect(service).toBeDefined();
17
68
  });
69
+
70
+ describe('createOne', () => {
71
+ it('should create a single role record', async () => {
72
+ const createArgs = {
73
+ data: {
74
+ name: 'test_name',
75
+ description: 'test_description',
76
+ creator: 'test_creator',
77
+ },
78
+ } as any;
79
+ const expectedResult = { id: 1, ...createArgs.data };
80
+
81
+ prisma.role.create.mockResolvedValue(expectedResult);
82
+
83
+ const result = await service.createOne(createArgs);
84
+
85
+ expect(prisma.role.create).toHaveBeenCalledWith(createArgs);
86
+ expect(prisma.role.create).toHaveBeenCalledTimes(1);
87
+ expect(result).toEqual(expectedResult);
88
+ });
89
+
90
+ it('should throw error when create fails', async () => {
91
+ const createArgs = {
92
+ data: {
93
+ name: 'test_name',
94
+ description: 'test_description',
95
+ creator: 'test_creator',
96
+ },
97
+ } as any;
98
+ const error = new Error('Database connection failed');
99
+
100
+ prisma.role.create.mockRejectedValue(error);
101
+
102
+ await expect(service.createOne(createArgs)).rejects.toThrow(
103
+ 'Database connection failed',
104
+ );
105
+ });
106
+ });
107
+
108
+ describe('createMany', () => {
109
+ it('should create multiple role records with skipDuplicates', async () => {
110
+ const data = [
111
+ {
112
+ name: 'test_name',
113
+ description: 'test_description',
114
+ creator: 'test_creator',
115
+ },
116
+ {
117
+ name: 'test_name',
118
+ description: 'test_description',
119
+ creator: 'test_creator',
120
+ },
121
+ ] as any;
122
+ const expectedResult = { count: 2 };
123
+
124
+ prisma.role.createMany.mockResolvedValue(expectedResult);
125
+
126
+ const result = await service.createMany(data);
127
+
128
+ expect(prisma.role.createMany).toHaveBeenCalledWith({
129
+ data,
130
+ skipDuplicates: true,
131
+ });
132
+ expect(result).toEqual(expectedResult);
133
+ });
134
+ });
135
+
136
+ describe('findMany', () => {
137
+ it('should return entities with itemCount', async () => {
138
+ const findArgs = {
139
+ where: {},
140
+ take: 10,
141
+ skip: 0,
142
+ } as any;
143
+ const mockEntities = [{ id: 1 }, { id: 2 }];
144
+
145
+ prisma.role.count.mockResolvedValue(2);
146
+ prisma.role.findMany.mockResolvedValue(mockEntities);
147
+
148
+ const result = await service.findMany(findArgs);
149
+
150
+ expect(prisma.role.count).toHaveBeenCalledWith({ where: findArgs.where });
151
+ expect(prisma.role.findMany).toHaveBeenCalledWith(findArgs);
152
+ expect(result).toEqual({
153
+ itemCount: 2,
154
+ entities: mockEntities,
155
+ });
156
+ });
157
+
158
+ it('should return empty result when no records found', async () => {
159
+ const findArgs = { where: {} } as any;
160
+
161
+ prisma.role.count.mockResolvedValue(0);
162
+ prisma.role.findMany.mockResolvedValue([]);
163
+
164
+ const result = await service.findMany(findArgs);
165
+
166
+ expect(result).toEqual({ itemCount: 0, entities: [] });
167
+ });
168
+ });
169
+
170
+ describe('findUnique', () => {
171
+ it('should find a single role record by id', async () => {
172
+ const id = 1;
173
+ const select = { id: true };
174
+ const expectedResult = { id: 1 };
175
+
176
+ prisma.role.findUnique.mockResolvedValue(expectedResult);
177
+
178
+ const result = await service.findUnique(id, select);
179
+
180
+ expect(prisma.role.findUnique).toHaveBeenCalledWith({
181
+ where: { id },
182
+ select,
183
+ });
184
+ expect(result).toEqual(expectedResult);
185
+ });
186
+
187
+ it('should return null when record not found', async () => {
188
+ prisma.role.findUnique.mockResolvedValue(null);
189
+
190
+ const result = await service.findUnique(1, { id: true });
191
+
192
+ expect(result).toBeNull();
193
+ });
194
+ });
195
+
196
+ describe('updateUnique', () => {
197
+ it('should update a single role record', async () => {
198
+ const id = 1;
199
+ const data = {
200
+ name: 'test_name',
201
+ description: 'test_description',
202
+ creator: 'test_creator',
203
+ } as any;
204
+ const expectedResult = { id: 1, ...data };
205
+
206
+ prisma.role.update.mockResolvedValue(expectedResult);
207
+
208
+ const result = await service.updateUnique(id, data);
209
+
210
+ expect(prisma.role.update).toHaveBeenCalledWith({
211
+ where: { id },
212
+ data,
213
+ });
214
+ expect(result).toEqual(expectedResult);
215
+ });
216
+ });
217
+
218
+ describe('updateMany', () => {
219
+ it('should update multiple role records', async () => {
220
+ const updateArgs = {
221
+ where: {},
222
+ data: {
223
+ name: 'test_name',
224
+ description: 'test_description',
225
+ creator: 'test_creator',
226
+ },
227
+ } as any;
228
+ const expectedResult = { count: 5 };
229
+
230
+ prisma.role.updateMany.mockResolvedValue(expectedResult);
231
+
232
+ const result = await service.updateMany(updateArgs);
233
+
234
+ expect(prisma.role.updateMany).toHaveBeenCalledWith(updateArgs);
235
+ expect(result).toEqual(expectedResult);
236
+ });
237
+ });
238
+
239
+ describe('deleteUnique', () => {
240
+ it('should delete a single role record and return count', async () => {
241
+ const id = 1;
242
+
243
+ prisma.role.delete.mockResolvedValue({ id: 1 });
244
+
245
+ const result = await service.deleteUnique(id);
246
+
247
+ expect(prisma.role.delete).toHaveBeenCalledWith({ where: { id } });
248
+ expect(result).toEqual({ count: 1 });
249
+ });
250
+ });
251
+
252
+ describe('deleteMany', () => {
253
+ it('should delete multiple role records', async () => {
254
+ const deleteArgs = { where: {} } as any;
255
+ const expectedResult = { count: 3 };
256
+
257
+ prisma.role.deleteMany.mockResolvedValue(expectedResult);
258
+
259
+ const result = await service.deleteMany(deleteArgs);
260
+
261
+ expect(prisma.role.deleteMany).toHaveBeenCalledWith(deleteArgs);
262
+ expect(result).toEqual(expectedResult);
263
+ });
264
+ });
265
+
266
+ describe('count', () => {
267
+ it('should count role records with filter', async () => {
268
+ const countArgs = { where: {} } as any;
269
+
270
+ prisma.role.count.mockResolvedValue(10);
271
+
272
+ const result = await service.count(countArgs);
273
+
274
+ expect(prisma.role.count).toHaveBeenCalledWith(countArgs);
275
+ expect(result).toBe(10);
276
+ });
277
+
278
+ it('should count all records when no filter provided', async () => {
279
+ prisma.role.count.mockResolvedValue(100);
280
+
281
+ const result = await service.count();
282
+
283
+ expect(prisma.role.count).toHaveBeenCalledWith(undefined);
284
+ expect(result).toBe(100);
285
+ });
286
+ });
287
+
288
+ describe('groupBy', () => {
289
+ it('should group role records by specified field', async () => {
290
+ const groupByArgs = {
291
+ by: ['id'],
292
+ _count: { id: true },
293
+ } as any;
294
+ const expectedResult = [{ id: 1, _count: { id: 50 } }];
295
+
296
+ prisma.role.groupBy.mockResolvedValue(expectedResult);
297
+
298
+ const result = await service.groupBy(groupByArgs);
299
+
300
+ expect(prisma.role.groupBy).toHaveBeenCalledWith(groupByArgs);
301
+ expect(result).toEqual(expectedResult);
302
+ });
303
+ });
304
+
305
+ describe('aggregate', () => {
306
+ it('should aggregate role records', async () => {
307
+ const aggregateArgs = {
308
+ _count: true,
309
+ _max: { id: true },
310
+ _min: { id: true },
311
+ } as any;
312
+ const expectedResult = {
313
+ _count: 100,
314
+ _max: { id: 1 },
315
+ _min: { id: 1 },
316
+ };
317
+
318
+ prisma.role.aggregate.mockResolvedValue(expectedResult);
319
+
320
+ const result = await service.aggregate(aggregateArgs);
321
+
322
+ expect(prisma.role.aggregate).toHaveBeenCalledWith(aggregateArgs);
323
+ expect(result).toEqual(expectedResult);
324
+ });
325
+ });
18
326
  });
@@ -1,15 +1,44 @@
1
+ /// <reference types="jest" />
2
+ import { ConfigService } from '@nestjs/config';
1
3
  import { Test, TestingModule } from '@nestjs/testing';
4
+ import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
2
5
  import { RolePagesController } from './rolePages.controller';
3
6
  import { RolePagesService } from './rolePages.service';
4
7
 
8
+ // Mock RolePagesService:用空对象屏蔽真实 Service 的依赖(PrismaService 等),
9
+ // 避免 NestJS 在测试模块编译时去解析 Service 的构造参数。
10
+ const mockRolePagesService = {};
11
+
12
+ // Mock ConfigService
13
+ const mockConfigService = {
14
+ get: jest.fn(),
15
+ };
16
+
17
+ // Mock Logger
18
+ const mockLogger = {
19
+ child: jest.fn().mockReturnThis(),
20
+ info: jest.fn(),
21
+ error: jest.fn(),
22
+ warn: jest.fn(),
23
+ debug: jest.fn(),
24
+ };
25
+
5
26
  describe('RolePagesController', () => {
6
27
  let controller: RolePagesController;
7
28
 
8
29
  beforeEach(async () => {
9
30
  const module: TestingModule = await Test.createTestingModule({
10
31
  controllers: [RolePagesController],
11
- providers: [RolePagesService],
12
- }).compile();
32
+ providers: [
33
+ { provide: RolePagesService, useValue: mockRolePagesService },
34
+ { provide: ConfigService, useValue: mockConfigService },
35
+ { provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
36
+ ],
37
+ })
38
+ // 兜底:自动 mock 任何未显式提供的依赖
39
+ // (例如 ACGuard 内部的 __roles_builder__ token)
40
+ .useMocker(() => ({}))
41
+ .compile();
13
42
 
14
43
  controller = module.get<RolePagesController>(RolePagesController);
15
44
  });
@@ -1,18 +1,316 @@
1
+ /// <reference types="jest" />
2
+ import { ConfigService } from '@nestjs/config';
1
3
  import { Test, TestingModule } from '@nestjs/testing';
4
+ import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
5
+
6
+ // Mock nestjs-prisma 模块
7
+ jest.mock('nestjs-prisma', () => ({
8
+ PrismaService: jest.fn(),
9
+ }));
10
+
11
+ import { PrismaService } from 'nestjs-prisma';
2
12
  import { RolePagesService } from './rolePages.service';
3
13
 
14
+ // Mock PrismaService
15
+ const mockPrismaService = {
16
+ rolePages: {
17
+ create: jest.fn(),
18
+ createMany: jest.fn(),
19
+ findMany: jest.fn(),
20
+ findUnique: jest.fn(),
21
+ update: jest.fn(),
22
+ updateMany: jest.fn(),
23
+ delete: jest.fn(),
24
+ deleteMany: jest.fn(),
25
+ count: jest.fn(),
26
+ groupBy: jest.fn(),
27
+ aggregate: jest.fn(),
28
+ },
29
+ };
30
+
31
+ // Mock Logger
32
+ const mockLogger = {
33
+ child: jest.fn().mockReturnThis(),
34
+ info: jest.fn(),
35
+ error: jest.fn(),
36
+ warn: jest.fn(),
37
+ debug: jest.fn(),
38
+ };
39
+
40
+ // Mock ConfigService
41
+ const mockConfigService = {
42
+ get: jest.fn(),
43
+ };
44
+
4
45
  describe('RolePagesService', () => {
5
46
  let service: RolePagesService;
47
+ let prisma: typeof mockPrismaService;
6
48
 
7
49
  beforeEach(async () => {
50
+ // 每个测试前重置所有 mock
51
+ jest.clearAllMocks();
52
+
8
53
  const module: TestingModule = await Test.createTestingModule({
9
- providers: [RolePagesService],
54
+ providers: [
55
+ RolePagesService,
56
+ { provide: PrismaService, useValue: mockPrismaService },
57
+ { provide: ConfigService, useValue: mockConfigService },
58
+ { provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
59
+ ],
10
60
  }).compile();
11
61
 
12
62
  service = module.get<RolePagesService>(RolePagesService);
63
+ prisma = mockPrismaService;
13
64
  });
14
65
 
15
66
  it('should be defined', () => {
16
67
  expect(service).toBeDefined();
17
68
  });
69
+
70
+ describe('createOne', () => {
71
+ it('should create a single rolePages record', async () => {
72
+ const createArgs = {
73
+ data: {
74
+ creator: 'test_creator',
75
+ },
76
+ } as any;
77
+ const expectedResult = { id: 1, ...createArgs.data };
78
+
79
+ prisma.rolePages.create.mockResolvedValue(expectedResult);
80
+
81
+ const result = await service.createOne(createArgs);
82
+
83
+ expect(prisma.rolePages.create).toHaveBeenCalledWith(createArgs);
84
+ expect(prisma.rolePages.create).toHaveBeenCalledTimes(1);
85
+ expect(result).toEqual(expectedResult);
86
+ });
87
+
88
+ it('should throw error when create fails', async () => {
89
+ const createArgs = {
90
+ data: {
91
+ creator: 'test_creator',
92
+ },
93
+ } as any;
94
+ const error = new Error('Database connection failed');
95
+
96
+ prisma.rolePages.create.mockRejectedValue(error);
97
+
98
+ await expect(service.createOne(createArgs)).rejects.toThrow(
99
+ 'Database connection failed',
100
+ );
101
+ });
102
+ });
103
+
104
+ describe('createMany', () => {
105
+ it('should create multiple rolePages records with skipDuplicates', async () => {
106
+ const data = [
107
+ {
108
+ creator: 'test_creator',
109
+ },
110
+ {
111
+ creator: 'test_creator',
112
+ },
113
+ ] as any;
114
+ const expectedResult = { count: 2 };
115
+
116
+ prisma.rolePages.createMany.mockResolvedValue(expectedResult);
117
+
118
+ const result = await service.createMany(data);
119
+
120
+ expect(prisma.rolePages.createMany).toHaveBeenCalledWith({
121
+ data,
122
+ skipDuplicates: true,
123
+ });
124
+ expect(result).toEqual(expectedResult);
125
+ });
126
+ });
127
+
128
+ describe('findMany', () => {
129
+ it('should return entities with itemCount', async () => {
130
+ const findArgs = {
131
+ where: {},
132
+ take: 10,
133
+ skip: 0,
134
+ } as any;
135
+ const mockEntities = [{ id: 1 }, { id: 2 }];
136
+
137
+ prisma.rolePages.count.mockResolvedValue(2);
138
+ prisma.rolePages.findMany.mockResolvedValue(mockEntities);
139
+
140
+ const result = await service.findMany(findArgs);
141
+
142
+ expect(prisma.rolePages.count).toHaveBeenCalledWith({
143
+ where: findArgs.where,
144
+ });
145
+ expect(prisma.rolePages.findMany).toHaveBeenCalledWith(findArgs);
146
+ expect(result).toEqual({
147
+ itemCount: 2,
148
+ entities: mockEntities,
149
+ });
150
+ });
151
+
152
+ it('should return empty result when no records found', async () => {
153
+ const findArgs = { where: {} } as any;
154
+
155
+ prisma.rolePages.count.mockResolvedValue(0);
156
+ prisma.rolePages.findMany.mockResolvedValue([]);
157
+
158
+ const result = await service.findMany(findArgs);
159
+
160
+ expect(result).toEqual({ itemCount: 0, entities: [] });
161
+ });
162
+ });
163
+
164
+ describe('findUnique', () => {
165
+ it('should find a single rolePages record by id', async () => {
166
+ const id = 1;
167
+ const select = { id: true };
168
+ const expectedResult = { id: 1 };
169
+
170
+ prisma.rolePages.findUnique.mockResolvedValue(expectedResult);
171
+
172
+ const result = await service.findUnique(id, select);
173
+
174
+ expect(prisma.rolePages.findUnique).toHaveBeenCalledWith({
175
+ where: { id },
176
+ select,
177
+ });
178
+ expect(result).toEqual(expectedResult);
179
+ });
180
+
181
+ it('should return null when record not found', async () => {
182
+ prisma.rolePages.findUnique.mockResolvedValue(null);
183
+
184
+ const result = await service.findUnique(1, { id: true });
185
+
186
+ expect(result).toBeNull();
187
+ });
188
+ });
189
+
190
+ describe('updateUnique', () => {
191
+ it('should update a single rolePages record', async () => {
192
+ const id = 1;
193
+ const data = {
194
+ creator: 'test_creator',
195
+ } as any;
196
+ const expectedResult = { id: 1, ...data };
197
+
198
+ prisma.rolePages.update.mockResolvedValue(expectedResult);
199
+
200
+ const result = await service.updateUnique(id, data);
201
+
202
+ expect(prisma.rolePages.update).toHaveBeenCalledWith({
203
+ where: { id },
204
+ data,
205
+ });
206
+ expect(result).toEqual(expectedResult);
207
+ });
208
+ });
209
+
210
+ describe('updateMany', () => {
211
+ it('should update multiple rolePages records', async () => {
212
+ const updateArgs = {
213
+ where: {},
214
+ data: {
215
+ creator: 'test_creator',
216
+ },
217
+ } as any;
218
+ const expectedResult = { count: 5 };
219
+
220
+ prisma.rolePages.updateMany.mockResolvedValue(expectedResult);
221
+
222
+ const result = await service.updateMany(updateArgs);
223
+
224
+ expect(prisma.rolePages.updateMany).toHaveBeenCalledWith(updateArgs);
225
+ expect(result).toEqual(expectedResult);
226
+ });
227
+ });
228
+
229
+ describe('deleteUnique', () => {
230
+ it('should delete a single rolePages record and return count', async () => {
231
+ const id = 1;
232
+
233
+ prisma.rolePages.delete.mockResolvedValue({ id: 1 });
234
+
235
+ const result = await service.deleteUnique(id);
236
+
237
+ expect(prisma.rolePages.delete).toHaveBeenCalledWith({ where: { id } });
238
+ expect(result).toEqual({ count: 1 });
239
+ });
240
+ });
241
+
242
+ describe('deleteMany', () => {
243
+ it('should delete multiple rolePages records', async () => {
244
+ const deleteArgs = { where: {} } as any;
245
+ const expectedResult = { count: 3 };
246
+
247
+ prisma.rolePages.deleteMany.mockResolvedValue(expectedResult);
248
+
249
+ const result = await service.deleteMany(deleteArgs);
250
+
251
+ expect(prisma.rolePages.deleteMany).toHaveBeenCalledWith(deleteArgs);
252
+ expect(result).toEqual(expectedResult);
253
+ });
254
+ });
255
+
256
+ describe('count', () => {
257
+ it('should count rolePages records with filter', async () => {
258
+ const countArgs = { where: {} } as any;
259
+
260
+ prisma.rolePages.count.mockResolvedValue(10);
261
+
262
+ const result = await service.count(countArgs);
263
+
264
+ expect(prisma.rolePages.count).toHaveBeenCalledWith(countArgs);
265
+ expect(result).toBe(10);
266
+ });
267
+
268
+ it('should count all records when no filter provided', async () => {
269
+ prisma.rolePages.count.mockResolvedValue(100);
270
+
271
+ const result = await service.count();
272
+
273
+ expect(prisma.rolePages.count).toHaveBeenCalledWith(undefined);
274
+ expect(result).toBe(100);
275
+ });
276
+ });
277
+
278
+ describe('groupBy', () => {
279
+ it('should group rolePages records by specified field', async () => {
280
+ const groupByArgs = {
281
+ by: ['id'],
282
+ _count: { id: true },
283
+ } as any;
284
+ const expectedResult = [{ id: 1, _count: { id: 50 } }];
285
+
286
+ prisma.rolePages.groupBy.mockResolvedValue(expectedResult);
287
+
288
+ const result = await service.groupBy(groupByArgs);
289
+
290
+ expect(prisma.rolePages.groupBy).toHaveBeenCalledWith(groupByArgs);
291
+ expect(result).toEqual(expectedResult);
292
+ });
293
+ });
294
+
295
+ describe('aggregate', () => {
296
+ it('should aggregate rolePages records', async () => {
297
+ const aggregateArgs = {
298
+ _count: true,
299
+ _max: { id: true },
300
+ _min: { id: true },
301
+ } as any;
302
+ const expectedResult = {
303
+ _count: 100,
304
+ _max: { id: 1 },
305
+ _min: { id: 1 },
306
+ };
307
+
308
+ prisma.rolePages.aggregate.mockResolvedValue(expectedResult);
309
+
310
+ const result = await service.aggregate(aggregateArgs);
311
+
312
+ expect(prisma.rolePages.aggregate).toHaveBeenCalledWith(aggregateArgs);
313
+ expect(result).toEqual(expectedResult);
314
+ });
315
+ });
18
316
  });