data-overview.service.test.ts 11 KB

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