• 1. 电子存证合约
    • 1.1. 问题引入
    • 1.2. 数据结构的设计
    • 1.3. 电子存证合约的功能实现
    • 1.4. 合约使用方法
      • 1.4.1. 合约部署(Deploy)
      • 1.4.2. 合约执行(Save)
      • 1.4.3. 合约查询(Query)

    1. 电子存证合约

    1.1. 问题引入

    假设我们面临着这样的一个问题:“几个摄影师朋友找到你,他们的摄影作品上传到自己的blog后总是被其他人盗用,使用水印之类的方法也无法避免像截取部分这种情况,他们需要一个能证明摄影作品最早是由自己上传、而且具有法律效力可供自己进行维权的工具”

    显然区块链对于解决此问题有很大的帮助,它的不可篡改等特性很适合存证维权的场景,我们可以通过XuperChain来构建一个存取证据的智能合约(担心不被法院认可? 这里 或许能够解答你的疑问)

    下面我们就来教你帮助摄影师朋友开发一个能够存储照片版权、还能在发现被盗用后进行维权的智能合约

    1.2. 数据结构的设计

    对于摄影作品,通常是一个图片文件,其大小根据清晰度等原因可以多达几十MB(甚至更多),为避免存储空间浪费、以及保证区块链交易的效率,我们可以使用哈希算法(例如SHA256)只将图片的哈希值上链,而原图可以保存在其他地方

    我们可以这样定义“证据文件”的数据结构,包含哈希值和上传的时间戳

    1. type UserFile struct {
    2. Timestamp int64
    3. Hashval []byte
    4. }

    为了能够存储多个“证据文件”,并且能够服务于更多的摄影师朋友,我们可以定义一个上传者到文件的map

    1. type User struct {
    2. Owner string
    3. UserFiles map[string]*UserFile
    4. }

    代码样例可以参看:contractsdk/go/example/eleccert.go

    1.3. 电子存证合约的功能实现

    从场景我们可以大致推断,以下两个功能是必要的

    • 存储一个到“证据文件”区块链(save方法)
    • 获取已经存储过的某一个“证据文件”(query方法)更底层考虑,我们可以使用XuperChain提供的合约SDK功能 PutObject 和 GetObject 来提供实际的存取功能

    对于XuperChain中的智能合约,Initialize是一个必须实现的方法,当且仅当合约被部署的时候会运行一次,我们这里采用“每个摄影师部署自己的合约来存储自己需要的作品”这种方式,将一些和上传者相关的初始化操作放在函数中

    Save、Query和Initialize方法的具体实现可以参考代码样例

    1.4. 合约使用方法

    1.4.1. 合约部署(Deploy)

    编译并部署合约的过程可以参考 部署wasm合约 章节,注意资源消耗可以一开始不加 —fee 参数,执行后会给出需要消耗的资源数

    1.4.2. 合约执行(Save)

    执行合约进行“存证操作”的命令如下(运行需要使用 —fee 参数提供资源消耗):

    1. ./xchain-cli wasm invoke -a '下面json中args字段的内容' --method save -H localhost:37101 eleccert
    1. {
    2. "module_name": "wasm", # native or wasm
    3. "contract_name": "eleccert", # contract name
    4. "method_name": "save", # invoke or query
    5. "args": {
    6. "owner": "aaa", # user name
    7. "filehash": "存证文件的hash值",
    8. "timestamp": "存证的timestamp"
    9. }
    10. }

    1.4.3. 合约查询(Query)

    执行合约进行“取证操作”的命令如下(查询操作不需要提供资源):

    1. ./xchain-cli wasm query -a 'args内容' --method query -H localhost:37101 eleccert
    1. {
    2. "module_name": "native", # native or wasm
    3. "contract_name": "eleccert", # contract name
    4. "method_name": "query", # invoke or query
    5. "args": {
    6. "owner": "aaa", # user name
    7. "filehash": "文件hash值"
    8. }
    9. }
    10. # output 如下
    11. {
    12. "filehash": "文件hash值",
    13. "timestamp": "文件存入timestamp"
    14. }