file.service.test.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
  2. import { DataSource } from 'typeorm';
  3. import { FileService } from '../file.service';
  4. import { File } from '../file.entity';
  5. import { MinioService } from '../minio.service';
  6. import { logger } from '@/server/utils/logger';
  7. // Mock dependencies
  8. vi.mock('../minio.service');
  9. vi.mock('@/server/utils/logger');
  10. vi.mock('uuid', () => ({
  11. v4: () => 'test-uuid-123'
  12. }));
  13. describe('FileService', () => {
  14. let fileService: FileService;
  15. let mockDataSource: DataSource;
  16. let mockMinioService: MinioService;
  17. beforeEach(() => {
  18. mockDataSource = {
  19. getRepository: vi.fn()
  20. } as unknown as DataSource;
  21. mockMinioService = new MinioService();
  22. (MinioService as any).mockClear();
  23. fileService = new FileService(mockDataSource);
  24. });
  25. afterEach(() => {
  26. vi.clearAllMocks();
  27. });
  28. describe('createFile', () => {
  29. it('should create file with upload policy successfully', async () => {
  30. const mockFileData = {
  31. name: 'test.txt',
  32. type: 'text/plain',
  33. size: 1024,
  34. uploadUserId: 1
  35. };
  36. const mockUploadPolicy = {
  37. 'x-amz-algorithm': 'test-algorithm',
  38. 'x-amz-credential': 'test-credential',
  39. host: 'https://minio.example.com'
  40. };
  41. const mockSavedFile = {
  42. id: 1,
  43. ...mockFileData,
  44. path: '1/test-uuid-123-test.txt',
  45. uploadTime: new Date(),
  46. createdAt: new Date(),
  47. updatedAt: new Date()
  48. };
  49. // Mock MinioService
  50. vi.mocked(mockMinioService.generateUploadPolicy).mockResolvedValue(mockUploadPolicy);
  51. // Mock GenericCrudService methods
  52. vi.spyOn(fileService, 'create').mockResolvedValue(mockSavedFile as File);
  53. const result = await fileService.createFile(mockFileData);
  54. expect(mockMinioService.generateUploadPolicy).toHaveBeenCalledWith('1/test-uuid-123-test.txt');
  55. expect(fileService.create).toHaveBeenCalledWith(expect.objectContaining({
  56. name: 'test.txt',
  57. path: '1/test-uuid-123-test.txt',
  58. uploadUserId: 1
  59. }));
  60. expect(result).toEqual({
  61. file: mockSavedFile,
  62. uploadPolicy: mockUploadPolicy
  63. });
  64. });
  65. it('should handle errors during file creation', async () => {
  66. const mockFileData = {
  67. name: 'test.txt',
  68. uploadUserId: 1
  69. };
  70. vi.mocked(mockMinioService.generateUploadPolicy).mockRejectedValue(new Error('MinIO error'));
  71. await expect(fileService.createFile(mockFileData)).rejects.toThrow('文件创建失败');
  72. expect(logger.error).toHaveBeenCalled();
  73. });
  74. });
  75. describe('deleteFile', () => {
  76. it('should delete file successfully when file exists', async () => {
  77. const mockFile = {
  78. id: 1,
  79. path: '1/test-file.txt',
  80. name: 'test-file.txt'
  81. } as File;
  82. vi.spyOn(fileService, 'getById').mockResolvedValue(mockFile);
  83. vi.mocked(mockMinioService.objectExists).mockResolvedValue(true);
  84. vi.mocked(mockMinioService.deleteObject).mockResolvedValue(undefined);
  85. vi.spyOn(fileService, 'delete').mockResolvedValue(undefined);
  86. const result = await fileService.deleteFile(1);
  87. expect(fileService.getById).toHaveBeenCalledWith(1);
  88. expect(mockMinioService.objectExists).toHaveBeenCalledWith('d8dai', '1/test-file.txt');
  89. expect(mockMinioService.deleteObject).toHaveBeenCalledWith('d8dai', '1/test-file.txt');
  90. expect(fileService.delete).toHaveBeenCalledWith(1);
  91. expect(result).toBe(true);
  92. });
  93. it('should delete database record even when MinIO file not found', async () => {
  94. const mockFile = {
  95. id: 1,
  96. path: '1/test-file.txt',
  97. name: 'test-file.txt'
  98. } as File;
  99. vi.spyOn(fileService, 'getById').mockResolvedValue(mockFile);
  100. vi.mocked(mockMinioService.objectExists).mockResolvedValue(false);
  101. vi.spyOn(fileService, 'delete').mockResolvedValue(undefined);
  102. const result = await fileService.deleteFile(1);
  103. expect(mockMinioService.deleteObject).not.toHaveBeenCalled();
  104. expect(fileService.delete).toHaveBeenCalledWith(1);
  105. expect(result).toBe(true);
  106. expect(logger.error).toHaveBeenCalled();
  107. });
  108. it('should throw error when file not found', async () => {
  109. vi.spyOn(fileService, 'getById').mockResolvedValue(null);
  110. await expect(fileService.deleteFile(999)).rejects.toThrow('文件不存在');
  111. });
  112. });
  113. describe('getFileUrl', () => {
  114. it('should return file URL successfully', async () => {
  115. const mockFile = {
  116. id: 1,
  117. path: '1/test-file.txt'
  118. } as File;
  119. const mockPresignedUrl = 'https://minio.example.com/presigned-url';
  120. vi.spyOn(fileService, 'getById').mockResolvedValue(mockFile);
  121. vi.mocked(mockMinioService.getPresignedFileUrl).mockResolvedValue(mockPresignedUrl);
  122. const result = await fileService.getFileUrl(1);
  123. expect(fileService.getById).toHaveBeenCalledWith(1);
  124. expect(mockMinioService.getPresignedFileUrl).toHaveBeenCalledWith('d8dai', '1/test-file.txt');
  125. expect(result).toBe(mockPresignedUrl);
  126. });
  127. it('should throw error when file not found', async () => {
  128. vi.spyOn(fileService, 'getById').mockResolvedValue(null);
  129. await expect(fileService.getFileUrl(999)).rejects.toThrow('文件不存在');
  130. });
  131. });
  132. describe('getFileDownloadUrl', () => {
  133. it('should return download URL with filename', async () => {
  134. const mockFile = {
  135. id: 1,
  136. path: '1/test-file.txt',
  137. name: '测试文件.txt'
  138. } as File;
  139. const mockPresignedUrl = 'https://minio.example.com/download-url';
  140. vi.spyOn(fileService, 'getById').mockResolvedValue(mockFile);
  141. vi.mocked(mockMinioService.getPresignedFileDownloadUrl).mockResolvedValue(mockPresignedUrl);
  142. const result = await fileService.getFileDownloadUrl(1);
  143. expect(fileService.getById).toHaveBeenCalledWith(1);
  144. expect(mockMinioService.getPresignedFileDownloadUrl).toHaveBeenCalledWith(
  145. 'd8dai',
  146. '1/test-file.txt',
  147. '测试文件.txt'
  148. );
  149. expect(result).toEqual({
  150. url: mockPresignedUrl,
  151. filename: '测试文件.txt'
  152. });
  153. });
  154. it('should throw error when file not found', async () => {
  155. vi.spyOn(fileService, 'getById').mockResolvedValue(null);
  156. await expect(fileService.getFileDownloadUrl(999)).rejects.toThrow('文件不存在');
  157. });
  158. });
  159. describe('createMultipartUploadPolicy', () => {
  160. it('should create multipart upload policy successfully', async () => {
  161. const mockFileData = {
  162. name: 'large-file.zip',
  163. type: 'application/zip',
  164. uploadUserId: 1
  165. };
  166. const mockUploadId = 'upload-123';
  167. const mockUploadUrls = ['url1', 'url2', 'url3'];
  168. const mockSavedFile = {
  169. id: 1,
  170. ...mockFileData,
  171. path: '1/test-uuid-123-large-file.zip',
  172. uploadTime: new Date(),
  173. createdAt: new Date(),
  174. updatedAt: new Date()
  175. } as File;
  176. vi.mocked(mockMinioService.createMultipartUpload).mockResolvedValue(mockUploadId);
  177. vi.mocked(mockMinioService.generateMultipartUploadUrls).mockResolvedValue(mockUploadUrls);
  178. vi.spyOn(fileService, 'create').mockResolvedValue(mockSavedFile);
  179. const result = await fileService.createMultipartUploadPolicy(mockFileData, 3);
  180. expect(mockMinioService.createMultipartUpload).toHaveBeenCalledWith('d8dai', '1/test-uuid-123-large-file.zip');
  181. expect(mockMinioService.generateMultipartUploadUrls).toHaveBeenCalledWith(
  182. 'd8dai',
  183. '1/test-uuid-123-large-file.zip',
  184. mockUploadId,
  185. 3
  186. );
  187. expect(result).toEqual({
  188. file: mockSavedFile,
  189. uploadId: mockUploadId,
  190. uploadUrls: mockUploadUrls,
  191. bucket: 'd8dai',
  192. key: '1/test-uuid-123-large-file.zip'
  193. });
  194. });
  195. it('should handle errors during multipart upload creation', async () => {
  196. const mockFileData = {
  197. name: 'large-file.zip',
  198. uploadUserId: 1
  199. };
  200. vi.mocked(mockMinioService.createMultipartUpload).mockRejectedValue(new Error('MinIO error'));
  201. await expect(fileService.createMultipartUploadPolicy(mockFileData, 3)).rejects.toThrow('创建多部分上传策略失败');
  202. expect(logger.error).toHaveBeenCalled();
  203. });
  204. });
  205. describe('completeMultipartUpload', () => {
  206. it('should complete multipart upload successfully', async () => {
  207. const uploadData = {
  208. uploadId: 'upload-123',
  209. bucket: 'd8dai',
  210. key: '1/test-file.txt',
  211. parts: [
  212. { partNumber: 1, etag: 'etag1' },
  213. { partNumber: 2, etag: 'etag2' }
  214. ]
  215. };
  216. const mockFile = {
  217. id: 1,
  218. path: '1/test-file.txt',
  219. size: 0,
  220. updatedAt: new Date()
  221. } as File;
  222. const mockCompleteResult = { size: 2048 };
  223. const mockFileUrl = 'https://minio.example.com/file.txt';
  224. vi.mocked(mockMinioService.completeMultipartUpload).mockResolvedValue(mockCompleteResult);
  225. vi.mocked(mockMinioService.getFileUrl).mockReturnValue(mockFileUrl);
  226. vi.spyOn(fileService.repository, 'findOneBy').mockResolvedValue(mockFile);
  227. vi.spyOn(fileService.repository, 'save').mockResolvedValue(mockFile);
  228. const result = await fileService.completeMultipartUpload(uploadData);
  229. expect(mockMinioService.completeMultipartUpload).toHaveBeenCalledWith(
  230. 'd8dai',
  231. '1/test-file.txt',
  232. 'upload-123',
  233. [{ PartNumber: 1, ETag: 'etag1' }, { PartNumber: 2, ETag: 'etag2' }]
  234. );
  235. expect(fileService.repository.findOneBy).toHaveBeenCalledWith({ path: '1/test-file.txt' });
  236. expect(fileService.repository.save).toHaveBeenCalledWith(expect.objectContaining({
  237. size: 2048
  238. }));
  239. expect(result).toEqual({
  240. fileId: 1,
  241. url: mockFileUrl,
  242. key: '1/test-file.txt',
  243. size: 2048
  244. });
  245. });
  246. it('should throw error when file record not found', async () => {
  247. const uploadData = {
  248. uploadId: 'upload-123',
  249. bucket: 'd8dai',
  250. key: '1/nonexistent.txt',
  251. parts: [{ partNumber: 1, etag: 'etag1' }]
  252. };
  253. vi.mocked(mockMinioService.completeMultipartUpload).mockResolvedValue({ size: 1024 });
  254. vi.spyOn(fileService.repository, 'findOneBy').mockResolvedValue(null);
  255. await expect(fileService.completeMultipartUpload(uploadData)).rejects.toThrow('文件记录不存在');
  256. });
  257. it('should handle errors during completion', async () => {
  258. const uploadData = {
  259. uploadId: 'upload-123',
  260. bucket: 'd8dai',
  261. key: '1/test-file.txt',
  262. parts: [{ partNumber: 1, etag: 'etag1' }]
  263. };
  264. vi.mocked(mockMinioService.completeMultipartUpload).mockRejectedValue(new Error('Completion failed'));
  265. await expect(fileService.completeMultipartUpload(uploadData)).rejects.toThrow('完成分片上传失败');
  266. expect(logger.error).toHaveBeenCalled();
  267. });
  268. });
  269. });