mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-28 00:00:42 +00:00
- Search results now include per-signal attribution (vec/bm25/graph rank+score) threaded through RRF fusion to memory_recall output and auto-recall debug logs - New --report flag on sleep command shows post-cycle quality metrics (extraction coverage, entity graph density, decay distribution) - New `health` subcommand with 5-section dashboard: memory overview, extraction health, entity graph, tag health, decay distribution Supports --agent scoping and --json output Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
203 lines
4.8 KiB
TypeScript
203 lines
4.8 KiB
TypeScript
/**
|
|
* Graph schema types, Cypher query templates, and constants for memory-neo4j.
|
|
*/
|
|
|
|
// ============================================================================
|
|
// Shared Types
|
|
// ============================================================================
|
|
|
|
export type Logger = {
|
|
info: (msg: string) => void;
|
|
warn: (msg: string) => void;
|
|
error: (msg: string) => void;
|
|
debug?: (msg: string) => void;
|
|
};
|
|
|
|
// ============================================================================
|
|
// Node Types
|
|
// ============================================================================
|
|
|
|
export type MemoryCategory = "core" | "preference" | "fact" | "decision" | "entity" | "other";
|
|
export type EntityType = "person" | "organization" | "location" | "event" | "concept";
|
|
export type ExtractionStatus = "pending" | "complete" | "failed" | "skipped";
|
|
export type MemorySource =
|
|
| "user"
|
|
| "auto-capture"
|
|
| "auto-capture-assistant"
|
|
| "memory-watcher"
|
|
| "import";
|
|
|
|
export type MemoryNode = {
|
|
id: string;
|
|
text: string;
|
|
embedding: number[];
|
|
importance: number;
|
|
category: MemoryCategory;
|
|
source: MemorySource;
|
|
createdAt: string;
|
|
updatedAt: string;
|
|
extractionStatus: ExtractionStatus;
|
|
extractionRetries: number;
|
|
agentId: string;
|
|
sessionKey?: string;
|
|
retrievalCount: number;
|
|
lastRetrievedAt?: string;
|
|
};
|
|
|
|
export type EntityNode = {
|
|
id: string;
|
|
name: string;
|
|
type: EntityType;
|
|
aliases: string[];
|
|
description?: string;
|
|
firstSeen: string;
|
|
lastSeen: string;
|
|
mentionCount: number;
|
|
};
|
|
|
|
export type TagNode = {
|
|
id: string;
|
|
name: string;
|
|
category: string;
|
|
createdAt: string;
|
|
};
|
|
|
|
// ============================================================================
|
|
// Extraction Types
|
|
// ============================================================================
|
|
|
|
export type ExtractedEntity = {
|
|
name: string;
|
|
type: EntityType;
|
|
aliases?: string[];
|
|
description?: string;
|
|
};
|
|
|
|
export type ExtractedRelationship = {
|
|
source: string;
|
|
target: string;
|
|
type: string;
|
|
confidence: number;
|
|
};
|
|
|
|
export type ExtractedTag = {
|
|
name: string;
|
|
category: string;
|
|
};
|
|
|
|
export type ExtractionResult = {
|
|
category?: MemoryCategory;
|
|
entities: ExtractedEntity[];
|
|
relationships: ExtractedRelationship[];
|
|
tags: ExtractedTag[];
|
|
};
|
|
|
|
// ============================================================================
|
|
// Search Types
|
|
// ============================================================================
|
|
|
|
export type SearchSignalResult = {
|
|
id: string;
|
|
text: string;
|
|
category: string;
|
|
importance: number;
|
|
createdAt: string;
|
|
score: number;
|
|
};
|
|
|
|
export type SignalAttribution = {
|
|
rank: number; // 1-indexed, 0 = absent from this signal
|
|
score: number; // raw signal score, 0 = absent
|
|
};
|
|
|
|
export type HybridSearchResult = {
|
|
id: string;
|
|
text: string;
|
|
category: string;
|
|
importance: number;
|
|
createdAt: string;
|
|
score: number;
|
|
signals?: {
|
|
vector: SignalAttribution;
|
|
bm25: SignalAttribution;
|
|
graph: SignalAttribution;
|
|
};
|
|
};
|
|
|
|
// ============================================================================
|
|
// Input Types
|
|
// ============================================================================
|
|
|
|
export type StoreMemoryInput = {
|
|
id: string;
|
|
text: string;
|
|
embedding: number[];
|
|
importance: number;
|
|
category: MemoryCategory;
|
|
source: MemorySource;
|
|
extractionStatus: ExtractionStatus;
|
|
agentId: string;
|
|
sessionKey?: string;
|
|
};
|
|
|
|
export type MergeEntityInput = {
|
|
id: string;
|
|
name: string;
|
|
type: EntityType;
|
|
aliases?: string[];
|
|
description?: string;
|
|
};
|
|
|
|
// ============================================================================
|
|
// Constants
|
|
// ============================================================================
|
|
|
|
export const MEMORY_CATEGORIES = [
|
|
"core",
|
|
"preference",
|
|
"fact",
|
|
"decision",
|
|
"entity",
|
|
"other",
|
|
] as const;
|
|
|
|
export const ENTITY_TYPES = ["person", "organization", "location", "event", "concept"] as const;
|
|
|
|
export const ALLOWED_RELATIONSHIP_TYPES = new Set([
|
|
"WORKS_AT",
|
|
"LIVES_AT",
|
|
"KNOWS",
|
|
"MARRIED_TO",
|
|
"PREFERS",
|
|
"DECIDED",
|
|
"RELATED_TO",
|
|
]);
|
|
|
|
// ============================================================================
|
|
// Lucene Helpers
|
|
// ============================================================================
|
|
|
|
const LUCENE_SPECIAL_CHARS = /[+\-&|!(){}[\]^"~*?:\\/]/g;
|
|
|
|
/**
|
|
* Escape special characters for Lucene fulltext search queries.
|
|
*/
|
|
export function escapeLucene(query: string): string {
|
|
return query.replace(LUCENE_SPECIAL_CHARS, "\\$&");
|
|
}
|
|
|
|
/**
|
|
* Validate that a relationship type is in the allowed set.
|
|
* Prevents Cypher injection via dynamic relationship type.
|
|
*/
|
|
export function validateRelationshipType(type: string): boolean {
|
|
return ALLOWED_RELATIONSHIP_TYPES.has(type);
|
|
}
|
|
|
|
/**
|
|
* Create a canonical key for a pair of IDs (sorted for order-independence).
|
|
*/
|
|
export function makePairKey(a: string, b: string): string {
|
|
return a < b ? `${a}:${b}` : `${b}:${a}`;
|
|
}
|