- Published on
Code Breakdown @ Substrate Node
- Authors
- Name
- Tin Chung
- twitter@chungquantin
Language: Vietnamese
Cấp độ: Intermediate
Code Breakdown @ Substrate Node
Đôi lời về Code Breakdown - Đây là series mà TheLowLevelers tách và phân tích mã nguồn của các dự án mã nguồn mở nhằm để đem đến các bạn một góc nhìn chi tiết nhất về những gì xảy ra trong các dòng code.
Language: Vietnamese
Bài viết liên quan
- Polkadot SDK - Substrate
- Substrate Documentation - Architecture and Rust libraries
- Build a substrate node from scratch
Khái quát hạ tầng của Blockchain
Blockchain là mạng lưới tập hợp các máy tính con khác nhau liên kết và giao tiếp, chia sẻ thông tin thông qua mạng lưới. Xét trên tư duy nguyên bản, thì Blockchain bản chất là mạng Internet mà chúng ta đang sử dụng hằng ngày hiện tại tuy nhiên với những cái tiến đột phá hơn về mặt bảo mật và kinh tế số. Ở một khía cạnh khác, bạn có thể xem blockchain là state machine phi tập trung.
Blockchain cung cấp khả năng cho người dùng submit giao dịch. Các giao dịch không được thực thi sẽ được lưu trữ ở transaction pool và blockchain có logic để xử lý các giao dịch này, kiểm tra và cập nhật kho lưu trữ dựa trên các trạng thái được thay đổi bởi các giao dịch này. Để các giao dịch được xác thức và chấp thuận bởi toàn bộ mạng lưới, cơ chế đồng thuận của các node trong mạng lưới là yêu tố bắt buộc.
Các máy tính tham gia trở thành một phần của mạng lưới blockchain được gọi là các node. Và trong Polkadot, các node này sẽ được xây dựng bằng Polkadot SDK - Substrate.của bộ công cụ phát triển Polkadot SDK
Kiến trúc của Substrate
substrate-node-template
Cấu trúc của Substrate node bao gồm hai thành phần chính:
- Core client and Outer node services: Đóng vai trò xử lý các hoạt động của mạng lưới như là tìm kiếm các node trong cùng mạng lưới (peer node), quản lý các yêu cầu giao dịch (transaction requests), đạt được sự đồng thuận giữa các peers và phản hồi tới các RPC calls
Outer node đóng vai trò tìm kiếm các node ngang hàng (peer discovery), transaction pooling, đồng thuận (consensus)...đọc thêm
Nhìn qua mã nguồn của substrate-node-template
, ta có thể thấy được Substrate bao gồm 3 package chín
node/
: Mã nguồn chính của một substrate node, bao gồm client và các outer node servic Node bao gồm các khả năng Networking được phát triển trênlibp2p
, cung cấp đồng thuận và là một RPC server. Node đóng vai trò tham gia vào mạng lưới blockchain và và trung gian cho giao tiếp giữa phái Client và Runtime.runtime/
: Mã nguồn chính của runtime cho Substrate Node, chứa các logic chính của một blockchain bao gồm các vai trò như xác thực node và thực thi các thay đổi trạng thái của mạng lưới.
Runtime cho Substrate Node bao gồm tất cả các business logic cho việc thực thi giao dịch, lưu trữ các trạng thái thay đổi và tương tác với các outer node.
pallet/
: Runtime được cấu thành bởi các FRAME pallets, bạn có thể đọc thêm tại đây
FRAME pallets là các module và thư viện hỗ trợ mà bạn có thể sử dụng để phát triển các logic cho Runtime tuỳ vào bài toán mà blockchain của bạn đang cố gắng giải quyết.
Tập hợp các loại thư viện cốt lõi của Substrate
Trong Substrate, chúng ta bao gồm 3 loại tập hợp thư viện chính:
- Sustrate Core (
sc_*
): Thư viện cốt lõi (Core client libraries) cho các outer node services Như đã nói trước đó về vai trò của các outer node,core client libraries
sẽ bao gồm các thư viện chủ yếu về networking, consensus hay block execution như:sc_blockchain
,sc_rpc
,sc_service
,sc_offchain
, `sc_network frame_*
: FRAME libraries dành cho *Runtime-- Substrate Primitive (
sp_*
): Các Primitive libraries dùng để triển khai các hàm thức nền tảng (underlying functions) và interface cho giao tiếp giữa các thư viện với nhau.
Vì Substrate Node được xây lên từ rất nhiều thành phần thư viện khác nhau, nếu bạn muốn tìm hiểu thêm, bạn có thể xem qua các thư viện tại đây
Giải thích mã nguồn của Node
Upstream dependencies
sc-cli = thư viện của Substrate CLI
sp-core = thư viện cung cấp các Substrate type
sc-executor = thư viện đóng vai trò thực thi và xử lý các lệnh được gửi đến runtime
sc-network = thư viện mạng ngang hàng cụ thể cho Substrate
sc-service = thư viện lõi của Substrate chứa logic để chạy network, client...và quản lý giao tiếp giữa các dịch vụ
sc-telemetry = thư viện chính cho telemetry service của Substrate, hầu hết các hệ thống micro-service đều sẽ có telemetry để đóng vai trò là bác sĩ chuẩn đoán và thông báo khi hệ thống có lỗi.
sc-transaction-pool = mã nguồn của substrate transaction pool
sc-transaction-pool-api = thư viện client API để tương tác với transaction pool
sc-offchain = chứa các Substrate offchain workers
sc-statement-store = nơi lưu trữ các statement
sc-consensus-aura = giao thức đồng thuận Authority Round (Aura) của Substrate
sp-consensus-aura = thư viện primitive cho Authority Round (Aura)
sc-consensus = tổng hợp các mã nguồn cho các cơ chế đồng thuận phổ biến
sc-consensus-grandpa = dùng để tích hợp đồng thuận GRANDPA vào Substrate
sp-consensus-grandpa = thư viện primitive để tích hợp GRANDPA
sc-client-api = giao diện client của Substrate
sp-runtime = gồm lượng lớn types và utilities được sử dụng trong Substrate runtime
sp-io = gồm các giao diện để runtime giao tiếp với thế giới bên ngoài
sp-timestamp = dùng cho timestamps
frame-system = system pallets
pallet-transaction-payment = pallet này gồm các basic logic để trả phí tối thiếu cho giao địch được thực thi
node/
Cấu trúc folder benchmarking.rs
: Dùng để benchmarking Substrate nodechain_spec.rs
: Mã nguồn cho chain specification, bao gồm các thông tin của mạng blockchain mà Substrate node kết nối đến hay trạng thái căn bản mà các nodes phải chấp thuận để sản sinh blocks.
/// Specialized `ChainSpec`. This is a specialization of the general Substrate ChainSpec type.
pub type ChainSpec = sc_service::GenericChainSpec<RuntimeGenesisConfig>;
Ví dụ đây Chain Specification cấu hình cho môi trường development
pub fn development_config() -> Result<ChainSpec, String> {
let wasm_binary = WASM_BINARY.ok_or_else(|| "Development wasm not available".to_string())?;
// Bạn có thể check qua mã nguồn của hàm from_genesis() để hiểu ý nghĩa của từng parameters
Ok(ChainSpec::from_genesis(
// name: tên của chain
"Development",
// id: id của chain
"dev",
// chain_type: loại chain
ChainType::Development,
move || {
// testnet_genesis(): cấu hình trạng thái căn bản của storage cho FRAME modules
testnet_genesis(
wasm_binary,
// Initial PoA authorities
vec![authority_keys_from_seed("Alice")],
// Sudo account
get_account_id_from_seed::<sr25519::Public>("Alice"),
// Pre-funded accounts
vec![
get_account_id_from_seed::<sr25519::Public>("Alice"),
get_account_id_from_seed::<sr25519::Public>("Bob"),
get_account_id_from_seed::<sr25519::Public>("Alice//stash"),
get_account_id_from_seed::<sr25519::Public>("Bob//stash"),
],
true,
)
},
// Bootnodes
vec![],
// Telemetry
None,
// Protocol ID
None,
None,
// Properties
None,
// Extensions
None,
))
}
cli.rs
: Bao gồm các subcommand cho CLI của Substrate. Cũng không quá khác biệt với một CLI app thông thườngcommand.rs
: Các hàm thực thi logic tương ứng với các subcommand đã được khai báo trongcli.rs
rpc.rs
: Tập hợp các hàm RPC cụ thể của nodeservice.rs
: Mã nguồn khởi tạo executor, network, client, consensus và kết nối với chain từ specification ởchain_spec.rs
. Trong source code củaservice.rs
bạn sẽ thấy có 2 hàm chính lànew_partial()
vànew_full()
. Trong đó,new_full()
sẽ gọinew_partial()
để xây dựng dịch vụ cho full client,new_partial()
thì gồm các logic chính sau:
- Khởi tạo telemetry service từ các các thông tin từ
chain_spec.rs
và spawn telemetry service trên worker được khởi tạo
// 63 -72
let telemetry = config
.telemetry_endpoints
.clone()
.filter(|x| !x.is_empty())
.map(|endpoints| -> Result<_, sc_telemetry::Error> {
// Khởi tạo worker cho telemetry service với bufer size là 16
let worker = TelemetryWorker::new(16)?;
let telemetry = worker.handle().new_telemetry(endpoints);
Ok((worker, telemetry))
})
.transpose()?;
...
// 83 - 86
let telemetry = telemetry.map(|(worker, telemetry)| {
task_manager.spawn_handle().spawn("telemetry", None, worker.run());
telemetry
});
- Khởi tạo transaction pool
// 90 -96
let transaction_pool = sc_transaction_pool::BasicPool::new_full(
config.transaction_pool.clone(),
config.role.is_authority().into(),
config.prometheus_registry(),
task_manager.spawn_essential_handle(),
client.clone(),
);
- Import block từ đồng thuận GRANDPA
// 98 -103
let (grandpa_block_import, grandpa_link) = sc_consensus_grandpa::block_import(
client.clone(),
&client,
select_chain.clone(),
telemetry.as_ref().map(|x| x.handle()),
)?;
- Sau đó, ở hàm
new_full()
, Network hoàn thiện sẽ được cấu hình qua
// 155
let mut net_config = sc_network::config::FullNetworkConfiguration::new(&config.network);
...
// 171 - 181
let (network, system_rpc_tx, tx_handler_controller, network_starter, sync_service) =
sc_service::build_network(sc_service::BuildNetworkParams {
config: &config,
net_config,
client: client.clone(),
transaction_pool: transaction_pool.clone(),
spawn_handle: task_manager.spawn_handle(),
import_queue,
block_announce_validator_builder: None,
warp_sync_params: Some(WarpSyncParams::WithProvider(warp_sync)),
})?;
Phần còn lại của hàm new_full()
được giải thích và comment khá kỹ trong code. Ở phần sau, đồng thuận GRANDPA sẽ được cấu hình và khởi tạo cùng với thuật toán đồng thuận Authority Round (Aura).