# 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' } }); ```