require ('dotenv').config(); const express = require('express'); const {getEnv, PORTS, EPISODIC} = require('@nexusai/shared'); const { getDB } = require('./db'); const { createProject, getProjects, getProject, updateProject, deleteProject } = require('./db/projects'); const { createSummary, getSummary, getSummariesBySession, getSummariesByProject, updateSummary, deleteSummary } = require('./db/summaries'); const episodic = require('./episodic'); const semantic = require('./semantic'); const entities = require('./entities'); const app = express(); app.use(express.json()); const PORT = getEnv('PORT', PORTS.MEMORY); //initialize database on startup const db = getDB(); semantic.initCollections() .then(() => console.log(`QDrant collections ready`)) .catch(err => console.error(`QDrant initialization error:`, err.message)); // Health check endpoint app.get('/health', (req, res) => { res.json({ service: 'Memory Service', status: 'healthy' }); }); /************************************ */ /********** Session Routes ********** */ /************************************ */ // Creates a new session with an external ID and optional metadata app.get('/sessions', (req, res) => { const { limit = EPISODIC.DEFAULT_PAGE_SIZE, offset = EPISODIC.DEFAULT_OFFSET, projectId } = req.query; const parsedProjectId = projectId && projectId !== 'null' ? Number(projectId) : null; const sessions = episodic.getSessions(Number(limit), Number(offset), parsedProjectId); res.json(sessions); }); app.post('/sessions', (req, res) => { const { externalId, metadata } = req.body; if (!externalId) { return res.status(400).json({ error: 'externalId is required' }); } try { const session = episodic.createSession(externalId, metadata); res.status(201).json(session); } catch (err) { res.status(409).json({ error: 'Session already exists', detail: err.message }); } }); // Retrieves a session by its external ID app.get('/sessions/by-external/:externalId', (req, res) => { const session = episodic.getSessionByExternalId(req.params.externalId); if (!session) return res.status(404).json({ error: 'Session not found' }); res.json(session); }); // Retrieves a session by its internal ID app.get('/sessions/:id', (req, res) => { const session = episodic.getSession(req.params.id); if (!session) return res.status(404).json({ error: 'Session not found' }); res.json(session); }); app.patch('/sessions/by-external/:externalId', (req, res) => { const { name, projectId } = req.body; try { const session = episodic.updateSessionByExternalId(req.params.externalId, {name, projectId }); res.json(session); } catch (err) { res.status(500).json({error: err.message }); } }); // Deletes a session and all associated episodes app.delete('/sessions/by-external/:externalId', (req, res) => { episodic.deleteSessionByExternalId(req.params.externalId); res.status(204).send(); }); /************************************* */ /********** Episodic Routes ********** */ /************************************* */ app.post('/episodes', async (req, res) => { const { sessionId, userMessage, aiResponse, tokenCount, projectId } = req.body; if (!sessionId || !userMessage || !aiResponse) { return res.status(400).json({ error: 'sessionId, userMessage and aiResponse are required' }); } const episode = await episodic.createEpisode(sessionId, userMessage, aiResponse, tokenCount, projectId); res.status(201).json(episode); }); app.get('/episodes', (req, res) => { const { limit = 50, offset = 0, sessionId, q } = req.query; if (q) { const results = episodic.searchEpisodes(q, Number(limit)); return res.json({ episodes: results, total: results.length }); } const db = getDB(); let episodes; if (sessionId) { episodes = episodic.getEpisodesBySession(Number(sessionId), Number(limit), Number(offset)); } else { episodes = db.prepare( `SELECT * FROM episodes ORDER BY created_at DESC LIMIT ? OFFSET ?` ).all(Number(limit), Number(offset)).map(row => require('@nexusai/shared').parseRow(row)); } const total = db.prepare(`SELECT COUNT(*) as count FROM episodes`).get().count; res.json({ episodes, total }); }); // Search MUST come before /:id — otherwise 'search' gets captured as an id app.get('/episodes/search', (req, res) => { const { q, limit = EPISODIC.DEFAULT_PAGE_SIZE } = req.query; if (!q) return res.status(400).json({ error: 'q (query) parameter is required' }); const results = episodic.searchEpisodes(q, Number(limit)); res.json(results); }); app.get('/episodes/:id', (req, res) => { const episode = episodic.getEpisode(req.params.id); if (!episode) return res.status(404).json({ error: 'Episode not found' }); res.json(episode); }); // Get paginated episodes for a session app.get('/sessions/:id/episodes', (req, res) => { const { limit = EPISODIC.DEFAULT_PAGE_SIZE, offset = EPISODIC.DEFAULT_OFFSET } = req.query; const episodes = episodic.getEpisodesBySession( req.params.id, Number(limit), Number(offset) ); res.json(episodes); }); app.delete('/episodes/:id', (req, res) => { const id = Number(req.params.id); episodic.deleteEpisode(id); semantic.deleteEpisode(id) // fire-and-forget .catch(err => console.error(`[Memory] Qdrant delete failed for episode ${id}:`, err.message)); res.status(204).send(); }); /*********************************** */ /********** Entity Routes ********** */ /*********************************** */ //Upsert an entity, creates or updates if already exists app.post('/entities', (req, res) => { const {name, type, notes, metadata} = req.body; if (!name || !type) { return res.status(400).json({ error: 'name and type are required' }); } const entity = entities.upsertEntity(name, type, notes, metadata); res.status(201).json(entity); }); // Get all entities of a given type app.get('/entities/by-type/:type', (req, res) => { res.json(entities.getEntitiesByType(req.params.type)); }); // Get an entity by ID app.get('/entities/:id', (req, res) => { const entity = entities.getEntity(req.params.id); if (!entity) return res.status(404).json({ error: 'Entity not found' }); res.json(entity); }); // Delete an entity by ID app.delete('/entities/:id', (req, res) => { entities.deleteEntity(req.params.id); res.status(204).send(); }); /***************************************** */ /********** Relationship Routes ********** */ /***************************************** */ // Upsert a relationship between two entities app.post('/relationships', (req, res) => { const {fromId, toId, label, metadata } = req.body; if (!fromId || !toId || !label) { return res.status(400).json({ error: 'fromId, toId and label are required' }); } const relationship = entities.upsertRelationship(fromId, toId, label, metadata); res.status(201).json(relationship); }); // Get all relationships for a given entity ID app.get('/entities/:id/relationships', (req, res) => { res.json(entities.getRelationshipsByEntity(req.params.id)); }); // Delete a specific relationship app.delete('/relationships', (req, res) => { const {fromId, toId, label} = req.body; if (!fromId || !toId || !label) { return res.status(400).json({ error: 'fromId, toId and label are required' }); } entities.deleteRelationship(fromId, toId, label); res.status(204).send(); }) /*********************************** */ /********** Project Routes ********** */ /*********************************** */ app.post('/projects', (req, res) => { const { name, description, colour, icon } = req.body; if (!name?.trim()) return res.status(400).json({ error: 'name is required' }); try { res.status(201).json(createProject({ name: name.trim(), description, colour, icon })); } catch (err) { res.status(500).json({ error: err.message }); } }); app.get('/projects', (req, res) => { res.json(getProjects()); }); app.get('/projects/:id', (req, res) => { const project = getProject(req.params.id); if (!project) return res.status(404).json({ error: 'Not found' }); res.json(project); }); app.patch('/projects/:id', (req, res) => { const project = getProject(req.params.id); if (!project) return res.status(404).json({ error: 'Not found' }); res.json(updateProject(req.params.id, req.body)); }); app.delete('/projects/:id', (req, res) => { const project = getProject(req.params.id); if (!project) return res.status(404).json({ error: 'Not found' }); deleteProject(req.params.id); res.status(204).send(); }); /*********************************** */ /********** Summary Routes ********** */ /*********************************** */ // Create a summary (called by orchestration, fire-and-forget style) app.post('/summaries', (req, res) => { const { sessionId, projectId, content, tokenCount, episodeRange, metadata } = req.body; if (!content) return res.status(400).json({ error: 'content is required' }); if (!sessionId && !projectId) return res.status(400).json({ error: 'sessionId or projectId is required' }); try { const summary = createSummary({ sessionId, projectId, content, tokenCount, episodeRange, metadata }); res.status(201).json(summary); } catch (err) { res.status(500).json({ error: err.message }); } }); // Get summaries for a session app.get('/sessions/:id/summaries', (req, res) => { res.json(getSummariesBySession(req.params.id)); }); // Get summaries for a project app.get('/projects/:id/summaries', (req, res) => { res.json(getSummariesByProject(req.params.id)); }); // Update a summary (for cumulative updates) app.patch('/summaries/:id', (req, res) => { const summary = getSummary(req.params.id); if (!summary) return res.status(404).json({ error: 'Not found' }); res.json(updateSummary(req.params.id, req.body)); }); // Delete a summary app.delete('/summaries/:id', (req, res) => { deleteSummary(req.params.id); res.status(204).send(); }); /********************************** */ /********** Start Server ********** */ /********************************** */ app.listen(PORT, () => { console.log(`Memory Service is running on port ${PORT}`); });