sqlite_graphrag/commands/
read.rs1use crate::errors::AppError;
2use crate::i18n::erros;
3use crate::output;
4use crate::paths::AppPaths;
5use crate::storage::connection::open_ro;
6use crate::storage::memories;
7use serde::Serialize;
8
9#[derive(clap::Args)]
10pub struct ReadArgs {
11 #[arg(value_name = "NAME", conflicts_with = "name")]
13 pub name_positional: Option<String>,
14 #[arg(long)]
16 pub name: Option<String>,
17 #[arg(long, default_value = "global")]
18 pub namespace: Option<String>,
19 #[arg(long, help = "No-op; JSON is always emitted on stdout")]
20 pub json: bool,
21 #[arg(long, env = "SQLITE_GRAPHRAG_DB_PATH")]
22 pub db: Option<String>,
23}
24
25#[derive(Serialize)]
26struct ReadResponse {
27 id: i64,
29 memory_id: i64,
31 namespace: String,
32 name: String,
33 #[serde(rename = "type")]
35 type_alias: String,
36 memory_type: String,
37 description: String,
38 body: String,
39 body_hash: String,
40 session_id: Option<String>,
41 source: String,
42 metadata: serde_json::Value,
43 version: i64,
45 created_at: i64,
46 created_at_iso: String,
48 updated_at: i64,
49 updated_at_iso: String,
51 elapsed_ms: u64,
53}
54
55fn epoch_to_iso(epoch: i64) -> String {
56 crate::tz::epoch_para_iso(epoch)
57}
58
59pub fn run(args: ReadArgs) -> Result<(), AppError> {
60 let inicio = std::time::Instant::now();
61 let name = args.name_positional.or(args.name).ok_or_else(|| {
63 AppError::Validation("name required: pass as positional argument or via --name".to_string())
64 })?;
65 let namespace = crate::namespace::resolve_namespace(args.namespace.as_deref())?;
66 let paths = AppPaths::resolve(args.db.as_deref())?;
67 if !paths.db.exists() {
68 return Err(AppError::NotFound(
69 crate::i18n::erros::banco_nao_encontrado(&paths.db.display().to_string()),
70 ));
71 }
72 let conn = open_ro(&paths.db)?;
73
74 match memories::read_by_name(&conn, &namespace, &name)? {
75 Some(row) => {
76 let version: i64 = conn
78 .query_row(
79 "SELECT COALESCE(MAX(version), 1) FROM memory_versions WHERE memory_id=?1",
80 rusqlite::params![row.id],
81 |r| r.get(0),
82 )
83 .unwrap_or(1);
84
85 let response = ReadResponse {
86 id: row.id,
87 memory_id: row.id,
88 namespace: row.namespace,
89 name: row.name,
90 type_alias: row.memory_type.clone(),
91 memory_type: row.memory_type,
92 description: row.description,
93 body: row.body,
94 body_hash: row.body_hash,
95 session_id: row.session_id,
96 source: row.source,
97 metadata: serde_json::from_str::<serde_json::Value>(&row.metadata)
98 .unwrap_or(serde_json::Value::Null),
99 version,
100 created_at: row.created_at,
101 created_at_iso: epoch_to_iso(row.created_at),
102 updated_at: row.updated_at,
103 updated_at_iso: epoch_to_iso(row.updated_at),
104 elapsed_ms: inicio.elapsed().as_millis() as u64,
105 };
106 output::emit_json(&response)?;
107 }
108 None => {
109 return Err(AppError::NotFound(erros::memoria_nao_encontrada(
110 &name, &namespace,
111 )))
112 }
113 }
114
115 Ok(())
116}
117
118#[cfg(test)]
119mod testes {
120 use super::*;
121
122 #[test]
123 fn epoch_to_iso_converte_zero_para_epoch_unix() {
124 let resultado = epoch_to_iso(0);
125 assert!(
126 resultado.starts_with("1970-01-01T00:00:00"),
127 "epoch 0 deve mapear para 1970-01-01T00:00:00, obtido: {resultado}"
128 );
129 }
130
131 #[test]
132 fn epoch_to_iso_converte_timestamp_conhecido() {
133 let resultado = epoch_to_iso(1_705_320_000);
134 assert!(
135 resultado.starts_with("2024-01-15"),
136 "timestamp 1705320000 deve mapear para 2024-01-15, obtido: {resultado}"
137 );
138 }
139
140 #[test]
141 fn epoch_to_iso_retorna_fallback_para_epoch_negativo_invalido() {
142 let resultado = epoch_to_iso(i64::MIN);
143 assert!(
144 !resultado.is_empty(),
145 "deve retornar string não vazia mesmo para epoch inválido"
146 );
147 }
148
149 #[test]
150 fn read_response_serializa_aliases_id_e_memory_id() {
151 let resp = ReadResponse {
152 id: 42,
153 memory_id: 42,
154 namespace: "global".to_string(),
155 name: "minha-mem".to_string(),
156 type_alias: "fact".to_string(),
157 memory_type: "fact".to_string(),
158 description: "desc".to_string(),
159 body: "corpo".to_string(),
160 body_hash: "abc123".to_string(),
161 session_id: None,
162 source: "agent".to_string(),
163 metadata: serde_json::json!({}),
164 version: 1,
165 created_at: 1_705_320_000,
166 created_at_iso: "2024-01-15T12:00:00Z".to_string(),
167 updated_at: 1_705_320_000,
168 updated_at_iso: "2024-01-15T12:00:00Z".to_string(),
169 elapsed_ms: 5,
170 };
171
172 let json = serde_json::to_value(&resp).expect("serialização falhou");
173 assert_eq!(json["id"], 42);
174 assert_eq!(json["memory_id"], 42);
175 assert_eq!(json["type"], "fact");
176 assert_eq!(json["memory_type"], "fact");
177 assert_eq!(json["elapsed_ms"], 5u64);
178 assert!(
179 json["session_id"].is_null(),
180 "session_id None deve serializar como null"
181 );
182 assert!(
184 json["metadata"].is_object(),
185 "metadata deve ser um objeto JSON"
186 );
187 }
188
189 #[test]
190 fn read_response_session_id_some_serializa_string() {
191 let resp = ReadResponse {
192 id: 1,
193 memory_id: 1,
194 namespace: "global".to_string(),
195 name: "mem".to_string(),
196 type_alias: "skill".to_string(),
197 memory_type: "skill".to_string(),
198 description: "d".to_string(),
199 body: "b".to_string(),
200 body_hash: "h".to_string(),
201 session_id: Some("sess-123".to_string()),
202 source: "agent".to_string(),
203 metadata: serde_json::json!({}),
204 version: 2,
205 created_at: 0,
206 created_at_iso: "1970-01-01T00:00:00Z".to_string(),
207 updated_at: 0,
208 updated_at_iso: "1970-01-01T00:00:00Z".to_string(),
209 elapsed_ms: 0,
210 };
211
212 let json = serde_json::to_value(&resp).expect("serialização falhou");
213 assert_eq!(json["session_id"], "sess-123");
214 }
215
216 #[test]
217 fn read_response_elapsed_ms_esta_presente() {
218 let resp = ReadResponse {
219 id: 7,
220 memory_id: 7,
221 namespace: "ns".to_string(),
222 name: "n".to_string(),
223 type_alias: "procedure".to_string(),
224 memory_type: "procedure".to_string(),
225 description: "d".to_string(),
226 body: "b".to_string(),
227 body_hash: "h".to_string(),
228 session_id: None,
229 source: "agent".to_string(),
230 metadata: serde_json::json!({}),
231 version: 3,
232 created_at: 1000,
233 created_at_iso: "1970-01-01T00:16:40Z".to_string(),
234 updated_at: 2000,
235 updated_at_iso: "1970-01-01T00:33:20Z".to_string(),
236 elapsed_ms: 123,
237 };
238
239 let json = serde_json::to_value(&resp).expect("serialização falhou");
240 assert_eq!(json["elapsed_ms"], 123u64);
241 assert!(json["created_at_iso"].is_string());
242 assert!(json["updated_at_iso"].is_string());
243 }
244
245 #[test]
246 fn read_response_metadata_object_nao_string_escapada() {
247 let resp = ReadResponse {
249 id: 3,
250 memory_id: 3,
251 namespace: "ns".to_string(),
252 name: "meta-test".to_string(),
253 type_alias: "fact".to_string(),
254 memory_type: "fact".to_string(),
255 description: "d".to_string(),
256 body: "b".to_string(),
257 body_hash: "h".to_string(),
258 session_id: None,
259 source: "agent".to_string(),
260 metadata: serde_json::json!({"chave": "valor", "numero": 42}),
261 version: 1,
262 created_at: 0,
263 created_at_iso: "1970-01-01T00:00:00Z".to_string(),
264 updated_at: 0,
265 updated_at_iso: "1970-01-01T00:00:00Z".to_string(),
266 elapsed_ms: 1,
267 };
268
269 let json = serde_json::to_value(&resp).expect("serialização falhou");
270 assert!(json["metadata"].is_object());
272 assert_eq!(json["metadata"]["chave"], "valor");
273 assert_eq!(json["metadata"]["numero"], 42);
274 }
275
276 #[test]
277 fn read_response_metadata_fallback_para_null_em_json_invalido() {
278 let raw = "json-invalido{{{";
280 let parsed =
281 serde_json::from_str::<serde_json::Value>(raw).unwrap_or(serde_json::Value::Null);
282 assert!(parsed.is_null());
283 }
284}