1. replaceAll (Find & Replace) — une seule transaction ProseMirror au lieu d'un forEach cassé. Tous les matchs sont maintenant remplacés. 2. Link Preview unwrap — deleteNode() au lieu de clearer les attrs qui laissaient un nœud fantôme invisible dans le document. 3. Conversion Markdown → richtext — breaks: true dans marked.parse() Les simple newlines sont maintenant convertis en <br>. + préserve les blocs custom (toggle, callout, math, columns, outline, link-preview) en commentaires HTML lors de l'export MD. 4. emitNoteChange exercices — shape corrigée (type:'created' attend un objet Note, pas noteId/notebookId séparés). 5. Raccourcis clavier sans conflit : Cmd+Shift+C → Cmd+Alt+C (callout, avant: copier) Cmd+Shift+O → Cmd+Alt+O (outline, avant: historique/signets) Cmd+Shift+L → Cmd+Alt+L (colonnes, avant: lock screen macOS)
76 lines
1.9 KiB
TypeScript
76 lines
1.9 KiB
TypeScript
'use server'
|
|
|
|
import { semanticSearchService, SearchResult } from '@/lib/ai/services/semantic-search.service'
|
|
import { auth } from '@/auth'
|
|
import { reserveUsageOrThrow, QuotaExceededError } from '@/lib/entitlements'
|
|
|
|
export interface SemanticSearchResponse {
|
|
results: SearchResult[]
|
|
query: string
|
|
totalResults: number
|
|
}
|
|
|
|
/**
|
|
* Perform hybrid semantic + keyword search
|
|
* Supports contextual search within notebook (IA5)
|
|
*/
|
|
export async function semanticSearch(
|
|
query: string,
|
|
options?: {
|
|
limit?: number
|
|
threshold?: number
|
|
notebookId?: string // NEW: Filter by notebook for contextual search (IA5)
|
|
}
|
|
): Promise<SemanticSearchResponse> {
|
|
const session = await auth();
|
|
if (session?.user?.id) {
|
|
try {
|
|
await reserveUsageOrThrow(session.user.id, 'semantic_search');
|
|
} catch (err) {
|
|
if (err instanceof QuotaExceededError) throw err;
|
|
console.error('[semantic-search] Quota check error (fail-open):', err);
|
|
}
|
|
}
|
|
|
|
try {
|
|
const results = await semanticSearchService.search(query, {
|
|
limit: options?.limit || 20,
|
|
threshold: options?.threshold || 0.3,
|
|
notebookId: options?.notebookId // NEW: Pass notebook filter
|
|
})
|
|
|
|
return {
|
|
results,
|
|
query,
|
|
totalResults: results.length
|
|
}
|
|
} catch (error) {
|
|
console.error('Error in semantic search action:', error)
|
|
throw error
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Index a note for semantic search (generate embedding)
|
|
*/
|
|
export async function indexNote(noteId: string): Promise<void> {
|
|
try {
|
|
await semanticSearchService.indexNote(noteId)
|
|
} catch (error) {
|
|
console.error('Error indexing note:', error)
|
|
throw error
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Batch index notes (for initial setup)
|
|
*/
|
|
export async function batchIndexNotes(noteIds: string[]): Promise<void> {
|
|
try {
|
|
await semanticSearchService.indexBatchNotes(noteIds)
|
|
} catch (error) {
|
|
console.error('Error batch indexing notes:', error)
|
|
throw error
|
|
}
|
|
}
|