Skip to content

Commit

Permalink
Merge pull request #4576 from driftluo/add-limit-to-get-cells
Browse files Browse the repository at this point in the history
fix: add limit to get cells
  • Loading branch information
quake authored Aug 13, 2024
2 parents 114205b + 33e8ae8 commit 3c827e9
Show file tree
Hide file tree
Showing 14 changed files with 131 additions and 24 deletions.
6 changes: 6 additions & 0 deletions resource/ckb.toml
Original file line number Diff line number Diff line change
Expand Up @@ -212,6 +212,12 @@ block_uncles_cache_size = 30
# cell_filter = "let script = output.type;script!=() && script.code_hash == \"0x00000000000000000000000000000000000000000000000000545950455f4944\""
# # The initial tip can be set higher than the current indexer tip as the starting height for indexing.
# init_tip_hash = "0x8fbd0ec887159d2814cee475911600e3589849670f5ee1ed9798b38fdeef4e44"
# By default, there is no limitation on the size of indexer request
# However, because serde json serialization consumes too much memory(10x),
# it may cause the physical machine to become unresponsive.
# We recommend a consumption limit of 2g, which is 400 as the limit,
# which is a safer approach
# request_limit = 400
#
# # CKB rich-indexer has its unique configuration.
# [indexer_v2.rich_indexer]
Expand Down
4 changes: 4 additions & 0 deletions util/app-config/src/configs/indexer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,9 @@ pub struct IndexerConfig {
/// The init tip block hash
#[serde(default)]
pub init_tip_hash: Option<H256>,
/// limit of indexer reqeust
#[serde(default)]
pub request_limit: Option<usize>,
/// Rich indexer config options
#[serde(default)]
pub rich_indexer: RichIndexerConfig,
Expand All @@ -56,6 +59,7 @@ impl Default for IndexerConfig {
db_background_jobs: None,
db_keep_log_file_num: None,
init_tip_hash: None,
request_limit: None,
rich_indexer: RichIndexerConfig::default(),
}
}
Expand Down
2 changes: 2 additions & 0 deletions util/gen-types/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,10 @@ pub use molecule::bytes;

cfg_if::cfg_if! {
if #[cfg(feature = "std")] {
#[allow(unused_imports)]
use std::{vec, borrow};
} else {
#[allow(unused_imports)]
use alloc::{vec, borrow};
}
}
64 changes: 61 additions & 3 deletions util/indexer/src/service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ use rocksdb::{prelude::*, Direction, IteratorMode};
use std::convert::TryInto;
use std::num::NonZeroUsize;
use std::sync::{Arc, RwLock};
use std::usize;

pub(crate) const SUBSCRIBER_NAME: &str = "Indexer";
const DEFAULT_LOG_KEEP_NUM: usize = 1;
Expand All @@ -31,6 +32,7 @@ pub struct IndexerService {
sync: IndexerSyncService,
block_filter: Option<String>,
cell_filter: Option<String>,
request_limit: usize,
}

impl IndexerService {
Expand All @@ -56,6 +58,7 @@ impl IndexerService {
sync,
block_filter: config.block_filter.clone(),
cell_filter: config.cell_filter.clone(),
request_limit: config.request_limit.unwrap_or(usize::MAX),
}
}

Expand All @@ -67,6 +70,7 @@ impl IndexerService {
IndexerHandle {
store: self.store.clone(),
pool: self.sync.pool(),
request_limit: self.request_limit,
}
}

