Added a new `.coveragerc` file configuring coverage thresholds and exclusions. Included `pytest.ini` to enable coverage reporting for multiple backend modules (`main`, `llm`, `prompt`, `geoip`, `tts_asr`) with a 90 % fail‑under requirement and detailed HTML output. Implemented a suite of unit tests: * `test_geoip.py` – validates geo‑location lookup logic. * `test_llm_extended.py` – tests LLm response extraction and Ollama interactions. * `test_main_endpoints.py` – covers API endpoints for completions, OCR, and TTS. * `test_prompt_extended.py` – verifies language sanitization, timestamp generation, and prompt building. * `test_tts_asr_coverage.py` – checks device detection, cache clearing, and model loading under various environment configurations. * `test_tts_asr_extended.py` – further tests TTS/ASR device selection and time‑outs. Updated `backend/requirements.txt` to use newer, compatible packages, removed obsolete testing dependencies, and added `qwen-tts`. Modified `backend/tts_asr.py` to work with the new `Qwen3TTSModel`, simplified imports, and adjusted device mapping logic. Additionally, frontend changes added a new `TreeNodeItem` component, updated Markdown rendering, added TTS instruction fields, and reworked context menu handling. No breaking changes were introduced.
161 lines
4.3 KiB
JavaScript
161 lines
4.3 KiB
JavaScript
import { API_URL, API_KEY, TTS_URL, TTS_STATUS_URL, TTS_CONFIG_URL } from './config.js'
|
|
import { useSettingsStore } from '../stores/settings'
|
|
|
|
function generateRequestId() {
|
|
if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {
|
|
return crypto.randomUUID()
|
|
}
|
|
return `${Date.now()}-${Math.random().toString(16).slice(2)}`
|
|
}
|
|
|
|
function getCancelUrl(apiUrl) {
|
|
const normalized = String(apiUrl || '').replace(/\/+$/, '')
|
|
if (!normalized) return '/v1/completions/cancel'
|
|
if (normalized.endsWith('/v1/completions')) {
|
|
return `${normalized}/cancel`
|
|
}
|
|
return `${normalized}/cancel`
|
|
}
|
|
|
|
function normalizeAbortReason(reason) {
|
|
if (typeof reason === 'string' && reason.trim()) {
|
|
return reason.trim().slice(0, 64)
|
|
}
|
|
return 'abort'
|
|
}
|
|
|
|
async function sendCancelRequest(cancelUrl, requestId, reason) {
|
|
try {
|
|
await fetch(cancelUrl, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-API-Key': API_KEY,
|
|
},
|
|
body: JSON.stringify({
|
|
request_id: requestId,
|
|
reason,
|
|
}),
|
|
})
|
|
} catch {
|
|
// Cancel request failed silently
|
|
}
|
|
}
|
|
|
|
export async function fetchSuggestion(prefix, suffix, languageId, signal, apiUrl = API_URL) {
|
|
let normalizedLanguageId = 'markdown'
|
|
if (typeof languageId === 'string' && languageId.trim()) {
|
|
normalizedLanguageId = languageId.trim()
|
|
} else if (languageId && typeof languageId === 'object' && 'aborted' in languageId) {
|
|
signal = languageId
|
|
}
|
|
if (typeof signal === 'string') {
|
|
apiUrl = signal
|
|
signal = undefined
|
|
}
|
|
const requestId = generateRequestId()
|
|
const cancelUrl = getCancelUrl(apiUrl)
|
|
|
|
const onAbort = () => {
|
|
const reason = normalizeAbortReason(signal?.reason)
|
|
void sendCancelRequest(cancelUrl, requestId, reason)
|
|
}
|
|
|
|
if (signal) {
|
|
if (signal.aborted) {
|
|
onAbort()
|
|
} else {
|
|
signal.addEventListener('abort', onAbort, { once: true })
|
|
}
|
|
}
|
|
|
|
try {
|
|
const settings = useSettingsStore()
|
|
const headers = {
|
|
'Content-Type': 'application/json',
|
|
'X-Request-Id': requestId,
|
|
'X-API-Key': API_KEY,
|
|
}
|
|
|
|
const body = {
|
|
prefix,
|
|
suffix,
|
|
languageId: normalizedLanguageId,
|
|
model_thinking: settings.modelThinking,
|
|
privacy_mode: settings.privacyMode,
|
|
user_preferences: {
|
|
language: settings.language,
|
|
currency: settings.currency,
|
|
timezone: settings.detectedTimezone,
|
|
},
|
|
}
|
|
|
|
const res = await fetch(apiUrl, {
|
|
method: 'POST',
|
|
headers,
|
|
body: JSON.stringify(body),
|
|
signal,
|
|
})
|
|
|
|
if (!res.ok) {
|
|
const errorText = await res.text()
|
|
throw new Error(`HTTP ${res.status}: ${errorText}`)
|
|
}
|
|
|
|
const data = await res.json()
|
|
return data.content || ''
|
|
} catch (e) {
|
|
if (e.name === 'AbortError') {
|
|
// ignore abort
|
|
} else {
|
|
throw e
|
|
}
|
|
} finally {
|
|
if (signal) {
|
|
signal.removeEventListener('abort', onAbort)
|
|
}
|
|
}
|
|
}
|
|
|
|
export async function fetchTTS(text, instruct = '', apiUrl = TTS_URL) {
|
|
const res = await fetch(apiUrl, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-API-Key': API_KEY,
|
|
},
|
|
body: JSON.stringify({ text, instruct, speaker: 'Vivian', format: 'wav' }),
|
|
})
|
|
|
|
if (!res.ok) {
|
|
const errorText = await res.text()
|
|
throw new Error(`TTS HTTP ${res.status}: ${errorText}`)
|
|
}
|
|
|
|
return res.json()
|
|
}
|
|
|
|
export async function fetchTTSStatus(apiUrl = TTS_STATUS_URL) {
|
|
const res = await fetch(apiUrl, {
|
|
headers: { 'X-API-Key': API_KEY },
|
|
})
|
|
|
|
if (!res.ok) {
|
|
throw new Error(`TTS Status HTTP ${res.status}`)
|
|
}
|
|
|
|
return res.json()
|
|
}
|
|
|
|
export async function fetchTTSConfig(apiUrl = TTS_CONFIG_URL) {
|
|
const res = await fetch(apiUrl, {
|
|
headers: { 'X-API-Key': API_KEY },
|
|
})
|
|
|
|
if (!res.ok) {
|
|
throw new Error(`TTS Config HTTP ${res.status}`)
|
|
}
|
|
|
|
return res.json()
|
|
}
|