- HashedEntry / EntryHash / compute_hash / build_chain 撤去、JSONL は 1 行 1 LogEntry - SessionOrigin.at_hash → at_turn_index (TurnEnd 由来) に置換 - Pod 側 SessionHead mutex を ArcSwap<SessionId> + AtomicUsize の SessionState に置換 - ensure_head_or_fork は store の entry count と writer の append tally で判定 - session-store から sha2 / hex 依存、pod から parking_lot 依存を削除
209 lines
5.7 KiB
Rust
209 lines
5.7 KiB
Rust
use llm_worker::WorkerResult;
|
|
use llm_worker::llm_client::types::{Item, RequestConfig};
|
|
use session_store::{FsStore, LogEntry, Store, TraceEntry, collect_state, new_session_id};
|
|
|
|
#[test]
|
|
fn round_trip_write_and_read() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
let entries = vec![
|
|
LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: Some("You are helpful.".into()),
|
|
config: RequestConfig::default().with_max_tokens(1024),
|
|
history: vec![],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
},
|
|
LogEntry::UserInput {
|
|
ts: 2000,
|
|
segments: vec![protocol::Segment::text("Hello")],
|
|
},
|
|
LogEntry::AssistantItem {
|
|
ts: 3000,
|
|
item: Item::assistant_message("Hi there!").into(),
|
|
},
|
|
LogEntry::TurnEnd {
|
|
ts: 3100,
|
|
turn_count: 1,
|
|
},
|
|
LogEntry::RunCompleted {
|
|
ts: 3200,
|
|
interrupted: false,
|
|
result: WorkerResult::Finished,
|
|
},
|
|
];
|
|
|
|
for entry in &entries {
|
|
store.append(id, entry).unwrap();
|
|
}
|
|
|
|
let read_back = store.read_all(id).unwrap();
|
|
assert_eq!(read_back.len(), entries.len());
|
|
|
|
let state = collect_state(&read_back);
|
|
assert_eq!(state.system_prompt.as_deref(), Some("You are helpful."));
|
|
assert_eq!(state.config.max_tokens, Some(1024));
|
|
assert_eq!(state.history.len(), 2);
|
|
assert_eq!(state.turn_count, 1);
|
|
assert!(!state.last_run_interrupted);
|
|
assert_eq!(state.entries_count, entries.len());
|
|
}
|
|
|
|
#[test]
|
|
fn create_session_writes_all_entries() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
let entries = [LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: None,
|
|
config: RequestConfig::default(),
|
|
history: vec![
|
|
Item::user_message("seed").into(),
|
|
Item::assistant_message("ok").into(),
|
|
],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
}];
|
|
|
|
store.create_session(id, &entries).unwrap();
|
|
let read_back = store.read_all(id).unwrap();
|
|
assert_eq!(read_back.len(), 1);
|
|
|
|
let state = collect_state(&read_back);
|
|
assert_eq!(state.history.len(), 2);
|
|
}
|
|
|
|
#[test]
|
|
fn list_sessions_returns_newest_first() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
|
|
let id1 = new_session_id();
|
|
// Small delay to ensure different UUID v7 timestamps
|
|
std::thread::sleep(std::time::Duration::from_millis(2));
|
|
let id2 = new_session_id();
|
|
|
|
let entry = LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: None,
|
|
config: RequestConfig::default(),
|
|
history: vec![],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
};
|
|
|
|
store.append(id1, &entry).unwrap();
|
|
store.append(id2, &entry).unwrap();
|
|
|
|
let sessions = store.list_sessions().unwrap();
|
|
assert_eq!(sessions.len(), 2);
|
|
assert_eq!(sessions[0], id2); // newest first
|
|
assert_eq!(sessions[1], id1);
|
|
}
|
|
|
|
#[test]
|
|
fn exists_returns_correct_state() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
assert!(!store.exists(id).unwrap());
|
|
|
|
store
|
|
.append(
|
|
id,
|
|
&LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: None,
|
|
config: RequestConfig::default(),
|
|
history: vec![],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
},
|
|
)
|
|
.unwrap();
|
|
|
|
assert!(store.exists(id).unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn not_found_error_for_missing_session() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
let result = store.read_all(id);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn trace_entries_in_separate_file() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
store
|
|
.append(
|
|
id,
|
|
&LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: None,
|
|
config: RequestConfig::default(),
|
|
history: vec![],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
},
|
|
)
|
|
.unwrap();
|
|
|
|
let trace = TraceEntry {
|
|
ts: 1500,
|
|
turn: 0,
|
|
event: llm_worker::llm_client::event::Event::Ping(
|
|
llm_worker::llm_client::event::PingEvent { timestamp: None },
|
|
),
|
|
};
|
|
store.append_trace(id, &trace).unwrap();
|
|
|
|
// Log should have 1 entry, unaffected by trace
|
|
let log = store.read_all(id).unwrap();
|
|
assert_eq!(log.len(), 1);
|
|
|
|
// Trace file should exist separately
|
|
let trace_path = dir.path().join(format!("{id}.trace.jsonl"));
|
|
assert!(trace_path.exists());
|
|
}
|
|
|
|
#[test]
|
|
fn read_entry_count_matches_append_tally() {
|
|
let dir = tempfile::tempdir().unwrap();
|
|
let store = FsStore::new(dir.path()).unwrap();
|
|
let id = new_session_id();
|
|
|
|
let entries = [
|
|
LogEntry::SessionStart {
|
|
ts: 1000,
|
|
system_prompt: None,
|
|
config: RequestConfig::default(),
|
|
history: vec![],
|
|
forked_from: None,
|
|
compacted_from: None,
|
|
},
|
|
LogEntry::UserInput {
|
|
ts: 2000,
|
|
segments: vec![protocol::Segment::text("Hello")],
|
|
},
|
|
];
|
|
|
|
for entry in &entries {
|
|
store.append(id, entry).unwrap();
|
|
}
|
|
|
|
assert_eq!(store.read_entry_count(id).unwrap(), entries.len());
|
|
}
|