export interface ChatMessage { role: 'user' | 'assistant'; content: string; } export interface ChatSource { fileName: string; content: string; score: number; chunkIndex: number; fileId?: string; } export class ChatService { async *streamChat( message: string, history: ChatMessage[], authToken: string, userLanguage: string = 'ja', selectedEmbeddingId?: string, selectedLLMId?: string, // 追加: 選択された LLM ID selectedGroups?: string[], // 追加: 選択されたグループ selectedFiles?: string[], // 追加: 選択されたファイル historyId?: string, // 追加: 会話履歴 ID enableRerank?: boolean, // 追加: Rerank を有効にする selectedRerankId?: string, // 追加: Rerank モデル ID temperature?: number, // 追加: temperature パラメータ maxTokens?: number, // 追加: maxTokens パラメータ topK?: number, // 追加: topK パラメータ similarityThreshold?: number, // 追加: similarityThreshold パラメータ rerankSimilarityThreshold?: number // 追加: rerankSimilarityThreshold パラメータ ): AsyncGenerator<{ type: 'content' | 'sources' | 'error' | 'historyId'; data: any }> { try { const response = await fetch('/api/chat/stream', { method: 'POST', headers: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${authToken}`, 'x-user-language': userLanguage || localStorage.getItem('userLanguage') || 'ja', }, body: JSON.stringify({ message, history, userLanguage, selectedEmbeddingId, selectedLLMId, // Pass LLM ID selectedGroups, // グループフィルタパラメータを渡す selectedFiles, // ファイルフィルタパラメータを渡す historyId, // 履歴 ID を渡す enableRerank, selectedRerankId, temperature, // temperature パラメータを渡す maxTokens, // maxTokens パラメータを渡す topK, // topK パラメータを渡す similarityThreshold, // similarityThreshold パラメータを渡す rerankSimilarityThreshold // rerankSimilarityThreshold パラメータを渡す }), }); if (!response.ok) { let errorMessage = 'リクエストに失敗しました'; try { const error = await response.json(); errorMessage = error.error || error.message || 'リクエストに失敗しました'; } catch { errorMessage = `サーバーエラー: ${response.status}`; } yield { type: 'error', data: errorMessage }; return; } const reader = response.body?.getReader(); if (!reader) { yield { type: 'error', data: 'レスポンスストリームを読み取れません' }; return; } const decoder = new TextDecoder(); let buffer = ''; while (true) { const { done, value } = await reader.read(); if (done) break; buffer += decoder.decode(value, { stream: true }); const lines = buffer.split('\n'); buffer = lines.pop() || ''; for (const line of lines) { if (line.startsWith('data: ')) { const data = line.slice(6); if (data === '[DONE]') { return; } try { const parsed = JSON.parse(data); yield parsed; } catch (e) { console.warn('Failed to parse SSE data:', data); } } } } } catch (error) { yield { type: 'error', data: error.message || 'ネットワークエラー' }; } } async *streamAssist( instruction: string, context: string, authToken: string ): AsyncGenerator<{ type: 'content' | 'error'; data: any }> { try { const response = await fetch('/api/chat/assist', { method: 'POST', headers: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${authToken}`, 'x-user-language': localStorage.getItem('userLanguage') || 'ja', }, body: JSON.stringify({ instruction, context }), }); if (!response.ok) { yield { type: 'error', data: 'リクエストに失敗しました' }; return; } const reader = response.body?.getReader(); if (!reader) return; const decoder = new TextDecoder(); let buffer = ''; while (true) { const { done, value } = await reader.read(); if (done) break; buffer += decoder.decode(value, { stream: true }); const lines = buffer.split('\n'); buffer = lines.pop() || ''; for (const line of lines) { if (line.startsWith('data: ')) { const data = line.slice(6); if (data === '[DONE]') return; try { yield JSON.parse(data); } catch (e) { console.warn(e) } } } } } catch (error: any) { yield { type: 'error', data: error.message }; } } } export const chatService = new ChatService();