3. 创建合约

超级链支持丰富的智能合约开发语言,比如go,Solitidy,C++,Java等。

3.1. 编写合约

源码可以参考 xuperchain/core/contractsdk/go/example/math/math.go

主要实现struct中initialize,invoke和query三个方法来实现自己的逻辑

  1. func (m *math) Initialize(nci code.Context) code.Response { ... }
  2. func (m *math) Invoke(nci code.Context) code.Response { ... }
  3. func (m *math) Query(nci code.Context) code.Response { ... }

每个函数的入口参数均为 code.Context ,具体结构可参考 xuperchain/core/contractsdk/go/code/context.go 接口中定义了如何获取传入方法的参数,如何使用读写功能,以及如何在链上进行交易/区块的查询、转账或调用其他合约

  1. type Context interface {
  2. Args() map[string][]byte
  3. Caller() string
  4. Initiator() string
  5. AuthRequire() []string
  6. PutObject(key []byte, value []byte) error
  7. GetObject(key []byte) ([]byte, error)
  8. DeleteObject(key []byte) error
  9. NewIterator(start, limit []byte) Iterator
  10. QueryTx(txid []byte) (*TxStatus, error)
  11. QueryBlock(blockid []byte) (*Block, error)
  12. Transfer(to string, amount *big.Int) error
  13. Call(module, contract, method string, args map[string][]byte) (*Response, error)
  14. }

对于C++版本的合约,可以参考代码 contractsdk/cpp/example/counter.cc 原理和Golang合约是一致的

注解

除了 Initialize 外的其他函数,是可以自行定义函数名的,可参考contractsdk/go/example/counter/counter.go中的具体实例,在之后调用合约时写明函数名即可

3.2. 部署wasm合约

  1. 编译合约 - Golang

    注意合约编译环境与源码编译环境一致,编译参数如下

    1. GOOS=js GOARCH=wasm go build XXX.go
  2. 编译合约 - C++

    对于C++合约,已提供编译脚本,位于 contractsdk/cpp/build.sh,需要注意的是,脚本依赖从hub.baidubce.com拉取的docker镜像,请在编译前确认docker相关环境是可用的

  3. 部署wasm合约

    将编译好的合约二进制文件(以counter为例)放到目录node/data/blockchain/${chain name}/wasm/下,这里我们默认的链名 ${chain name}=xuper

    部署合约的操作需要由合约账号完成,部署操作同样需要支付手续费,操作前需要确保合约账号下有足够的余额

    示例中我们的环境里创建了一条名为xuper的链,包含一个合约账号 XC1111111111111111@xuper

    账号的acl查询如下:

    查询acl

    为部署合约,我们需要事先准备一个符合权限的地址列表(示例中将其保存在 data/acl/addrs 文件),这里因为acl里只有一个AK,我们只需在文件中添加一行(如果acl中需要多个AK,那么编辑文件,每行填写一个即可)

    1. echo "XC1111111111111111@xuper/dpzuVdosQrF2kmzumhVeFQZa1aYcdgFpN" > data/acl/addrs

    然后我们按照以下命令来部署wasm合约counter

    1. ./xchain-cli wasm deploy --account XC1111111111111111@xuper --cname counter -m -a '{"creator": "someone"}' -A data/acl/addrs -o tx.output --keys data/keys --name xuper -H localhost:37101 counter

    此命令看起来很长,但是其中很多参数都有默认值,我们先来看一下参数的含义:

    • wasm deploy :此为部署wasm合约的命令参数,不做过多解释

    • --account XC1111111111111111@xuper :此为部署wasm合约的账号(只有合约账号才能进行合约的部署)

    • --cname counter :这里的counter是指部署后在链上的合约名字,可以自行命名(但有规则,长度在4~16字符)

    • -m :意为多重签名的方式,目前版本的xchain部署wasm合约都需要以这种方式

    • -a '{"creator": "someone"}' :此为传入合约的参数,供合约Initialize方法使用(此参数并非必须,只不过此处的counter合约需要传一个”creator”参数,参见contractsdk/cpp/example/counter.cc)

    • -A data/acl/addrs :此即为需要收集签名的列表,默认路径为data/acl/addrs,如不是则需要显式传入(注意权重要满足acl要求)

    • -o tx.output :此为输出的tx文件,可不传,默认文件名为tx.out

    • --keys data/keys :此为部署发起者的密钥地址,可不传,默认值即为data/keys(部署发起者也要进行签名)

    • --name xuper :此为区块链名称,默认为xuper,如果创建链名称不是xuper则需要显式传入

    • -H localhost:37101 :xchain服务的地址,默认是本机的37101端口,如不是则需要显式传入

    • 最后的counter是合约编译好的文件(编译完成默认是counter.wasm)

    在此处,我们大部分参数取的是默认值,所以命令参数不必这么多了

    1. ./xchain-cli wasm deploy --account XC1111111111111111@xuper --cname counter -m -a '{"creator": "someone"}' counter

    运行效果如下

    发起wasm合约部署

    运行时会提示手续费的数目,使用 –fee 参数传入即可

    然后收集所需AK的签名,因为示例中我们只有一个AK(同时也是发起者),所以只需要签名一次

    1. ./xchain-cli multisig sign --tx tx.out --output sign.out --keys data/keys

    这里的 --output --keys 参数也有默认值(输出到sign.out文件,密钥位于data/keys),可以不加。运行后我们即可获得此AK的签名

    运行效果如下

    对tx签名

    收集完发起者和acl需要的签名后,我们即可发送交易,完成合约部署了

    1. ./xchain-cli multisig send --tx tx.out sign.out sign.out

    这里 multisig send 为发送多重签名的命令参数, --tx 是交易文件,后边的两个参数分别为发起者的签名和acl的签名(acl中有多个AK时,用逗号连接多个签名文件)。运行命令可得到交易上链后的id,我们也可以使用以下命令来查询部署结果

    1. ./xchain-cli account contracts --account XC1111111111111111@xuper

    会显示此合约账号部署过的所有合约

    运行效果如下

    发送部署交易

