990ea0ae5f
4인 팀 운영 슬래시 (v2.2.173~189):
- 일과 리듬: /morning, /evening, /weekly, /standup
- 트래커 (event-sourced .astra/*.jsonl): /runway, /customers, /hire
- 작업·결정: /task, /blocked, /onesie, /decisions
- 외부 출력: /draft, /feedback
- 분석: /cohort (MoM 추세)
ASTRA 추론·검색 엔진 (v2.2.183~192):
- v2.2.183 Conflict Surface — scoring.conflictSeverity 를 [CONFLICT WARNINGS] 블록으로
서피스 + 교차-문서 발산(Jaccard) 감지
- v2.2.184 Chain-of-Verification — [VERIFICATION CHECKLIST] 답변 작성 전 그라운딩 자기 점검
(instructional, strictMode 옵션)
- v2.2.185 Actionability Scoring — 최근 슬래시 명령 + 열린 파일 신호로 검색 결과 재가중
- v2.2.186 Temporal Markers + Distillation Loop — LongTerm/Episodic 만료 필터 +
30일+ stale episode → LongTerm 'episode-digest' 승급 (수동 /memory distill + 세션 종료 자동)
- v2.2.187 Hierarchical Context Window + LLM Semantic Re-rank — 3-level 추상도 매칭
+ 토큰 예산 통과 후 LLM 1회로 의도-부합 재정렬 (opt-in)
- v2.2.190 Intent Clarification + Citation Trace — 모호 차원 감지 시 역질문 우선
+ 답변 끝 사용 출처 한 줄 정리
- v2.2.191 Post-hoc Self-Check — 답변 완료 후 별도 LLM 호출 1회로 답함/그라운딩/모순 평가,
footer 한 줄로 표시 (opt-in, semantic re-rank 와 같은 안전 fallback 패턴)
- v2.2.192 Terminology Dictionary — .astra/glossary.md 사용자 편집 파일 + Term Check
지침 통합 + /glossary init/path/reload
- v2.2.193 /help — 카테고리별 명령 목록 + 6종 verification 블록 현재 on/off
신규 모듈:
- src/retrieval/{conflictBlock,coveBlock,actionabilityScoring,hierarchicalLevel,
semanticRerank,intentClarification,citationTrace,terminologyBlock}.ts
- src/memory/distillation.ts + types.ts 에 expiresAt/promoted/episode-digest 추가
- src/agent/postHocSelfCheck.ts
- src/features/{customers,feedback,hire,runway}/*.ts (event-sourced stores)
ASTRA 검증 5종 자동 주입 (buildAstraModeSystemPrompt, casual 모드 제외):
[INTENT CLARIFICATION GUIDANCE] (답변 시작 전) → [TERMINOLOGY DICTIONARY] +
[CONFLICT WARNINGS] + [VERIFICATION CHECKLIST] (작성 중) → [CITATION TRACE] (끝)
+ 6번째: Post-hoc Self-Check footer (답변 완료 후, opt-in)
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
226 lines
7.6 KiB
TypeScript
226 lines
7.6 KiB
TypeScript
/**
|
|
* ============================================================
|
|
* MemoryManager — 5-Layer Cognitive Memory System (통합 진입점)
|
|
*
|
|
* Astra의 모든 메모리 레이어를 통합 관리하는 중앙 매니저입니다.
|
|
*
|
|
* ① Short-Term Memory — 현재 대화 흐름 (FIFO)
|
|
* ② Long-Term Memory — 사용자 취향/규칙/결정
|
|
* ③ Project Memory — 프로젝트별 지식
|
|
* ④ Procedural Memory — 반복 작업 절차 (skill.md)
|
|
* ⑤ Episodic Memory — 과거 대화/결정 흐름
|
|
* ============================================================
|
|
*/
|
|
|
|
import { BrainProfile } from '../config';
|
|
import { ShortTermMemory } from './ShortTermMemory';
|
|
import { LongTermMemory } from './LongTermMemory';
|
|
import { ProjectMemory } from './ProjectMemory';
|
|
import { ProceduralMemory } from './ProceduralMemory';
|
|
import { EpisodicMemory } from './EpisodicMemory';
|
|
import { MemoryExtractor } from './MemoryExtractor';
|
|
import { MemoryContextResult, MemoryConfig } from './types';
|
|
import {
|
|
distillStaleEpisodes,
|
|
shouldAutoDistill,
|
|
recordDistillationRun,
|
|
type DistillationArchiveMode,
|
|
} from './distillation';
|
|
|
|
export { ShortTermMemory } from './ShortTermMemory';
|
|
export { LongTermMemory } from './LongTermMemory';
|
|
export { ProjectMemory } from './ProjectMemory';
|
|
export { ProceduralMemory } from './ProceduralMemory';
|
|
export { EpisodicMemory } from './EpisodicMemory';
|
|
export { MemoryExtractor } from './MemoryExtractor';
|
|
export {
|
|
distillStaleEpisodes,
|
|
getLastDistillationRun,
|
|
recordDistillationRun,
|
|
shouldAutoDistill,
|
|
DEFAULT_DISTILLATION_OPTIONS,
|
|
type DistillationOptions,
|
|
type DistillationReport,
|
|
type DistillationMarker,
|
|
type DistillationArchiveMode,
|
|
} from './distillation';
|
|
export * from './types';
|
|
|
|
export class MemoryManager {
|
|
private shortTerm: ShortTermMemory;
|
|
private longTerm: LongTermMemory;
|
|
private procedural: ProceduralMemory;
|
|
private episodic: EpisodicMemory;
|
|
private extractor: MemoryExtractor;
|
|
|
|
// Project Memory는 workspace별로 lazy-init
|
|
private projectMemories = new Map<string, ProjectMemory>();
|
|
|
|
private config: MemoryConfig;
|
|
|
|
constructor(brainPath: string, config?: Partial<MemoryConfig>) {
|
|
this.config = {
|
|
enabled: true,
|
|
shortTermLimit: 8,
|
|
longTermMaxEntries: 100,
|
|
episodicMaxEpisodes: 50,
|
|
projectMemoryEnabled: true,
|
|
proceduralMemoryEnabled: true,
|
|
episodicMemoryEnabled: true,
|
|
...config
|
|
};
|
|
|
|
this.shortTerm = new ShortTermMemory();
|
|
this.longTerm = new LongTermMemory(brainPath, this.config.longTermMaxEntries);
|
|
this.procedural = new ProceduralMemory(brainPath);
|
|
this.episodic = new EpisodicMemory(brainPath, this.config.episodicMaxEpisodes);
|
|
this.extractor = new MemoryExtractor();
|
|
}
|
|
|
|
// ─── Context Building (핵심 API) ───
|
|
|
|
/**
|
|
* 프롬프트에 대해 모든 메모리 레이어에서 관련 컨텍스트를 수집합니다.
|
|
* agent.ts의 buildMemoryContext()를 대체합니다.
|
|
*/
|
|
public buildContext(
|
|
currentPrompt: string,
|
|
visibleHistory: Array<{ role: string; content: string }>,
|
|
summarize: (text: string, maxLen: number) => string,
|
|
workspacePath?: string
|
|
): string {
|
|
if (!this.config.enabled) return '';
|
|
|
|
const layers: MemoryContextResult[] = [];
|
|
|
|
// ① Short-Term Memory
|
|
const stm = this.shortTerm.buildContext(
|
|
visibleHistory,
|
|
this.config.shortTermLimit,
|
|
summarize
|
|
);
|
|
if (stm) layers.push(stm);
|
|
|
|
// ② Long-Term Memory
|
|
const ltm = this.longTerm.buildContext(currentPrompt);
|
|
if (ltm) layers.push(ltm);
|
|
|
|
// ③ Project Memory
|
|
if (this.config.projectMemoryEnabled && workspacePath) {
|
|
const pm = this.getProjectMemory(workspacePath);
|
|
const pmCtx = pm.buildContext(currentPrompt);
|
|
if (pmCtx) layers.push(pmCtx);
|
|
}
|
|
|
|
// ④ Procedural Memory
|
|
if (this.config.proceduralMemoryEnabled) {
|
|
const proc = this.procedural.buildContext(currentPrompt);
|
|
if (proc) layers.push(proc);
|
|
}
|
|
|
|
// ⑤ Episodic Memory
|
|
if (this.config.episodicMemoryEnabled) {
|
|
const ep = this.episodic.buildContext(currentPrompt);
|
|
if (ep) layers.push(ep);
|
|
}
|
|
|
|
if (layers.length === 0) return '';
|
|
|
|
// 관련도 순으로 정렬
|
|
layers.sort((a, b) => b.relevance - a.relevance);
|
|
|
|
const sections = layers
|
|
.map((layer) => `### ${layer.label}\n${layer.content}`)
|
|
.join('\n\n');
|
|
|
|
return [
|
|
'',
|
|
'[MEMORY CONTEXT]',
|
|
'Review this layered memory before preparing the answer. Use it only when relevant, and prefer the current user request when there is conflict.',
|
|
sections
|
|
].join('\n');
|
|
}
|
|
|
|
// ─── Session Lifecycle ───
|
|
|
|
/**
|
|
* 세션 종료 시 호출하여 모든 메모리 레이어에 대해 추출을 수행합니다.
|
|
*/
|
|
public onSessionEnd(
|
|
sessionId: string,
|
|
messages: Array<{ role: string; content: string; timestamp?: number }>,
|
|
workspacePath?: string,
|
|
distillationOpts?: {
|
|
enabled: boolean;
|
|
ageThresholdDays: number;
|
|
intervalDays: number;
|
|
archiveMode: DistillationArchiveMode;
|
|
brainPath: string;
|
|
},
|
|
): void {
|
|
if (!this.config.enabled) return;
|
|
|
|
const projectMemory = workspacePath
|
|
? this.getProjectMemory(workspacePath)
|
|
: null;
|
|
|
|
try {
|
|
this.extractor.extractFromSession(
|
|
sessionId,
|
|
messages,
|
|
this.longTerm,
|
|
this.episodic,
|
|
projectMemory,
|
|
workspacePath
|
|
);
|
|
} catch { /* memory extraction should never break the main flow */ }
|
|
|
|
// Persist long-term memory
|
|
this.longTerm.save();
|
|
|
|
// Auto-distillation — Distillation Loop 가 enabled 이고 interval 충족 시 stale
|
|
// episodes 를 LongTerm digest 로 승급. 세션 종료 시점이 자연스러움 — 사용자가
|
|
// 다음 세션 시작 전 한 번 cleanup.
|
|
if (distillationOpts?.enabled && shouldAutoDistill(distillationOpts.brainPath, distillationOpts.intervalDays)) {
|
|
try {
|
|
const report = distillStaleEpisodes(this.episodic, this.longTerm, distillationOpts.brainPath, {
|
|
ageThresholdDays: distillationOpts.ageThresholdDays,
|
|
archiveMode: distillationOpts.archiveMode,
|
|
});
|
|
recordDistillationRun(distillationOpts.brainPath, report);
|
|
} catch { /* distillation should never break session end */ }
|
|
}
|
|
}
|
|
|
|
// ─── Direct Access (for UI & advanced features) ───
|
|
|
|
public getLongTermMemory(): LongTermMemory {
|
|
return this.longTerm;
|
|
}
|
|
|
|
public getProceduralMemory(): ProceduralMemory {
|
|
return this.procedural;
|
|
}
|
|
|
|
public getEpisodicMemory(): EpisodicMemory {
|
|
return this.episodic;
|
|
}
|
|
|
|
public getProjectMemory(workspacePath: string): ProjectMemory {
|
|
if (!this.projectMemories.has(workspacePath)) {
|
|
this.projectMemories.set(workspacePath, new ProjectMemory(workspacePath));
|
|
}
|
|
return this.projectMemories.get(workspacePath)!;
|
|
}
|
|
|
|
// ─── Config ───
|
|
|
|
public updateConfig(partial: Partial<MemoryConfig>): void {
|
|
Object.assign(this.config, partial);
|
|
}
|
|
|
|
public getConfig(): MemoryConfig {
|
|
return { ...this.config };
|
|
}
|
|
}
|