data-overview.service.test.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. import { describe, it, expect, beforeEach, vi } from 'vitest';
  2. import { DataSource, Repository } from 'typeorm';
  3. import { OrderMt } from '@d8d/orders-module-mt';
  4. import { DataOverviewServiceMt, TimeFilterParams, SummaryStatistics } from '../../src/services/data-overview.service';
  5. // Mock redisUtil and AppDataSource - use hoisted to ensure availability before vi.mock
  6. const { mockRedisUtil, mockAppDataSource } = vi.hoisted(() => {
  7. return {
  8. mockRedisUtil: {
  9. get: vi.fn(),
  10. set: vi.fn(),
  11. keys: vi.fn(),
  12. del: vi.fn()
  13. },
  14. mockAppDataSource: {}
  15. };
  16. });
  17. vi.mock('@d8d/shared-utils', async () => {
  18. const actual = await vi.importActual<typeof import('@d8d/shared-utils')>('@d8d/shared-utils');
  19. return {
  20. ...actual,
  21. redisUtil: mockRedisUtil,
  22. AppDataSource: mockAppDataSource
  23. };
  24. });
  25. describe('DataOverviewServiceMt', () => {
  26. let service: DataOverviewServiceMt;
  27. let mockDataSource: DataSource;
  28. let mockOrderRepository: Repository<OrderMt>;
  29. beforeEach(() => {
  30. // Mock Order Repository
  31. mockOrderRepository = {
  32. createQueryBuilder: vi.fn(() => ({
  33. select: vi.fn().mockReturnThis(),
  34. where: vi.fn().mockReturnThis(),
  35. andWhere: vi.fn().mockReturnThis(),
  36. setParameters: vi.fn().mockReturnThis(),
  37. getRawOne: vi.fn()
  38. }))
  39. } as any;
  40. // Mock DataSource
  41. mockDataSource = {
  42. getRepository: vi.fn((entity) => {
  43. if (entity === OrderMt) {
  44. return mockOrderRepository;
  45. }
  46. return {} as any;
  47. })
  48. } as any;
  49. // Reset redisUtil mocks
  50. mockRedisUtil.get.mockReset();
  51. mockRedisUtil.set.mockReset();
  52. mockRedisUtil.keys.mockReset();
  53. mockRedisUtil.del.mockReset();
  54. service = new DataOverviewServiceMt(mockDataSource);
  55. });
  56. describe('getDateRange', () => {
  57. it('应该返回今天的时间范围(默认)', () => {
  58. const now = new Date('2025-12-26T10:00:00Z');
  59. vi.setSystemTime(now);
  60. const params: TimeFilterParams = {};
  61. const result = service['getDateRange'](params);
  62. const expectedStart = new Date('2025-12-26T00:00:00Z');
  63. const expectedEnd = now; // 对于今天的时间范围,结束时间是当前时间
  64. expect(result.startDate.toISOString()).toBe(expectedStart.toISOString());
  65. expect(result.endDate.toISOString()).toBe(expectedEnd.toISOString());
  66. });
  67. it('应该返回昨天的时间范围', () => {
  68. const now = new Date('2025-12-26T10:00:00Z');
  69. vi.setSystemTime(now);
  70. const params: TimeFilterParams = { timeRange: 'yesterday' };
  71. const result = service['getDateRange'](params);
  72. const expectedStart = new Date('2025-12-25T00:00:00Z');
  73. const expectedEnd = new Date('2025-12-25T23:59:59.999Z');
  74. expect(result.startDate.toISOString()).toBe(expectedStart.toISOString());
  75. expect(result.endDate.toISOString()).toBe(expectedEnd.toISOString());
  76. });
  77. it('应该返回最近7天的时间范围', () => {
  78. const now = new Date('2025-12-26T10:00:00Z');
  79. vi.setSystemTime(now);
  80. const params: TimeFilterParams = { timeRange: 'last7days' };
  81. const result = service['getDateRange'](params);
  82. const expectedStart = new Date('2025-12-19T10:00:00Z'); // 7天前
  83. const expectedEnd = now;
  84. expect(result.startDate.toISOString()).toBe(expectedStart.toISOString());
  85. expect(result.endDate.toISOString()).toBe(expectedEnd.toISOString());
  86. });
  87. it('应该返回最近30天的时间范围', () => {
  88. const now = new Date('2025-12-26T10:00:00Z');
  89. vi.setSystemTime(now);
  90. const params: TimeFilterParams = { timeRange: 'last30days' };
  91. const result = service['getDateRange'](params);
  92. const expectedStart = new Date('2025-11-26T10:00:00Z'); // 30天前
  93. const expectedEnd = now;
  94. expect(result.startDate.toISOString()).toBe(expectedStart.toISOString());
  95. expect(result.endDate.toISOString()).toBe(expectedEnd.toISOString());
  96. });
  97. it('应该返回自定义时间范围', () => {
  98. const params: TimeFilterParams = {
  99. timeRange: 'custom',
  100. startDate: '2025-01-01T00:00:00Z',
  101. endDate: '2025-01-31T23:59:59Z'
  102. };
  103. const result = service['getDateRange'](params);
  104. expect(result.startDate.toISOString()).toBe('2025-01-01T00:00:00.000Z');
  105. expect(result.endDate.toISOString()).toBe('2025-01-31T23:59:59.000Z');
  106. });
  107. it('当自定义时间范围缺少参数时应该使用默认值', () => {
  108. const now = new Date('2025-12-26T10:00:00Z');
  109. vi.setSystemTime(now);
  110. const params: TimeFilterParams = { timeRange: 'custom' }; // 缺少startDate和endDate
  111. const result = service['getDateRange'](params);
  112. const expectedStart = new Date('2025-12-26T00:00:00Z');
  113. const expectedEnd = now; // 当自定义范围缺少参数时,使用默认的今天范围,结束时间为当前时间
  114. expect(result.startDate.toISOString()).toBe(expectedStart.toISOString());
  115. expect(result.endDate.toISOString()).toBe(expectedEnd.toISOString());
  116. });
  117. });
  118. describe('getSummaryStatistics', () => {
  119. it('应该从缓存返回统计数据', async () => {
  120. const tenantId = 1;
  121. const params: TimeFilterParams = { timeRange: 'today' };
  122. const cacheKey = `data_overview:summary:${tenantId}:today::`;
  123. const cachedStats: SummaryStatistics = {
  124. totalSales: 10000,
  125. totalOrders: 50,
  126. wechatSales: 6000,
  127. wechatOrders: 30,
  128. creditSales: 4000,
  129. creditOrders: 20,
  130. todaySales: 500,
  131. todayOrders: 5
  132. };
  133. mockRedisUtil.get.mockResolvedValue(JSON.stringify(cachedStats));
  134. const result = await service.getSummaryStatistics(tenantId, params);
  135. expect(result).toEqual(cachedStats);
  136. expect(mockRedisUtil.get).toHaveBeenCalledWith(cacheKey);
  137. expect(mockRedisUtil.set).not.toHaveBeenCalled();
  138. });
  139. it('当缓存未命中时应该查询数据库并设置缓存', async () => {
  140. const tenantId = 1;
  141. const params: TimeFilterParams = { timeRange: 'today' };
  142. const cacheKey = `data_overview:summary:${tenantId}:today::`;
  143. mockRedisUtil.get.mockResolvedValue(null);
  144. // Mock database query result
  145. const mockQueryBuilder = {
  146. select: vi.fn().mockReturnThis(),
  147. where: vi.fn().mockReturnThis(),
  148. andWhere: vi.fn().mockReturnThis(),
  149. setParameters: vi.fn().mockReturnThis(),
  150. getRawOne: vi.fn().mockResolvedValue({
  151. total_orders: '50',
  152. total_sales: '10000.50',
  153. wechat_sales: '6000.00',
  154. credit_sales: '4000.50',
  155. wechat_orders: '30',
  156. credit_orders: '20'
  157. })
  158. };
  159. const mockTodayQueryBuilder = {
  160. select: vi.fn().mockReturnThis(),
  161. where: vi.fn().mockReturnThis(),
  162. andWhere: vi.fn().mockReturnThis(),
  163. getRawOne: vi.fn().mockResolvedValue({
  164. today_orders: '5',
  165. today_sales: '500.00'
  166. })
  167. };
  168. vi.mocked(mockOrderRepository.createQueryBuilder)
  169. .mockReturnValueOnce(mockQueryBuilder as any)
  170. .mockReturnValueOnce(mockTodayQueryBuilder as any);
  171. const result = await service.getSummaryStatistics(tenantId, params);
  172. expect(result.totalSales).toBe(10000.50);
  173. expect(result.totalOrders).toBe(50);
  174. expect(result.wechatSales).toBe(6000);
  175. expect(result.creditSales).toBe(4000.50);
  176. expect(result.wechatOrders).toBe(30);
  177. expect(result.creditOrders).toBe(20);
  178. expect(result.todaySales).toBe(500);
  179. expect(result.todayOrders).toBe(5);
  180. expect(mockRedisUtil.set).toHaveBeenCalledWith(
  181. cacheKey,
  182. expect.any(String),
  183. 5 * 60 // 5分钟TTL(今日数据)
  184. );
  185. });
  186. it('应该为历史数据设置30分钟缓存', async () => {
  187. const tenantId = 1;
  188. const params: TimeFilterParams = { timeRange: 'last7days' };
  189. const cacheKey = `data_overview:summary:${tenantId}:last7days::`;
  190. mockRedisUtil.get.mockResolvedValue(null);
  191. const mockQueryBuilder = {
  192. select: vi.fn().mockReturnThis(),
  193. where: vi.fn().mockReturnThis(),
  194. andWhere: vi.fn().mockReturnThis(),
  195. setParameters: vi.fn().mockReturnThis(),
  196. getRawOne: vi.fn().mockResolvedValue({
  197. total_orders: '100',
  198. total_sales: '20000.00',
  199. wechat_sales: '12000.00',
  200. credit_sales: '8000.00',
  201. wechat_orders: '60',
  202. credit_orders: '40'
  203. })
  204. };
  205. const mockTodayQueryBuilder = {
  206. select: vi.fn().mockReturnThis(),
  207. where: vi.fn().mockReturnThis(),
  208. andWhere: vi.fn().mockReturnThis(),
  209. getRawOne: vi.fn().mockResolvedValue({
  210. today_orders: '10',
  211. today_sales: '1000.00'
  212. })
  213. };
  214. vi.mocked(mockOrderRepository.createQueryBuilder)
  215. .mockReturnValueOnce(mockQueryBuilder as any)
  216. .mockReturnValueOnce(mockTodayQueryBuilder as any);
  217. await service.getSummaryStatistics(tenantId, params);
  218. expect(mockRedisUtil.set).toHaveBeenCalledWith(
  219. cacheKey,
  220. expect.any(String),
  221. 30 * 60 // 30分钟TTL(历史数据)
  222. );
  223. });
  224. });
  225. describe('getTodayStatistics', () => {
  226. it('应该从缓存返回今日统计数据', async () => {
  227. const tenantId = 1;
  228. const today = new Date().toISOString().split('T')[0];
  229. const cacheKey = `data_overview:today:${tenantId}:${today}`;
  230. const cachedStats = { todaySales: 500, todayOrders: 5 };
  231. mockRedisUtil.get.mockResolvedValue(JSON.stringify(cachedStats));
  232. const result = await service.getTodayStatistics(tenantId);
  233. expect(result).toEqual(cachedStats);
  234. expect(mockRedisUtil.get).toHaveBeenCalledWith(cacheKey);
  235. expect(mockRedisUtil.set).not.toHaveBeenCalled();
  236. });
  237. it('当缓存未命中时应该查询数据库并设置缓存', async () => {
  238. const tenantId = 1;
  239. const today = new Date().toISOString().split('T')[0];
  240. const cacheKey = `data_overview:today:${tenantId}:${today}`;
  241. mockRedisUtil.get.mockResolvedValue(null);
  242. const mockQueryBuilder = {
  243. select: vi.fn().mockReturnThis(),
  244. where: vi.fn().mockReturnThis(),
  245. andWhere: vi.fn().mockReturnThis(),
  246. getRawOne: vi.fn().mockResolvedValue({
  247. today_orders: '5',
  248. today_sales: '500.00'
  249. })
  250. };
  251. vi.mocked(mockOrderRepository.createQueryBuilder).mockReturnValue(mockQueryBuilder as any);
  252. const result = await service.getTodayStatistics(tenantId);
  253. expect(result.todaySales).toBe(500);
  254. expect(result.todayOrders).toBe(5);
  255. expect(mockRedisUtil.set).toHaveBeenCalledWith(cacheKey, expect.any(String), 5 * 60);
  256. });
  257. });
  258. describe('clearCache', () => {
  259. it('应该清理指定租户的所有缓存', async () => {
  260. const tenantId = 1;
  261. const cacheKeys = [
  262. 'data_overview:summary:1:today::',
  263. 'data_overview:today:1:2025-12-26'
  264. ];
  265. mockRedisUtil.keys.mockResolvedValue(cacheKeys);
  266. await service.clearCache(tenantId);
  267. expect(mockRedisUtil.keys).toHaveBeenCalledWith('data_overview:*:1:*');
  268. expect(mockRedisUtil.del).toHaveBeenCalledWith(...cacheKeys);
  269. });
  270. it('当没有缓存键时不应该调用del', async () => {
  271. const tenantId = 1;
  272. mockRedisUtil.keys.mockResolvedValue([]);
  273. await service.clearCache(tenantId);
  274. expect(mockRedisUtil.keys).toHaveBeenCalledWith('data_overview:*:1:*');
  275. expect(mockRedisUtil.del).not.toHaveBeenCalled();
  276. });
  277. });
  278. });