Expand Down Expand Up @@ -124,6 +128,7 @@ impl IndexerService {
pub struct IndexerHandle {
pub(crate) store: RocksdbStore,
pub(crate) pool: Option<Arc<RwLock<Pool>>>,
request_limit: usize,
}

impl IndexerHandle {
Expand Down Expand Up @@ -168,6 +173,12 @@ impl IndexerHandle {
if limit == 0 {
return Err(Error::invalid_params("limit should be greater than 0"));
}
if limit > self.request_limit {
return Err(Error::invalid_params(format!(
"limit must be less than {}",
self.request_limit,
)));
}

let (prefix, from_key, direction, skip) = build_query_options(
&search_key,
Expand Down Expand Up @@ -334,6 +345,12 @@ impl IndexerHandle {
if limit == 0 {
return Err(Error::invalid_params("limit should be greater than 0"));
}
if limit > self.request_limit {
return Err(Error::invalid_params(format!(
"limit must be less than {}",
self.request_limit,
)));
}

if search_key
.script_search_mode
Expand Down Expand Up @@ -926,12 +943,13 @@ mod tests {

#[test]
fn rpc() {
let store = new_store("rpc");
let store: RocksdbStore = new_store("rpc");
let pool = Arc::new(RwLock::new(Pool::default()));
let indexer = Indexer::new(store.clone(), 10, 100, None, CustomFilters::new(None, None));
let rpc = IndexerHandle {
store,
pool: Some(Arc::clone(&pool)),
request_limit: usize::MAX,
};

// setup test data
Expand Down Expand Up @@ -1516,7 +1534,11 @@ mod tests {
fn script_search_mode_rpc() {
let store = new_store("script_search_mode_rpc");
let indexer = Indexer::new(store.clone(), 10, 100, None, CustomFilters::new(None, None));
let rpc = IndexerHandle { store, pool: None };
let rpc = IndexerHandle {
store,
pool: None,
request_limit: usize::MAX,
};

// setup test data
let lock_script1 = ScriptBuilder::default()
Expand Down Expand Up @@ -1755,11 +1777,47 @@ mod tests {
);
}

#[test]
fn test_request_limit() {
let store = new_store("script_search_mode_rpc");
let rpc = IndexerHandle {
store,
pool: None,
request_limit: 2,
};

let lock_script1 = ScriptBuilder::default()
.code_hash(H256(rand::random()).pack())
.hash_type(ScriptHashType::Type.into())
.args(Bytes::from(b"lock_script1".to_vec()).pack())
.build();
let data = [0u8; 7];
let res = rpc.get_cells(
IndexerSearchKey {
script: lock_script1.into(),
filter: Some(IndexerSearchKeyFilter {
output_data: Some(JsonBytes::from_vec(data.to_vec())),
output_data_filter_mode: Some(IndexerSearchMode::Prefix),
..Default::default()
}),
..Default::default()
},
IndexerOrder::Asc,
1000.into(),
None,
);
assert!(res.is_err())
}

#[test]
fn output_data_filter_mode_rpc() {
let store = new_store("script_search_mode_rpc");
let indexer = Indexer::new(store.clone(), 10, 100, None, CustomFilters::new(None, None));
let rpc = IndexerHandle { store, pool: None };
let rpc = IndexerHandle {
store,
pool: None,
request_limit: usize::MAX,
};

// setup test data
let lock_script1 = ScriptBuilder::default()
Expand Down
2 changes: 1 addition & 1 deletion util/jsonrpc-types/src/blockchain.rs
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ impl From<packed::Script> for Script {
fn from(input: packed::Script) -> Script {
Script {
code_hash: input.code_hash().unpack(),
args: JsonBytes::from_bytes(input.args().unpack()),
args: JsonBytes::from_vec(input.args().unpack()),
hash_type: core::ScriptHashType::try_from(input.hash_type())
.expect("checked data")
.into(),
Expand Down
4 changes: 2 additions & 2 deletions util/jsonrpc-types/src/bytes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,13 +60,13 @@ impl JsonBytes {

impl From<packed::Bytes> for JsonBytes {
fn from(input: packed::Bytes) -> Self {
JsonBytes::from_bytes(input.raw_data())
JsonBytes::from_vec(input.raw_data().to_vec())
}
}

impl<'a> From<&'a packed::Bytes> for JsonBytes {
fn from(input: &'a packed::Bytes) -> Self {
JsonBytes::from_bytes(input.raw_data())
JsonBytes::from_vec(input.raw_data().to_vec())
}
}

Expand Down
4 changes: 4 additions & 0 deletions util/rich-indexer/src/indexer/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ use std::sync::{Arc, RwLock};
pub(crate) struct RichIndexer {
async_rich_indexer: AsyncRichIndexer,
async_runtime: Handle,
request_limit: usize,
}

impl RichIndexer {
Expand All @@ -46,10 +47,12 @@ impl RichIndexer {
pool: Option<Arc<RwLock<Pool>>>,
custom_filters: CustomFilters,
async_runtime: Handle,
request_limit: usize,
) -> Self {
Self {
async_rich_indexer: AsyncRichIndexer::new(store, pool, custom_filters),
async_runtime,
request_limit,
}
}
}
Expand All @@ -61,6 +64,7 @@ impl IndexerSync for RichIndexer {
self.async_rich_indexer.store.clone(),
self.async_rich_indexer.pool.clone(),
self.async_runtime.clone(),
self.request_limit,
);
indexer_handle
.get_indexer_tip()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,12 @@ impl AsyncRichIndexerHandle {
if limit == 0 {
return Err(Error::invalid_params("limit should be greater than 0"));
}
if limit as usize > self.request_limit {
return Err(Error::invalid_params(format!(
"limit must be less than {}",
self.request_limit,
)));
}

let mut param_index = 1;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,12 @@ impl AsyncRichIndexerHandle {
if limit == 0 {
return Err(Error::invalid_params("limit should be greater than 0"));
}
if limit as usize > self.request_limit {
return Err(Error::invalid_params(format!(
"limit must be less than {}",
self.request_limit,
)));
}
search_key.filter = convert_max_values_in_search_filter(&search_key.filter);

let mut tx = self
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,12 +23,17 @@ use std::sync::{Arc, RwLock};
pub struct AsyncRichIndexerHandle {
store: SQLXPool,
pool: Option<Arc<RwLock<Pool>>>,
request_limit: usize,
}

impl AsyncRichIndexerHandle {
/// Construct new AsyncRichIndexerHandle instance
pub fn new(store: SQLXPool, pool: Option<Arc<RwLock<Pool>>>) -> Self {
Self { store, pool }
pub fn new(store: SQLXPool, pool: Option<Arc<RwLock<Pool>>>, request_limit: usize) -> Self {
Self {
store,
pool,
request_limit,
}
}
}

Expand Down
9 changes: 7 additions & 2 deletions util/rich-indexer/src/indexer_handle/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,14 @@ pub struct RichIndexerHandle {

impl RichIndexerHandle {
/// Construct new RichIndexerHandle instance
pub fn new(store: SQLXPool, pool: Option<Arc<RwLock<Pool>>>, async_handle: Handle) -> Self {
pub fn new(
store: SQLXPool,
pool: Option<Arc<RwLock<Pool>>>,
async_handle: Handle,
request_limit: usize,
) -> Self {
Self {
async_handle: AsyncRichIndexerHandle::new(store, pool),
async_handle: AsyncRichIndexerHandle::new(store, pool, request_limit),
async_runtime: async_handle,
}
}
Expand Down
8 changes: 7 additions & 1 deletion util/rich-indexer/src/service.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
//!The rich-indexer service.

use std::usize;

use crate::indexer::RichIndexer;
use crate::store::SQLXPool;
use crate::{AsyncRichIndexerHandle, RichIndexerHandle};
Expand All @@ -19,6 +21,7 @@ pub struct RichIndexerService {
block_filter: Option<String>,
cell_filter: Option<String>,
async_handle: Handle,
request_limit: usize,
}

impl RichIndexerService {
Expand Down Expand Up @@ -47,6 +50,7 @@ impl RichIndexerService {
block_filter: config.block_filter.clone(),
cell_filter: config.cell_filter.clone(),
async_handle,
request_limit: config.request_limit.unwrap_or(usize::MAX),
}
}

Expand All @@ -56,6 +60,7 @@ impl RichIndexerService {
self.sync.pool(),
CustomFilters::new(self.block_filter.as_deref(), self.cell_filter.as_deref()),
self.async_handle.clone(),
self.request_limit,
)
}

Expand Down Expand Up @@ -83,6 +88,7 @@ impl RichIndexerService {
self.store.clone(),
self.sync.pool(),
self.async_handle.clone(),
self.request_limit,
)
}

Expand All @@ -91,6 +97,6 @@ impl RichIndexerService {
/// The returned handle can be used to get data from rich-indexer,
/// and can be cloned to allow moving the Handle to other threads.
pub fn async_handle(&self) -> AsyncRichIndexerHandle {
AsyncRichIndexerHandle::new(self.store.clone(), self.sync.pool())
AsyncRichIndexerHandle::new(self.store.clone(), self.sync.pool(), self.request_limit)
}
}
6 changes: 4 additions & 2 deletions util/rich-indexer/src/tests/insert.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
use std::usize;

use super::*;

use ckb_types::{
Expand Down Expand Up @@ -76,7 +78,7 @@ async fn with_custom_block_filter() {
None,
),
);
let indexer_handle = AsyncRichIndexerHandle::new(storage, None);
let indexer_handle = AsyncRichIndexerHandle::new(storage, None, usize::MAX);

let lock_script1 = ScriptBuilder::default()
.code_hash(H256(rand::random()).pack())
Expand Down Expand Up @@ -290,7 +292,7 @@ async fn with_custom_cell_filter() {
Some(r#"output.type?.args == "0x747970655f73637269707431""#),
),
);
let indexer_handle = AsyncRichIndexerHandle::new(storage, None);
let indexer_handle = AsyncRichIndexerHandle::new(storage, None, usize::MAX);

let lock_script1 = ScriptBuilder::default()
.code_hash(H256(rand::random()).pack())
Expand Down
Loading

0 comments on commit 3c827e9

Please sign in to comment.