153 lines
4.4 KiB
JavaScript
153 lines
4.4 KiB
JavaScript
// Model Service for WiFi-DensePose UI
|
|
// Manages model loading, listing, LoRA profiles, and lifecycle events.
|
|
|
|
import { apiService } from './api.service.js';
|
|
|
|
export class ModelService {
|
|
constructor() {
|
|
this.activeModel = null;
|
|
this.listeners = {};
|
|
this.logger = this.createLogger();
|
|
}
|
|
|
|
createLogger() {
|
|
return {
|
|
debug: (...args) => console.debug('[MODEL-DEBUG]', new Date().toISOString(), ...args),
|
|
info: (...args) => console.info('[MODEL-INFO]', new Date().toISOString(), ...args),
|
|
warn: (...args) => console.warn('[MODEL-WARN]', new Date().toISOString(), ...args),
|
|
error: (...args) => console.error('[MODEL-ERROR]', new Date().toISOString(), ...args)
|
|
};
|
|
}
|
|
|
|
// --- Event emitter helpers ---
|
|
|
|
on(event, callback) {
|
|
if (!this.listeners[event]) {
|
|
this.listeners[event] = [];
|
|
}
|
|
this.listeners[event].push(callback);
|
|
return () => this.off(event, callback);
|
|
}
|
|
|
|
off(event, callback) {
|
|
if (!this.listeners[event]) return;
|
|
this.listeners[event] = this.listeners[event].filter(cb => cb !== callback);
|
|
}
|
|
|
|
emit(event, data) {
|
|
if (!this.listeners[event]) return;
|
|
this.listeners[event].forEach(cb => {
|
|
try { cb(data); } catch (err) { this.logger.error('Listener error', { event, err }); }
|
|
});
|
|
}
|
|
|
|
// --- API methods ---
|
|
|
|
async listModels() {
|
|
try {
|
|
const data = await apiService.get('/api/v1/models');
|
|
this.logger.info('Listed models', { count: data?.models?.length ?? 0 });
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to list models', { error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async getModel(id) {
|
|
try {
|
|
const data = await apiService.get(`/api/v1/models/${encodeURIComponent(id)}`);
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to get model', { id, error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async loadModel(modelId) {
|
|
try {
|
|
this.logger.info('Loading model', { modelId });
|
|
const data = await apiService.post('/api/v1/models/load', { model_id: modelId });
|
|
this.activeModel = { model_id: modelId };
|
|
this.emit('model-loaded', { model_id: modelId });
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to load model', { modelId, error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async unloadModel() {
|
|
try {
|
|
this.logger.info('Unloading model');
|
|
const data = await apiService.post('/api/v1/models/unload', {});
|
|
this.activeModel = null;
|
|
this.emit('model-unloaded', {});
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to unload model', { error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async getActiveModel() {
|
|
try {
|
|
const data = await apiService.get('/api/v1/models/active');
|
|
this.activeModel = data || null;
|
|
return this.activeModel;
|
|
} catch (error) {
|
|
if (error.status === 404) {
|
|
this.activeModel = null;
|
|
return null;
|
|
}
|
|
this.logger.error('Failed to get active model', { error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async activateLoraProfile(modelId, profileName) {
|
|
try {
|
|
this.logger.info('Activating LoRA profile', { modelId, profileName });
|
|
const data = await apiService.post(
|
|
'/api/v1/models/lora/activate',
|
|
{ model_id: modelId, profile_name: profileName }
|
|
);
|
|
this.emit('lora-activated', { model_id: modelId, profile: profileName });
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to activate LoRA', { modelId, profileName, error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async getLoraProfiles() {
|
|
try {
|
|
const data = await apiService.get('/api/v1/models/lora/profiles');
|
|
return data?.profiles ?? [];
|
|
} catch (error) {
|
|
this.logger.error('Failed to get LoRA profiles', { error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
async deleteModel(id) {
|
|
try {
|
|
this.logger.info('Deleting model', { id });
|
|
const data = await apiService.delete(`/api/v1/models/${encodeURIComponent(id)}`);
|
|
return data;
|
|
} catch (error) {
|
|
this.logger.error('Failed to delete model', { id, error: error.message });
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
dispose() {
|
|
this.listeners = {};
|
|
this.activeModel = null;
|
|
this.logger.info('ModelService disposed');
|
|
}
|
|
}
|
|
|
|
// Create singleton instance
|
|
export const modelService = new ModelService();
|