Files
Antigravity e881004c77
Some checks failed
CI / Lint, Test & Build (push) Failing after 1m7s
CI / Deploy production (on server) (push) Has been skipped
feat(insights): fix DBSCAN, Persian embeddings crash, D3 physics layouts, and D3 node not found runtime error
2026-05-24 18:57:33 +00:00

220 lines
7.4 KiB
TypeScript

import { NextRequest, NextResponse } from 'next/server'
import { auth } from '@/auth'
import prisma from '@/lib/prisma'
import { clusteringService } from '@/lib/ai/services/clustering.service'
import { bridgeNotesService } from '@/lib/ai/services/bridge-notes.service'
/**
* GET /api/clusters
* Get all clusters for the current user.
* Returns cached clusters + bridge notes enriched with cluster names.
*/
export async function GET(request: NextRequest) {
try {
const session = await auth()
if (!session?.user?.id) {
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
}
const userId = session.user.id
// Check for stored results (even if stale/périmés)
const stored = await clusteringService.getStoredClusters(userId)
if (stored) {
const cached = stored.clusters
// Fetch notes with their cluster assignments
const notes = await prisma.note.findMany({
where: { userId, trashedAt: null },
select: { id: true, title: true, content: true }
})
// Get cluster member mappings
const clusterMembers = await prisma.clusterMember.findMany({
where: { userId },
select: { noteId: true, clusterId: true, isCentral: true }
})
const noteClusterMap = new Map(clusterMembers.map(cm => [cm.noteId, { clusterId: cm.clusterId, isCentral: cm.isCentral }]))
const notesWithClusters = notes.map(n => {
const mapping = noteClusterMap.get(n.id)
return {
...n,
clusterId: mapping?.clusterId,
isCentral: mapping?.isCentral || false
}
})
// Fetch bridge notes with enrichment (cluster names + note details)
const bridgeNotesData = await prisma.bridgeNote.findMany({
where: { userId },
orderBy: { bridgeScore: 'desc' },
take: 10
})
let enrichedBridgeNotes: object[] = []
if (bridgeNotesData.length > 0) {
const bridgeNoteIds = bridgeNotesData.map(b => b.noteId)
const bridgeNoteDetails = await prisma.note.findMany({
where: { id: { in: bridgeNoteIds } },
select: { id: true, title: true, content: true }
})
const bridgeNoteDetailsMap = new Map(bridgeNoteDetails.map(n => [n.id, n]))
enrichedBridgeNotes = bridgeNotesData.map(b => {
const clustersConnected = JSON.parse(b.clustersConnected) as number[]
return {
noteId: b.noteId,
bridgeScore: b.bridgeScore,
clustersConnected,
clusterNames: clustersConnected.map(
cid => cached.find(c => c.clusterId === cid)?.name || `Cluster ${cid}`
),
note: bridgeNoteDetailsMap.get(b.noteId)
}
})
}
const embeddingCountRow = await prisma.$queryRawUnsafe<Array<{ count: bigint }>>(
`SELECT COUNT(*) FROM "NoteEmbedding" ne
INNER JOIN "Note" n ON n.id = ne."noteId"
WHERE n."userId" = $1 AND n."trashedAt" IS NULL`,
userId
)
return NextResponse.json({
clusters: cached,
notes: notesWithClusters,
bridgeNotes: enrichedBridgeNotes,
cached: true,
stale: stored.stale,
lastCalculated: stored.lastCalculated,
totalNotes: notes.length,
embeddingCount: Number(embeddingCountRow[0]?.count || 0),
})
}
// No cached results - return info about notes
const notesCount = await prisma.note.count({
where: { userId, trashedAt: null }
})
const embeddingCount = await prisma.$queryRawUnsafe<Array<{ count: bigint }>>(
`SELECT COUNT(*) FROM "NoteEmbedding" ne
INNER JOIN "Note" n ON n.id = ne."noteId"
WHERE n."userId" = $1 AND n."trashedAt" IS NULL`,
userId
)
return NextResponse.json({
clusters: [],
notes: [],
bridgeNotes: [],
totalNotes: notesCount,
embeddingCount: Number(embeddingCount[0]?.count || 0),
needsCalculation: true
})
} catch (error) {
console.error('Error fetching clusters:', error)
return NextResponse.json(
{ error: 'Failed to fetch clusters' },
{ status: 500 }
)
}
}
/**
* POST /api/clusters
* Trigger a full recalculation of clusters and bridge notes.
* Returns clusters + bridge notes enriched for immediate display.
*/
export async function POST(request: NextRequest) {
try {
const session = await auth()
if (!session?.user?.id) {
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
}
const userId = session.user.id
const body = await request.json().catch(() => ({}))
const force = Boolean(body?.force)
// 0. Indexer / réindexer les embeddings (texte complet, multi-chunks)
await clusteringService.ensureEmbeddings(userId, { force: force || true })
// 1. Run DBSCAN clustering
const results = await clusteringService.clusterNotes(userId)
if (results.clusters.length === 0) {
return NextResponse.json({
clusters: [],
bridgeNotes: [],
message: 'Could not generate clusters. Notes may be too diverse or not enough.',
noiseCount: results.noiseCount
})
}
// 2. Generate cluster names with AI
for (const cluster of results.clusters) {
cluster.name = await clusteringService.generateClusterName(cluster.clusterId, userId)
}
// 3. Save clustering results
await clusteringService.saveClusteringResults(userId, results)
// 4. Detect and save bridge notes
const bridgeNotes = await bridgeNotesService.detectBridgeNotes(userId)
await bridgeNotesService.saveBridgeNotes(userId, bridgeNotes)
// 5. Generate and save bridge suggestions
const suggestions = await bridgeNotesService.generateBridgeSuggestions(userId)
await bridgeNotesService.saveBridgeSuggestions(userId, suggestions)
// 6. Fetch notes with cluster assignments
const notes = await prisma.note.findMany({
where: { userId, trashedAt: null },
select: { id: true, title: true, content: true }
})
const clusterMembers = await prisma.clusterMember.findMany({
where: { userId },
select: { noteId: true, clusterId: true, isCentral: true }
})
const noteClusterMap = new Map(clusterMembers.map(cm => [cm.noteId, { clusterId: cm.clusterId, isCentral: cm.isCentral }]))
const notesWithClusters = notes.map(n => {
const mapping = noteClusterMap.get(n.id)
return {
...n,
clusterId: mapping?.clusterId,
isCentral: mapping?.isCentral || false
}
})
// 7. Enrich bridge notes with cluster names + note details
const noteMap = new Map(notes.map(n => [n.id, n]))
const enrichedBridgeNotes = bridgeNotes.slice(0, 10).map(b => ({
noteId: b.noteId,
bridgeScore: b.bridgeScore,
clustersConnected: b.clustersConnected,
clusterNames: b.clusterNames,
note: noteMap.get(b.noteId)
}))
return NextResponse.json({
clusters: results.clusters,
notes: notesWithClusters,
bridgeNotes: enrichedBridgeNotes,
totalNotes:
results.clusters.reduce((sum, c) => sum + c.noteIds.length, 0) + results.noiseCount,
noiseCount: results.noiseCount,
message: `Generated ${results.clusters.length} clusters with ${bridgeNotes.length} bridge notes`
})
} catch (error) {
console.error('Error recalculating clusters:', error)
return NextResponse.json(
{ error: 'Failed to recalculate clusters', details: String(error) },
{ status: 500 }
)
}
}