3.3. 部署native合约

如果本地搭建超级链环境,在部署、调用native合约之前,请先查看`conf/xchain.yaml` 中native一节,确保native合约功能开启。

  1. # 管理native合约的配置
  2. native:
  3. enable: true
  4. # docker相关配置
  5. docker:
  6. enable:false
  7. # 合约运行的镜像名字
  8. imageName: "docker.io/centos:7.5.1804"
  9. # cpu核数限制,可以为小数
  10. cpus: 1
  11. # 内存大小限制
  12. memory: "1G"
  13. # 停止合约的等待秒数,超时强制杀死
  14. stopTimeout: 3
  1. 编译合约 - Golang

    编译native合约时,只要保持环境和编译XuperChain源码时一致即可,我们还是以contractsdk/go/example中的counter合约为例

    1. cd contractsdk/go/example/counter
    2. go build
    3. # 产出二进制文件counter,用于合约部署
  2. 编译合约 - Java

    编译Java sdk:Java版本不低于Java1.8版本

    包管理器:maven,mvn版本3.6+

    1. # 编译java sdk
    2. cd contractsdk/java
    3. mvn install -f pom.xml
    4. # 产出二进制文件target/java-contract-sdk-0.1.0.jar,并自动安装到mvn本地仓库下

    编译native合约时,我们以contractsdk/java/example中的counter合约为例

    1. cd contractsdk/java/example/counter
    2. mvn package -f pom.xml
    3. # 产出二进制文件target/counter-0.1.0-jar-with-dependencies.jar,用于合约部署
  3. 部署合约

    部署native合约。针对不同语言实现的合约,主要通过 --runtime 字段进行区分

    1. # 部署golang native合约
    2. ./xchain-cli native deploy --account XC1111111111111111@xuper -a '{"creator":"XC1111111111111111@xuper"}' --fee 15587517 --runtime go counter --cname golangcounter
    3. # 部署结果
    4. # contract response: ok
    5. # The gas you cousume is: 14311874
    6. # The fee you pay is: 15587517
    7. # Tx id: af0d46f6df2edba4d9d9d07e1db457e5267274b1c9fe0611bb994c0aa7931933
    8. # 部署java native合约
    9. ./xchain-cli native deploy --account XC1111111111111111@xuper --fee 15587517 --runtime java counter-0.1.0-jar-with-dependencies.jar --cname javacounter
    10. # 部署结果
    11. # contract response: ok
    12. # The gas you cousume is: 14311876
    13. # The fee you pay is: 15587517
    14. # Tx id: 875d2c9129973a1c64811d7a5a55ca80743102abc30d19f012656fa52ee0f4f7
    • --runtime go :表示部署的是golang native合约

    • --runtime java:表示部署的是java native合约

  4. 合约调用

    调用native合约。针对不同语言实现的native合约,调用方式相同。通过合约名直接发起合约调用和查询

    1. # 调用golang native合约,Increase方法,golangcounter为合约名
    2. ./xchain-cli native invoke --method Increase -a '{"key":"test"}' golangcounter --fee 10
    3. # 调用结果
    4. # contract response: 1
    5. # The gas you cousume is: 6
    6. # The fee you pay is: 10
    7. # Tx id: b387e2247780a5f5da1070a931b37c4fc7f1b68c072768053a43cffe36f2e0fb
    8. # 调用golang native合约,Get方法,golangcounter为合约名
    9. ./xchain-cli native query --method Get -a '{"key":"test"}' golangcounter
    10. # 调用结果
    11. # contract response: 1
    12. # 调用java native合约,increase方法,javacounter为合约名
    13. ./xchain-cli native invoke --method increase -a '{"key":"test"}' javacounter --fee 10
    14. # 调用结果
    15. # contract response: 1
    16. # The gas you cousume is: 6
    17. # The fee you pay is: 10
    18. # Tx id: 4b46d9b1292481dcac3b504d5f8031e4eff44d8514c9508f121145cfa141d9db
    19. # 调用java native合约,get方法,javacounter为合约名
    20. ./xchain-cli native query --method get -a '{"key":"test"}' javacounter
    21. # 调用结果
    22. # contract response: 1146398290725d36631aa70f731bc3174e6484a9a

