// Wild Dragon MAM - API Helper Module const API_BASE = '/api/v1'; const CAPTURE_BASE = '/capture'; // Session ID for the active capture — set on start, cleared on stop let _captureSessionId = null; /** * Wrapper around fetch with JSON parsing and error handling */ async function api(path, options = {}) { const url = `${API_BASE}${path}`; const config = { credentials: 'include', headers: { 'Content-Type': 'application/json', ...options.headers, }, ...options, }; try { const response = await fetch(url, config); if (!response.ok) { const error = await response.json().catch(() => ({ message: response.statusText })); throw new Error(error.message || `API Error: ${response.status}`); } const data = await response.json(); return { success: true, data }; } catch (error) { console.error(`API Error on ${path}:`, error); return { success: false, error: error.message }; } } /** * Wrapper around fetch for capture endpoints (goes directly to capture service via nginx) */ async function captureApi(path, options = {}) { const url = `${CAPTURE_BASE}${path}`; const config = { credentials: 'include', headers: { 'Content-Type': 'application/json', ...options.headers, }, ...options, }; try { const response = await fetch(url, config); if (!response.ok) { const error = await response.json().catch(() => ({ message: response.statusText })); throw new Error(error.message || `Capture API Error: ${response.status}`); } const data = await response.json(); return { success: true, data }; } catch (error) { console.error(`Capture API Error on ${path}:`, error); return { success: false, error: error.message }; } } // ============================================================ // ASSET API CALLS // ============================================================ async function getAssets(filters = {}) { const params = new URLSearchParams(filters); const path = `/assets${params.toString() ? '?' + params.toString() : ''}`; return api(path); } async function getAsset(assetId) { return api(`/assets/${assetId}`); } async function getAssetStreamUrl(assetId) { return api(`/assets/${assetId}/stream`); } async function updateAsset(assetId, data) { return api(`/assets/${assetId}`, { method: 'PATCH', body: JSON.stringify(data), }); } async function searchAssets(query) { return api(`/assets?q=${encodeURIComponent(query)}`); } async function getAssetMetadata(assetId) { return api(`/assets/${assetId}`); } // ============================================================ // PROJECT API CALLS // ============================================================ async function getProjects() { return api('/projects'); } async function getProject(projectId) { return api(`/projects/${projectId}`); } async function createProject(name, description = '') { return api('/projects', { method: 'POST', body: JSON.stringify({ name, description }), }); } async function updateProject(projectId, data) { return api(`/projects/${projectId}`, { method: 'PATCH', body: JSON.stringify(data), }); } async function deleteProject(projectId) { return api(`/projects/${projectId}`, { method: 'DELETE', }); } // ============================================================ // BIN API CALLS // ============================================================ async function getBins(projectId) { return api(`/bins?project_id=${projectId}`); } async function getBin(projectId, binId) { return api(`/bins/${binId}`); } async function createBin(projectId, name, description = '') { return api('/bins', { method: 'POST', body: JSON.stringify({ project_id: projectId, name }), }); } async function updateBin(projectId, binId, data) { return api(`/bins/${binId}`, { method: 'PATCH', body: JSON.stringify(data), }); } async function deleteBin(projectId, binId) { return api(`/bins/${binId}`, { method: 'DELETE', }); } // ============================================================ // CAPTURE API CALLS // ============================================================ async function getCaptureDevices() { const result = await captureApi('/devices'); if (result.success && result.data) { const raw = Array.isArray(result.data) ? result.data : (result.data.devices || []); result.data = raw.map(d => ({ id: d.index !== undefined ? d.index : d.id, name: d.name, interface: d.interface || 'DeckLink', })); } return result; } async function getCaptureStatus() { return captureApi('/status'); } async function getRecordingStatus() { return captureApi('/status'); } async function startRecording(deviceIndex, projectId, binId, clipName) { const result = await captureApi('/start', { method: 'POST', body: JSON.stringify({ device: parseInt(deviceIndex, 10), project_id: projectId, bin_id: binId || null, clip_name: clipName, }), }); if (result.success && result.data && result.data.sessionId) { _captureSessionId = result.data.sessionId; } return result; } async function stopRecording() { let sessionId = _captureSessionId; if (!sessionId) { const statusResult = await captureApi('/status'); if (statusResult.success && statusResult.data && statusResult.data.sessionId) { sessionId = statusResult.data.sessionId; } } const result = await captureApi('/stop', { method: 'POST', body: JSON.stringify({ session_id: sessionId }), }); if (result.success) { _captureSessionId = null; } return result; } async function getRecentCaptures(limit = 10) { return api(`/assets?limit=${limit}`); } async function getRecordingTimecode() { return { success: true, data: { timecode: null } }; } // ============================================================ // UTILITY FUNCTIONS // ============================================================ function formatFileSize(bytes) { if (bytes === 0) return '0 B'; const k = 1024; const sizes = ['B', 'KB', 'MB', 'GB', 'TB']; const i = Math.floor(Math.log(bytes) / Math.log(k)); return Math.round((bytes / Math.pow(k, i)) * 100) / 100 + ' ' + sizes[i]; } function formatDuration(seconds) { if (!seconds || seconds < 0) return '00:00:00'; const hours = Math.floor(seconds / 3600); const minutes = Math.floor((seconds % 3600) / 60); const secs = Math.floor(seconds % 60); return [hours, minutes, secs].map(v => v.toString().padStart(2, '0')).join(':'); } function getStatusLabel(status) { const labels = { ingesting: 'Ingesting', processing: 'Processing', ready: 'Ready', error: 'Error', archived: 'Archived', }; return labels[status] || status; } function getStatusBadgeClass(status) { return `badge-${status}`; } function debounce(func, wait) { let timeout; return function executedFunction(...args) { const later = () => { clearTimeout(timeout); func(...args); }; clearTimeout(timeout); timeout = setTimeout(later, wait); }; } function throttle(func, limit) { let inThrottle; return function(...args) { if (!inThrottle) { func.apply(this, args); inThrottle = true; setTimeout(() => inThrottle = false, limit); } }; } // ============================================================ // UPLOAD API CALLS // ============================================================ async function initUpload(data) { const body = { filename: data.filename, fileSize: data.file_size || data.fileSize, contentType: data.content_type || data.contentType, projectId: data.project_id || data.projectId, binId: data.bin_id || data.binId || null, tags: data.tags || null, }; return api('/upload/init', { method: 'POST', body: JSON.stringify(body) }); } async function completeUpload(data) { const body = { uploadId: data.upload_id || data.uploadId, key: data.key, assetId: data.asset_id || data.assetId, parts: (data.parts || []).map(p => ({ partNumber: p.part_number || p.partNumber, ETag: p.etag || p.ETag, })), }; return api('/upload/complete', { method: 'POST', body: JSON.stringify(body) }); } async function abortUpload(data) { const body = { uploadId: data.upload_id || data.uploadId, key: data.key, assetId: data.asset_id || data.assetId, }; return api('/upload/abort', { method: 'POST', body: JSON.stringify(body) }); } async function uploadPart(formData) { try { const response = await fetch('/api/v1/upload/part', { method: 'POST', credentials: 'include', body: formData, }); if (!response.ok) throw new Error(`Upload part failed: ${response.status}`); const data = await response.json(); return { success: true, data }; } catch (error) { return { success: false, error: error.message }; } } async function simpleUpload(formData) { try { const response = await fetch('/api/v1/upload/simple', { method: 'POST', credentials: 'include', body: formData, }); if (!response.ok) throw new Error(`Upload failed: ${response.status}`); const data = await response.json(); return { success: true, data }; } catch (error) { return { success: false, error: error.message }; } } // ============================================================ // RECORDER API CALLS // ============================================================ async function getRecorders() { return api('/recorders'); } async function createRecorder(data) { return api('/recorders', { method: 'POST', body: JSON.stringify(data) }); } async function startRecorder(id) { return api(`/recorders/${id}/start`, { method: 'POST' }); } async function stopRecorder(id) { return api(`/recorders/${id}/stop`, { method: 'POST' }); } async function deleteRecorder(id) { return api(`/recorders/${id}`, { method: 'DELETE' }); } async function getRecorderStatus(id) { return api(`/recorders/${id}/status`); } // ============================================================ // USERS API CALLS (admin) // ============================================================ async function getUsers() { return api('/users'); } async function createUser(data) { return api('/users', { method: 'POST', body: JSON.stringify(data) }); } async function updateUser(id, data) { return api(`/users/${id}`, { method: 'PATCH', body: JSON.stringify(data) }); } async function deleteUser(id) { return api(`/users/${id}`, { method: 'DELETE' }); } // ============================================================ // GROUPS API CALLS (admin) // ============================================================ async function getGroups() { return api('/groups'); } async function createGroup(data) { return api('/groups', { method: 'POST', body: JSON.stringify(data) }); } async function updateGroup(id, data) { return api(`/groups/${id}`, { method: 'PATCH', body: JSON.stringify(data) }); } async function deleteGroup(id) { return api(`/groups/${id}`, { method: 'DELETE' }); } async function getGroupMembers(id) { return api(`/groups/${id}/members`); } async function addGroupMember(groupId, userId) { return api(`/groups/${groupId}/members`, { method: 'POST', body: JSON.stringify({ user_id: userId }), }); } async function removeGroupMember(groupId, userId) { return api(`/groups/${groupId}/members/${userId}`, { method: 'DELETE' }); } // ============================================================ // TOKENS API CALLS // ============================================================ async function getTokens() { return api('/tokens'); } async function createToken(data) { return api('/tokens', { method: 'POST', body: JSON.stringify(data) }); } async function revokeToken(id) { return api(`/tokens/${id}`, { method: 'DELETE' }); }