sqlite_graphrag/commands/
read.rs1use crate::errors::AppError;
4use crate::i18n::errors_msg;
5use crate::output;
6use crate::paths::AppPaths;
7use crate::storage::connection::open_ro;
8use crate::storage::memories;
9use serde::Serialize;
10
11#[derive(clap::Args)]
12#[command(after_long_help = "EXAMPLES:\n \
13 # Read a memory by name (positional)\n \
14 sqlite-graphrag read onboarding\n\n \
15 # Read using the named flag form\n \
16 sqlite-graphrag read --name onboarding\n\n \
17 # Read from a specific namespace\n \
18 sqlite-graphrag read onboarding --namespace my-project")]
19pub struct ReadArgs {
20 #[arg(
22 value_name = "NAME",
23 conflicts_with = "name",
24 help = "Memory name (kebab-case slug); alternative to --name"
25 )]
26 pub name_positional: Option<String>,
27 #[arg(long)]
29 pub name: Option<String>,
30 #[arg(long, default_value = "global")]
31 pub namespace: Option<String>,
32 #[arg(long, hide = true, help = "No-op; JSON is always emitted on stdout")]
33 pub json: bool,
34 #[arg(long, env = "SQLITE_GRAPHRAG_DB_PATH")]
35 pub db: Option<String>,
36}
37
38#[derive(Serialize)]
39struct ReadResponse {
40 id: i64,
42 memory_id: i64,
44 namespace: String,
45 name: String,
46 #[serde(rename = "type")]
48 type_alias: String,
49 memory_type: String,
50 description: String,
51 body: String,
52 body_hash: String,
53 session_id: Option<String>,
54 source: String,
55 metadata: serde_json::Value,
56 version: i64,
58 created_at: i64,
59 created_at_iso: String,
61 updated_at: i64,
62 updated_at_iso: String,
64 elapsed_ms: u64,
66}
67
68fn epoch_to_iso(epoch: i64) -> String {
69 crate::tz::epoch_to_iso(epoch)
70}
71
72pub fn run(args: ReadArgs) -> Result<(), AppError> {
73 let start = std::time::Instant::now();
74 let name = args.name_positional.or(args.name).ok_or_else(|| {
76 AppError::Validation("name required: pass as positional argument or via --name".to_string())
77 })?;
78 let namespace = crate::namespace::resolve_namespace(args.namespace.as_deref())?;
79 let paths = AppPaths::resolve(args.db.as_deref())?;
80 crate::storage::connection::ensure_db_ready(&paths)?;
81 let conn = open_ro(&paths.db)?;
82
83 match memories::read_by_name(&conn, &namespace, &name)? {
84 Some(row) => {
85 let version: i64 = conn
87 .query_row(
88 "SELECT COALESCE(MAX(version), 1) FROM memory_versions WHERE memory_id=?1",
89 rusqlite::params![row.id],
90 |r| r.get(0),
91 )
92 .unwrap_or(1);
93
94 let response = ReadResponse {
95 id: row.id,
96 memory_id: row.id,
97 namespace: row.namespace,
98 name: row.name,
99 type_alias: row.memory_type.clone(),
100 memory_type: row.memory_type,
101 description: row.description,
102 body: row.body,
103 body_hash: row.body_hash,
104 session_id: row.session_id,
105 source: row.source,
106 metadata: serde_json::from_str::<serde_json::Value>(&row.metadata)
107 .unwrap_or(serde_json::Value::Null),
108 version,
109 created_at: row.created_at,
110 created_at_iso: epoch_to_iso(row.created_at),
111 updated_at: row.updated_at,
112 updated_at_iso: epoch_to_iso(row.updated_at),
113 elapsed_ms: start.elapsed().as_millis() as u64,
114 };
115 output::emit_json(&response)?;
116 }
117 None => {
118 return Err(AppError::NotFound(errors_msg::memory_not_found(
119 &name, &namespace,
120 )))
121 }
122 }
123
124 Ok(())
125}
126
127#[cfg(test)]
128mod tests {
129 use super::*;
130
131 #[test]
132 fn epoch_to_iso_converts_zero_to_unix_epoch() {
133 let result = epoch_to_iso(0);
134 assert!(
135 result.starts_with("1970-01-01T00:00:00"),
136 "epoch 0 must map to 1970-01-01T00:00:00, got: {result}"
137 );
138 }
139
140 #[test]
141 fn epoch_to_iso_converts_known_timestamp() {
142 let result = epoch_to_iso(1_705_320_000);
143 assert!(
144 result.starts_with("2024-01-15"),
145 "timestamp 1705320000 must map to 2024-01-15, got: {result}"
146 );
147 }
148
149 #[test]
150 fn epoch_to_iso_returns_fallback_for_invalid_negative_epoch() {
151 let result = epoch_to_iso(i64::MIN);
152 assert!(
153 !result.is_empty(),
154 "must return a non-empty string even for invalid epoch"
155 );
156 }
157
158 #[test]
159 fn read_response_serializes_id_and_memory_id_aliases() {
160 let resp = ReadResponse {
161 id: 42,
162 memory_id: 42,
163 namespace: "global".to_string(),
164 name: "my-mem".to_string(),
165 type_alias: "fact".to_string(),
166 memory_type: "fact".to_string(),
167 description: "desc".to_string(),
168 body: "body".to_string(),
169 body_hash: "abc123".to_string(),
170 session_id: None,
171 source: "agent".to_string(),
172 metadata: serde_json::json!({}),
173 version: 1,
174 created_at: 1_705_320_000,
175 created_at_iso: "2024-01-15T12:00:00Z".to_string(),
176 updated_at: 1_705_320_000,
177 updated_at_iso: "2024-01-15T12:00:00Z".to_string(),
178 elapsed_ms: 5,
179 };
180
181 let json = serde_json::to_value(&resp).expect("serialization failed");
182 assert_eq!(json["id"], 42);
183 assert_eq!(json["memory_id"], 42);
184 assert_eq!(json["type"], "fact");
185 assert_eq!(json["memory_type"], "fact");
186 assert_eq!(json["elapsed_ms"], 5u64);
187 assert!(
188 json["session_id"].is_null(),
189 "session_id None must serialize as null"
190 );
191 assert!(
193 json["metadata"].is_object(),
194 "metadata must be a JSON object"
195 );
196 }
197
198 #[test]
199 fn read_response_session_id_some_serializes_string() {
200 let resp = ReadResponse {
201 id: 1,
202 memory_id: 1,
203 namespace: "global".to_string(),
204 name: "mem".to_string(),
205 type_alias: "skill".to_string(),
206 memory_type: "skill".to_string(),
207 description: "d".to_string(),
208 body: "b".to_string(),
209 body_hash: "h".to_string(),
210 session_id: Some("sess-123".to_string()),
211 source: "agent".to_string(),
212 metadata: serde_json::json!({}),
213 version: 2,
214 created_at: 0,
215 created_at_iso: "1970-01-01T00:00:00Z".to_string(),
216 updated_at: 0,
217 updated_at_iso: "1970-01-01T00:00:00Z".to_string(),
218 elapsed_ms: 0,
219 };
220
221 let json = serde_json::to_value(&resp).expect("serialization failed");
222 assert_eq!(json["session_id"], "sess-123");
223 }
224
225 #[test]
226 fn read_response_elapsed_ms_is_present() {
227 let resp = ReadResponse {
228 id: 7,
229 memory_id: 7,
230 namespace: "ns".to_string(),
231 name: "n".to_string(),
232 type_alias: "procedure".to_string(),
233 memory_type: "procedure".to_string(),
234 description: "d".to_string(),
235 body: "b".to_string(),
236 body_hash: "h".to_string(),
237 session_id: None,
238 source: "agent".to_string(),
239 metadata: serde_json::json!({}),
240 version: 3,
241 created_at: 1000,
242 created_at_iso: "1970-01-01T00:16:40Z".to_string(),
243 updated_at: 2000,
244 updated_at_iso: "1970-01-01T00:33:20Z".to_string(),
245 elapsed_ms: 123,
246 };
247
248 let json = serde_json::to_value(&resp).expect("serialization failed");
249 assert_eq!(json["elapsed_ms"], 123u64);
250 assert!(json["created_at_iso"].is_string());
251 assert!(json["updated_at_iso"].is_string());
252 }
253
254 #[test]
255 fn read_response_metadata_object_not_escaped_string() {
256 let resp = ReadResponse {
258 id: 3,
259 memory_id: 3,
260 namespace: "ns".to_string(),
261 name: "meta-test".to_string(),
262 type_alias: "fact".to_string(),
263 memory_type: "fact".to_string(),
264 description: "d".to_string(),
265 body: "b".to_string(),
266 body_hash: "h".to_string(),
267 session_id: None,
268 source: "agent".to_string(),
269 metadata: serde_json::json!({"key": "value", "number": 42}),
270 version: 1,
271 created_at: 0,
272 created_at_iso: "1970-01-01T00:00:00Z".to_string(),
273 updated_at: 0,
274 updated_at_iso: "1970-01-01T00:00:00Z".to_string(),
275 elapsed_ms: 1,
276 };
277
278 let json = serde_json::to_value(&resp).expect("serialization failed");
279 assert!(json["metadata"].is_object());
281 assert_eq!(json["metadata"]["key"], "value");
282 assert_eq!(json["metadata"]["number"], 42);
283 }
284
285 #[test]
286 fn read_response_metadata_fallback_to_null_for_invalid_json() {
287 let raw = "invalid-json{{{";
289 let parsed =
290 serde_json::from_str::<serde_json::Value>(raw).unwrap_or(serde_json::Value::Null);
291 assert!(parsed.is_null());
292 }
293}