# Web3.js SDK 使用说明
> **chainmaker-web3-js** 是专为长安链(ChainMaker)区块链网络设计的 JavaScript SDK,定位为 Web3 应用开发的核心工具库。它提供类似 ethers.js 的 API 体验,方便开发者在浏览器或前端环境中安全、便捷地与 ChainMaker 区块链进行交互。
## 主要功能
- **交易签名**:支持 RSA/ECDSA/国密算法,目前所有链上操作均需签名认证,保障安全性。
- **交易发送**:可便捷发起链上交易、合约调用、合约部署等操作。
- **交易订阅**:支持监听链上区块事件,实现实时响应。
- **链信息查询**:获取链配置信息、区块详情、交易详情等。
- **合约交互**:提供合约部署、调用、参数编码等完整能力。
- **多端支持**:兼容浏览器环境,NodeJS 服务端环境,支持代理配置,适配多种前端场景。
## 前置依赖
由于 chainmaker-web3-js 是面向长安链,因此对链会有一些要求,目前需要满足以下前置条件:
1. ChainMaker 区块链网络,版本`>=2.3.8`,原因是 HTTP 代理及链订阅问题在该版本得到了修复,之前版本在使用上存在问题,如没开放跨域,WebSocket订阅。
2. HTTP协议使用Web3JS,链需开启HTTP API服务, 在 chainmaker.yml 配置文件中`开启 HTTP API `服务:
```yml
# restful api gateway
gateway:
# enable restful api
enabled: true
```
参考文档见[这里](https://docs.chainmaker.org.cn/v2.3.7/html/manage/%E9%95%BF%E5%AE%89%E9%93%BE%E9%85%8D%E7%BD%AE%E7%AE%A1%E7%90%86.html#chainmaker-yml)
3. 如想使用 HTTPS 协议,需要自己使用 Nginx/Caddy 等反向代理工具进行 TLS 终端代理,本身链节点的 gRPC TLS 还是处于关闭状态。
## 快速对比 (ethers.js vs chainmaker.js)
> 通过几个基本的用法,来方便用户了解 chainmaker-web3-js 与 ethers.js 的用法区别。
| 功能 | ethers.js 示例 | chainmaker-web3-js 示例 | 主要区别 |
| -------------- | --------------------------------------- | ------------------------------------------------------------ | --------------------------------------- |
| 获取链配置信息 | `provider.getNetwork()`
无需签名 | `provider.getNetwork()`
**需签名**,provider 需 signer/wallet | chainmaker 所有操作都需签名 |
| 获取交易信息 | `provider.getTransaction(txHash)` | `provider.getTransaction(txHash)` | chainmaker 需签名,provider 配置 signer |
| 获取区块信息 | `provider.getBlock(blockHash)` | `provider.getBlock(blockHash)` | chainmaker 需签名,provider 配置 signer |
| 部署合约 | `factory.deploy()`
合约工厂需 wallet | `factory.deploy({ contractName, ... })`
需指定合约名、版本、运行时等 | chainmaker 需更多参数,所有操作需签名 |
| 调用合约 | `contract.someMethod()` | `contract.someMethod()` | chainmaker 需签名,合约实例需 wallet |
| gRPC 代理支持 | 不需要 | 需配置 `proxyHost`,浏览器端使用的话通过 gRPC-Web 代理,推荐走HTTP方式 | chainmaker 浏览器端使用配置代理,grpc支持,但首推HTTP |
## 使用
如果是npm方式使用,node推荐为v22及以上版本。这里无明确版本要求,但建议使用最新的LTS版本。
### 介绍
- web3.providers.Web3Provider
- web3.Wallet
```shell
npm install @chainmaker/web3js
```
注意:同时支持CJS/ESM两种模块化规范。
### API 使用
### 获取链配置信息
```js
// 创建钱包实例
const wallet = new web3.Wallet(privateKey); // 这种wallet叫内部signer,将会自动处理签名逻辑
// 使用带签名器的 provider
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false,
signer: wallet
}
});
// 获取网络信息
const network = await provider.getNetwork();
console.log('Chain ID:', network.chainId);
console.log('链配置:', network.config);
```
### 获取交易信息
```typescript
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {s
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false,
signer: wallet
}
});
const transaction = await provider.getTransaction('0x1234567890abcdef...');
console.log(transaction);
```
### 获取区块信息
```typescript
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false,
signer: wallet
}
});
const block = await provider.getBlock('blockHash');
console.log(block);
```
### 部署合约
```typescript
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false,
signer: wallet
}
});
const factory = new web3.ContractFactory('contractName', bytecode, {
provider
});
const res = factory.deploy({
contractName: 'fact_contract_name' + Date.now(),
contractVersion: '1.0.0',
runtimeType: 'DOCKER_GO',
params: {}
});
```
### 调用合约
```typescript
const contract = new web3.Contract(contractAddress, [], wallet);
const tx = await contract.someMethod(/* args */);
```
### 订阅
> 目前仅支持区块订阅。
```typescript
async function subscribeBlock() {
_currentBlockListener = (block) => {
console.log(block);
console.log(`新块通知: 块高=${block.getBlockHeight()}`);
};
// 开始订阅
await provider.on(
{
event: 'block',
fromBlock: Number(startBlock),
toBlock: Number(endBlock),
withRWSet: false,
onlyHeader: true
},
_currentBlockListener
);
}
// 如果需要关闭订阅,使用 provider.off('block', _currentBlockListener);
```
### K1 曲线支持
> 当前ChainMaker K1曲线下的私钥是以hex字符串形式存储的,因此可以直接传入私钥字符串创建 Wallet 实例。
```typescript
const wallet2 = new Wallet('62edf6740bc60d75763c9b980a9df2da2916b1f9c34149e5582ba566e98fbb64');
```
## 证书 Cert 模式
> 证书模式下与 PK 相比,需要增加如下配置,组织 ID/证书文本内容/认证类型
```javascript
const wallet = new web3.Wallet(privateKey, null, {
certificate: '',
orgId: 'wx-org1.chainmaker.org',
authType: 'PermissionedWithCert'
});
```
## 结合钱包插件使用
chainmaker-web3-js 支持两种使用方式:
### 1. 直接使用 Wallet(内部签名器)
当直接使用 Wallet 传入私钥时,会自动处理签名逻辑:
```typescript
const wallet = new web3.Wallet(privateKey);
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false,
signer: wallet // 内部签名器
}
});
```
### 2. 结合钱包插件使用(外部签名器)
如果想结合浏览器扩展钱包或其他钱包插件使用,可以通过实现一个符合 `web3.Signer` 接口的适配器类来实现。这与 ethers.js 的 `Web3Provider` 模式完全一致。
#### 快速示例
```typescript
import { AbstractSigner } from '@chainmaker/web3js';
// 创建适配器类
class WalletPluginAdapter extends AbstractSigner {
constructor(walletPlugin, provider) {
super(provider);
this.walletPlugin = walletPlugin;
}
async getAddress() {
return await this.walletPlugin.getAddress();
}
async signTransaction(tx) {
const populatedTx = await this.populateTransaction(tx);
return await this.walletPlugin.signTransaction(populatedTx);
}
async signMessage(message) {
return await this.walletPlugin.signMessage(message);
}
async signTypedData(domain, types, value) {
return await this.walletPlugin.signTypedData(domain, types, value);
}
connect(provider) {
return new WalletPluginAdapter(this.walletPlugin, provider);
}
}
// 使用适配器
const walletPlugin = window.chainmakerWallet; // 浏览器扩展注入的钱包对象
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
chainmaker: {
chainId: 'chainmaker_pk',
protocol: 'http',
tlsEnable: false
}
});
const signer = new WalletPluginAdapter(walletPlugin, provider);
// 像使用 ethers.js 一样使用
const address = await signer.getAddress();
const tx = await signer.sendTransaction({ /* ... */ });
```
## 请求代理
代理配置字段为 `proxyHost`。这里的代理,仅仅为了保证节点IP还配置为真实的,JS通过转发请求地址来访问链节点,使用上也可以直接配置代理地址,nodeIp指向代理地址,proxyHost不配置。
举几个例子
1. 长安链节点为grpc,未开启HTTP,这里可以搭建一个gRPC-Web的代理,nodeIp还是指向链节点的grpc地址,代理配置为proxyHost,浏览器端即可正常发起请求,nodeIp还可以真实的指向链节点的地址。
2. 长安链节点开启了HTTP服务,nodeIp指向HTTP地址,但本身网络无法直连,这时也可以通过配置proxyHost来走HTTP代理。
```javascript
const provider = web3.getDefaultProvider('36.110.223.69:11406', {
// nodeIp: "36.110.223.23:12391",
chainmaker: {
chainId: 'chain1',
protocol: 'grpc',
tlsEnable: false,
signer: wallet,
proxyHost: 'https://127.0.0.1:9080'
}
});
```