3.4. 部署solidity合约

如果本地搭建超级链环境,在部署、调用solidity合约之前,请先查看`conf/xchain.yaml` 中evm一节,确保evm合约功能开启。

  1. # evm合约配置
  2. evm:
  3. driver: "evm"
  4. enable: true
  1. 编译合约 - Solidity

    使用solc编译solidity合约。安装solc编译器,请参见**https://solidity-cn.readthedocs.io/zh/latest/installing-solidity.html\*\*。

    1. solc --version
    2. // solc, the solidity compiler commandline interface
    3. // Version: 0.5.9+commit.c68bc34e.Darwin.appleclang
    4. // 以上打印说明编译器安装成功

    编译native合约时,我们以contractsdk/java/example中的counter合约为例

    1. cd contractsdk/evm/example/counter
    2. // 通过solc编译合约源码
    3. solc --bin --abi Counter.sol -o .
    4. // 合约二进制文件和abi文件分别存放在当前目录下,Counter.bin和Counter.abi。
  2. 部署合约

    部署solidity合约。

    1. # 部署solidity合约
    2. ./xchain-cli evm deploy --account XC1111111111111111@xuper --cname counterevm --fee 5200000 Counter.bin --abi Counter.abi
    3. # 其中--abi表示合约的abi文件
    4. # 部署结果
    5. # contract response: ok
    6. # The gas you cousume is: 1789
    7. # The fee you pay is: 22787517
    8. # Tx id: 78469246d86a92ad47e5c15991a55978075902809346e48533e09a8eb0e3a7e4
    • --abi Counter.abi :表示部署需要使用的abi文件,用于合约方法参数编解码

    • ``-a ``:如果合约需要构造函数,通过-a进行指定。与c++、golang等合约的部署和调用方式相同。

  3. 合约调用

    调用solidity合约。通过合约名直接发起合约调用和查询。

    1. # 调用solidity合约,increase方法,counterevm为合约名
    2. ./xchain-cli evm invoke --method increase -a '{"key":"test"}' counterevm --fee 22787517 --abi Counter.abi
    3. # 调用结果
    4. # contract response:
    5. # The gas you cousume is: 65
    6. # The fee you pay is: 22787517
    7. # Tx id: 94655ab00188de70c3ef2f91b9db0d156142ce92f91a5da20f0f1fc7830fb700
    8. # 调用solidity合约,get方法,counterevm为合约名
    9. ./xchain-cli native query --method Get -a '{"key":"test"}' counterevm --abi Counter.abi
    10. # 调用结果,其中0表示返回值的次序,1为返回值
    11. # key,value: 0 1
  4. 超级链账户与EVM账户地址转换

    超级链有普通地址、合约账户以及合约名,这三类账户在EVM运行时需要转换为以太坊的地址类型(16进制编码字符串,形如0x1234567890abcdef1234567890abcdef12345678格式)。超级链提供了上述三个地址与EVM地址类型转换工具。

    1. # xchain合约账户地址转evm地址,contract-account表示超级链合约账户
    2. ./xchain-cli evm addr-trans -t x2e -f XC1111111111111113@xuper
    3. result, 3131313231313131313131313131313131313133 contract-account
    4. # evm地址转xchain合约账户,contract-account表示超级链合约账户
    5. ./xchain-cli evm addr-trans -t e2x -f 3131313231313131313131313131313131313133
    6. result, XC1111111111111113@xuper contract-account
    7. # xchain普通账户地址转evm地址,xchain表示超级链普通账户
    8. ./xchain-cli evm addr-trans -t e2x -f 93F86A462A3174C7AD1281BCF400A9F18D244E06
    9. result, dpzuVdosQrF2kmzumhVeFQZa1aYcdgFpN xchain
    10. # xchain普通账户地址转evm地址,xchain表示超级链普通账户
    11. ./xchain-cli evm addr-trans -t x2e -f dpzuVdosQrF2kmzumhVeFQZa1aYcdgFpN
    12. result, 93F86A462A3174C7AD1281BCF400A9F18D244E06 xchain
    13. # xchain合约名地址转evm地址,contract-name表示超级链合约名
    14. ./xchain-cli evm addr-trans -t x2e -f storagedata11
    15. result, 313131312D2D2D73746F72616765646174613131 contract-name
    16. # evm地址转xchain合约名,contract-name表示超级链合约名
    17. ./xchain-cli evm addr-trans -t e2x -f 313131312D2D2D73746F72616765646174613131
    18. result, storagedata11 contract-name
    • x2e :表示超级链地址转换为EVM地址

    • e2x :表示EVM地址转换为超级链地址。

