import { apiClient } from './apiClient'; import { IndexingConfig } from '../types'; // web/services/uploadService.ts export const uploadService = { async uploadFile(file: File, authToken: string): Promise { const formData = new FormData(); formData.append('file', file); const response = await apiClient.request('/upload', { method: 'POST', body: formData, }); if (!response.ok) { const errorData = await response.json(); throw new Error(errorData.message || 'fileUploadFailed'); } return response.json(); }, async uploadFileWithConfig(file: File, config: IndexingConfig, authToken: string): Promise { const formData = new FormData(); formData.append('file', file); formData.append('chunkSize', config.chunkSize.toString()); formData.append('chunkOverlap', config.chunkOverlap.toString()); formData.append('embeddingModelId', config.embeddingModelId); if (config.mode) { formData.append('mode', config.mode); } if (config.groupIds && config.groupIds.length > 0) { formData.append('groupIds', JSON.stringify(config.groupIds)); } const response = await apiClient.request('/upload', { method: 'POST', body: formData, }); if (!response.ok) { const errorData = await response.json(); throw new Error(errorData.message || 'fileUploadFailed'); } return response.json(); }, async uploadText(content: string, title: string, config: IndexingConfig, authToken: string): Promise { const { data } = await apiClient.post('/upload/text', { content, title, chunkSize: config.chunkSize.toString(), chunkOverlap: config.chunkOverlap.toString(), embeddingModelId: config.embeddingModelId, mode: config.mode }); return data; }, async recommendMode(file: File): Promise { if (!file || !file.name) { return { recommendedMode: 'fast', reason: 'invalidFile', warnings: ['incompleteFileInfo'], }; } const ext = file.name.toLowerCase().split('.').pop(); const sizeMB = file.size / (1024 * 1024); const preciseFormats = ['pdf', 'doc', 'docx', 'ppt', 'pptx']; const supportedFormats = [...preciseFormats, 'xls', 'xlsx', 'txt', 'md', 'html', 'json', 'csv']; if (!supportedFormats.includes(ext || '')) { return { recommendedMode: 'fast', reason: `unsupportedFileFormat`, reasonArgs: [ext], warnings: ['willUseFastMode'], }; } if (!preciseFormats.includes(ext || '')) { return { recommendedMode: 'fast', reason: `formatNoPrecise`, reasonArgs: [ext], warnings: ['willUseFastMode'], }; } if (sizeMB < 5) { return { recommendedMode: 'fast', reason: 'smallFileFastOk', estimatedCost: 0, estimatedTime: sizeMB * 2, warnings: [], }; } if (sizeMB < 50) { return { recommendedMode: 'precise', reason: 'mixedContentPreciseRecommended', estimatedCost: Math.max(0.01, sizeMB * 0.01), estimatedTime: sizeMB * 8, warnings: ['willIncurApiCost'], }; } return { recommendedMode: 'precise', reason: 'largeFilePreciseRecommended', estimatedCost: sizeMB * 0.015, estimatedTime: sizeMB * 12, warnings: ['longProcessingTime', 'highApiCost', 'considerFileSplitting'], }; }, };