Skip to content

Commit 2e27ee2

Browse files
authored
Feat/get (#602)
* feat: add doc store * feat: add get doc by id * test: add test case
1 parent bb44ae3 commit 2e27ee2

File tree

9 files changed

+123
-13
lines changed

9 files changed

+123
-13
lines changed

sdk/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "db3.js",
3-
"version": "0.4.1",
3+
"version": "0.4.6",
44
"description": "DB3 Network Javascript API",
55
"author": "dbpunk labs",
66
"keywords": [

sdk/src/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ export {
2222
createFromExternal,
2323
} from './account/db3_account'
2424
export type { Client, ReadClient } from './client/types'
25+
export type { DocumentData, DocumentEntry } from './client/base'
2526
export type {
2627
Database,
2728
Collection,
@@ -30,7 +31,7 @@ export type {
3031
MutationResult,
3132
QueryResult,
3233
} from './store/types'
33-
export { addDoc, updateDoc, deleteDoc, queryDoc } from './store/document_v2'
34+
export { addDoc, updateDoc, deleteDoc, queryDoc, getDoc } from './store/document_v2'
3435

3536
export { SystemConfig, SystemStatus, Version } from './proto/db3_base'
3637
export {

sdk/src/provider/indexer_provider.ts

+15
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ import {
2525
RunQueryRequest,
2626
GetContractSyncStatusRequest,
2727
GetCollectionOfDatabaseRequest,
28+
GetDocRequest,
2829
} from '../proto/db3_indexer'
2930
import { SetupRequest, GetSystemStatusRequest } from '../proto/db3_system'
3031
import { Query } from '../proto/db3_database_v2'
@@ -61,6 +62,20 @@ export class IndexerProvider {
6162
throw new DB3Error(e as RpcError)
6263
}
6364
}
65+
66+
async getDoc(db: string, colName: string, id: string) {
67+
const request: GetDocRequest = {
68+
dbAddr: db,
69+
colName,
70+
id,
71+
}
72+
try {
73+
const { response } = await this.client.getDoc(request)
74+
return response
75+
} catch (e) {
76+
throw new DB3Error(e as RpcError)
77+
}
78+
}
6479
async setup(signature: string, payload: string) {
6580
try {
6681
const request: SetupRequest = {

sdk/src/store/document_v2.ts

+30
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ async function runQueryInternal<T>(col: Collection, query: Query) {
4040
id: doc.id,
4141
} as DocumentEntry<T>
4242
})
43+
4344
return {
4445
docs: entries,
4546
collection: col,
@@ -115,6 +116,35 @@ export async function queryDoc<T = DocumentData>(
115116
}
116117
}
117118

119+
/**
120+
*
121+
* This function gets a document from the database by its ID.
122+
*
123+
* ```ts
124+
* const doc = await getDoc(collection, "10")
125+
* const id = doc.id
126+
* const content = doc.doc
127+
* ```
128+
* @param col - the instance of collection
129+
* @param id - the id of document
130+
* @returns the {@link DocumentEntry} if the document was found. Otherwise, raises an error.
131+
**/
132+
export async function getDoc<T = DocumentData>(col: Collection, id: string) {
133+
const response = await col.db.client.indexer.getDoc(
134+
col.db.addr,
135+
col.name,
136+
id
137+
)
138+
if (response.document) {
139+
return {
140+
doc: JSON.parse(response.document.doc) as T,
141+
id: response.document.id,
142+
} as DocumentEntry<T>
143+
} else {
144+
throw new Error('no document was found with id ' + id)
145+
}
146+
}
147+
118148
export async function deleteDoc(col: Collection, ids: string[]) {
119149
const documentMutation: DocumentMutation = {
120150
collectionName: col.name,

sdk/tests/client_v2.test.ts

+17
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ import {
3636
deleteDoc,
3737
updateDoc,
3838
queryDoc,
39+
getDoc
3940
} from '../src/store/document_v2'
4041
import {
4142
createFromPrivateKey,
@@ -370,6 +371,21 @@ describe('test db3.js client module', () => {
370371
age: 1,
371372
})
372373
await new Promise((r) => setTimeout(r, 1000))
374+
{
375+
const doc = await getDoc(collection, doc2Ret.id)
376+
expect(doc).toBeDefined()
377+
expect(doc.id).toBe(doc2Ret.id)
378+
expect(doc.doc.city).toBe('beijing2')
379+
expect(doc.doc.author).toBe('imotai1')
380+
expect(doc.doc.age).toBe(1)
381+
}
382+
{
383+
try {
384+
const doc = await getDoc(collection, 1000000000000)
385+
except(1).toBe(0)
386+
} catch(e) {}
387+
}
388+
373389
{
374390
const queryStr = '/[city = beijing]'
375391
const resultSet = await queryDoc<Profile>(
@@ -465,6 +481,7 @@ describe('test db3.js client module', () => {
465481
age: 10,
466482
})
467483
await new Promise((r) => setTimeout(r, 2000))
484+
468485
{
469486
const queryStr = '/[city = beijing]'
470487
const resultSet = await queryDoc<Profile>(

src/node/src/indexer_impl.rs

+17-2
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,8 @@ use db3_event::event_processor::EventProcessorConfig;
2323
use db3_proto::db3_indexer_proto::indexer_node_server::IndexerNode;
2424
use db3_proto::db3_indexer_proto::{
2525
ContractSyncStatus, GetCollectionOfDatabaseRequest, GetCollectionOfDatabaseResponse,
26-
GetContractSyncStatusRequest, GetContractSyncStatusResponse, RunQueryRequest, RunQueryResponse,
26+
GetContractSyncStatusRequest, GetContractSyncStatusResponse, GetDocRequest, GetDocResponse,
27+
RunQueryRequest, RunQueryResponse,
2728
};
2829
use db3_proto::db3_mutation_v2_proto::MutationAction;
2930
use db3_proto::db3_storage_proto::block_response::MutationWrapper;
@@ -367,6 +368,21 @@ impl IndexerNode for IndexerNodeImpl {
367368
}))
368369
}
369370

371+
async fn get_doc(
372+
&self,
373+
request: Request<GetDocRequest>,
374+
) -> std::result::Result<Response<GetDocResponse>, Status> {
375+
let r = request.into_inner();
376+
let addr = DB3Address::from_hex(r.db_addr.as_str()).map_err(|e| {
377+
Status::invalid_argument(format!("fail to parse the db address for {e}"))
378+
})?;
379+
let document = self
380+
.db_store
381+
.get_doc(&addr, r.col_name.as_str(), r.id)
382+
.map_err(|e| Status::internal(format!("{e}")))?;
383+
Ok(Response::new(GetDocResponse { document }))
384+
}
385+
370386
async fn run_query(
371387
&self,
372388
request: Request<RunQueryRequest>,
@@ -376,7 +392,6 @@ impl IndexerNode for IndexerNodeImpl {
376392
Status::invalid_argument(format!("fail to parse the db address for {e}"))
377393
})?;
378394
if let Some(q) = &r.query {
379-
info!("query str {} q {:?}", q.query_str, q);
380395
let (documents, count) = self
381396
.db_store
382397
.query_docs(&addr, r.col_name.as_str(), q)

src/proto/proto/db3_indexer.proto

+11
Original file line numberDiff line numberDiff line change
@@ -61,9 +61,20 @@ message GetContractSyncStatusResponse {
6161

6262
message GetContractSyncStatusRequest {}
6363

64+
message GetDocRequest {
65+
string db_addr = 1;
66+
string col_name = 2;
67+
int64 id = 3;
68+
}
69+
70+
message GetDocResponse {
71+
db3_database_v2_proto.Document document = 1;
72+
}
73+
6474
service IndexerNode {
6575
rpc GetContractSyncStatus(GetContractSyncStatusRequest) returns (GetContractSyncStatusResponse) {}
6676
rpc GetCollectionOfDatabase(GetCollectionOfDatabaseRequest) returns (GetCollectionOfDatabaseResponse) {}
6777
// method for query document
6878
rpc RunQuery(RunQueryRequest) returns (RunQueryResponse) {}
79+
rpc GetDoc(GetDocRequest) returns (GetDocResponse) {}
6980
}

src/storage/src/db_store_v2.rs

+24
Original file line numberDiff line numberDiff line change
@@ -845,6 +845,30 @@ impl DBStoreV2 {
845845
Ok(())
846846
}
847847

848+
pub fn get_doc(
849+
&self,
850+
db_addr: &DB3Address,
851+
col_name: &str,
852+
doc_id: i64,
853+
) -> Result<Option<Document>> {
854+
if !self.is_db_collection_exist(db_addr, col_name)? {
855+
return Err(DB3Error::CollectionNotFound(
856+
col_name.to_string(),
857+
db_addr.to_hex(),
858+
));
859+
}
860+
if self.config.enable_doc_store {
861+
let doc = self.doc_store.get_doc(db_addr, col_name, doc_id)?;
862+
if let Some(d) = doc {
863+
Ok(Some(Document { id: doc_id, doc: d }))
864+
} else {
865+
Ok(None)
866+
}
867+
} else {
868+
Ok(None)
869+
}
870+
}
871+
848872
pub fn get_doc_key_from_doc_id(&self, doc_id: i64) -> Result<Vec<u8>> {
849873
let doc_owner_store_cf_handle = self
850874
.se

src/storage/src/doc_store.rs

+6-9
Original file line numberDiff line numberDiff line change
@@ -237,18 +237,15 @@ impl DocStore {
237237
}
238238
}
239239

240-
pub fn get_doc(&self, db_addr: &DB3Address, col_name: &str, id: i64) -> Result<String> {
240+
pub fn get_doc(&self, db_addr: &DB3Address, col_name: &str, id: i64) -> Result<Option<String>> {
241241
let db_opt = self.get_db_ref(db_addr);
242242
if let Some(db) = db_opt {
243243
let opt = db
244244
.get::<String>(col_name, id)
245245
.map_err(|e| DB3Error::WriteStoreError(format!("{e}")))?;
246-
Ok(opt)
246+
Ok(Some(opt))
247247
} else {
248-
Err(DB3Error::WriteStoreError(format!(
249-
"no database found with addr {}",
250-
db_addr.to_hex()
251-
)))
248+
Ok(None)
252249
}
253250
}
254251

@@ -349,7 +346,7 @@ mod tests {
349346
fn doc_get_test() {
350347
let (doc_store, id) = prepare_the_dataset();
351348
let doc_str = r#"{"f2":"f2", "f1":"f1"}"#;
352-
if let Ok(value) = doc_store.get_doc(&DB3Address::ZERO, "col1", id) {
349+
if let Ok(Some(value)) = doc_store.get_doc(&DB3Address::ZERO, "col1", id) {
353350
let value: serde_json::Value = serde_json::from_str(value.as_str()).unwrap();
354351
let right: serde_json::Value = serde_json::from_str(doc_str).unwrap();
355352
assert_eq!(value, right);
@@ -417,7 +414,7 @@ mod tests {
417414
fn doc_store_smoke_test() {
418415
let (doc_store, id) = prepare_the_dataset();
419416
let db_id = DB3Address::ZERO;
420-
if let Ok(value) = doc_store.get_doc(&db_id, "col1", id) {
417+
if let Ok(Some(value)) = doc_store.get_doc(&db_id, "col1", id) {
421418
println!("{}", value.as_str());
422419
let value: serde_json::Value = serde_json::from_str(value.as_str()).unwrap();
423420
assert_eq!(value["f1"].as_str(), Some("f1"));
@@ -475,7 +472,7 @@ mod tests {
475472
assert!(false);
476473
}
477474

478-
if let Ok(value) = doc_store.get_doc(&db_id, "col1", id) {
475+
if let Ok(Some(value)) = doc_store.get_doc(&db_id, "col1", id) {
479476
let value: serde_json::Value = serde_json::from_str(value.as_str()).unwrap();
480477
assert_eq!(value["test"].as_str(), Some("v2"));
481478
} else {

0 commit comments

Comments
 (0)