3.5. 设置合约方法的ACL

  1. 准备desc文件setMethodACL.desc

    1. {
    2. "module_name": "xkernel",
    3. "method_name": "SetMethodAcl",
    4. "args" : {
    5. "contract_name": "counter",
    6. "method_name": "increase",
    7. "acl": "{\"pm\": {\"rule\": 1,\"acceptValue\": 1.0},\"aksWeight\": {\"UU4kyZcQinAMsBSPRLUA34ebXrfZtB4Z8\": 1}}"
    8. }
    9. }

    参数说明:

    • module_name: 模块名称,用固定值xkernel

    • method_name :方法名称,用固定值SetMethodAcl

    • contract_name:合约名称

    • method_name:合约方法名称

    • acl:合约方法的acl

  2. 设置合约方法ACL

    设置合约方法ACL的操作,需符合合约账号的ACL,在3.2节,使用 XC1111111111111111@xuper 部署的counter合约,合约账号ACL里 只有1个AK,所以在data/acl/addrs中添加1行,如果合约账号ACL里有多个AK,则填写多行。

    1. echo "XC1111111111111111@xuper/dpzuVdosQrF2kmzumhVeFQZa1aYcdgFpN" > data/acl/addrs

    执行如下命令,设置ACL:

    1. ./xchain-cli multisig gen --desc ./setMethodACL.desc --fee 1 -H 127.0.0.1:37101
    2. ./xchain-cli multisig sign --output sign.out
    3. ./xchain-cli multisig send sign.out sign.out -H 127.0.0.1:37101
  3. 查看合约方法ACL

    1. [work@]$ deploy-env -> ./xchain-cli acl query --contract counter --method increase -H :37101
    2. # 执行结果
    3. # {
    4. # "pm": {
    5. # "rule": 1,
    6. # "acceptValue": 1
    7. # },
    8. # "aksWeight": {
    9. # "UU4kyZcQinAMsBSPRLUA34ebXrfZtB4Z8": 1
    10. # }
    11. # }