手记

以太访+IPFS实现一个云存储系统(存储就送币)

以太访+IPFS实现一个简单的云存储系统(存储就送币)

作者: AlexTan

CSDN:   http://blog.csdn.net/alextan_

Github: https://github.com/AlexTan-b-z

e-mail: alextanbz@gmail.com


一、介绍:

上一篇博文"以太坊+IPFS+WEB 电商平台开发讲解"介绍了用以太访+IPFS实现电商平台的思路、合约接口的实现以及一些相关的基本概念。这篇博文将讲解具体的一个简单的实战项目,及用以太访+IPFS实现的一个云存储,并用ERC20标准实现了自己的代币,以太访加IPFS实现存储就送代币。

  • 本系统用IPFS来充当存储介质,及用户所上传的东西都是存在IPFS上的,IPFS会返回一个地址(什么是地址?简单理解,你可以通过地址来找到你所存储文件的内容)。

  • 用合约来存储用户以及每个用户所存储的IPFS地址

  • 用合约实现一个ERC20标准的代币。代币名字叫CloudB,即云币,总发行量10000000。

  • 起初每存一次文件便会赠送10个云币,但会花费少量的gas费。

  • 待10000000个云币赠送完后,每存储一个文件会花费2个云币以及少量的gas费。

  • 查看文件不会产生任何费用。

二、什么是ERC20标准的代币:

ERC20是以太坊定义的一个代币标准。要求我们在实现代币的时候必须要遵守的协议,如指定代币名称、总量、实现代币交易函数等,只有支持了协议才能被以太坊钱包支持。其接口如下:

contract ERC20Interface {    string public constant name = "Token Name";    string public constant symbol = "SYM";    uint8 public constant decimals = 18;  // 18 is the most common number of decimal places    function totalSupply() public constant returns (uint);    function balanceOf(address tokenOwner) public constant returns (uint balance);    function allowance(address tokenOwner, address spender) public constant returns (uint remaining);    function transfer(address to, uint tokens) public returns (bool success);    function approve(address spender, uint tokens) public returns (bool success);    function transferFrom(address from, address to, uint tokens) public returns (bool success);    event Transfer(address indexed from, address indexed to, uint tokens);    event Approval(address indexed tokenOwner, address indexed spender, uint tokens);}

简单说明一下:

  • name : 代币名称;

  • symbol: 代币符号;

  • decimals: 代币小数点位数,代币的最小单位, 18表示我们可以拥有 .0000000000000000001单位个代币;

  • totalSupply() : 发行代币总量;

  • balanceOf(): 查看对应账号的代币余额;

  • transfer(): 实现代币交易,用于给用户发送代币(从我们的账户里);

  • transferFrom(): 实现代币用户之间的交易;

  • allowance(): 控制代币的交易,如可交易账号及资产;

  • approve(): 允许用户可花费的代币数;

代币合约代码(TokenERC20.sol):

pragma solidity ^0.4.16;interface tokenRecipient { function receiveApproval(address _from, uint256 _value, address _token, bytes _extraData) public; }contract TokenERC20 {    string public name;    string public symbol;    uint8 public decimals = 18;  // decimals 可以有的小数点个数,最小的代币单位。18 是建议的默认值    uint256 public totalSupply;    // 用mapping保存每个地址对应的余额    mapping (address => uint256) public balanceOf;    // 存储对账号的控制    mapping (address => mapping (address => uint256)) public allowance;    // 事件,用来通知客户端交易发生    event Transfer(address indexed from, address indexed to, uint256 value);    // 事件,用来通知客户端代币被消费    event Burn(address indexed from, uint256 value);    /**        * 初始化构造        */    function TokenERC20(uint256 initialSupply, string tokenName, string tokenSymbol, address rootWallet) public {        totalSupply = initialSupply * 10 ** uint256(decimals);  // 供应的份额,份额跟最小的代币单位有关,份额 = 币数 * 10 ** decimals。        balanceOf[rootWallet] = totalSupply;                // 指定账户地址拥有所有的代币        name = tokenName;                                   // 代币名称        symbol = tokenSymbol;                               // 代币符号    }    /**        * 代币交易转移的内部实现        */    function _transfer(address _from, address _to, uint _value) internal {        // 确保目标地址不为0x0,因为0x0地址代表销毁        require(_to != 0x0);        // 检查发送者余额        require(balanceOf[_from] >= _value);        // 确保转移为正数个        require(balanceOf[_to] + _value > balanceOf[_to]);        // 以下用来检查交易,        uint previousBalances = balanceOf[_from] + balanceOf[_to];        // Subtract from the sender        balanceOf[_from] -= _value;        // Add the same to the recipient        balanceOf[_to] += _value;        Transfer(_from, _to, _value);        // 用assert来检查代码逻辑。        assert(balanceOf[_from] + balanceOf[_to] == previousBalances);    }    /**        *  代币交易转移        * 从自己(创建交易者)账号发送`_value`个代币到 `_to`账号        *        * @param _to 接收者地址        * @param _value 转移数额        */    function transfer(address _to, uint256 _value) public {        _transfer(msg.sender, _to, _value);    }    /**        * 账号之间代币交易转移        * @param _from 发送者地址        * @param _to 接收者地址        * @param _value 转移数额        */    function transferFrom(address _from, address _to, uint256 _value) public payable returns (bool success) {        require(_value <= allowance[_from][msg.sender]);     // Check allowance        allowance[_from][msg.sender] -= _value;        _transfer(_from, _to, _value);        return true;    }    /**        * 设置某个地址(合约)可以创建交易者名义花费的代币数。        *        * 允许发送者`_spender` 花费不多于 `_value` 个代币        *        * @param _spender The address authorized to spend        * @param _value the max amount they can spend        */    function approve(address _spender, uint256 _value) public        returns (bool success) {            allowance[msg.sender][_spender] = _value;            return true;        }    /**        * 设置允许一个地址(合约)以我(创建交易者)的名义可最多花费的代币数。        *        * @param _spender 被授权的地址(合约)        * @param _value 最大可花费代币数        * @param _extraData 发送给合约的附加数据        */    function approveAndCall(address _spender, uint256 _value, bytes _extraData)        public        returns (bool success) {        tokenRecipient spender = tokenRecipient(_spender);        if (approve(_spender, _value)) {            // 通知合约            spender.receiveApproval(msg.sender, _value, this, _extraData);            return true;        }    }    /**        * 销毁我(创建交易者)账户中指定个代币        */    function burn(uint256 _value) public returns (bool success) {        require(balanceOf[msg.sender] >= _value);   // Check if the sender has enough        balanceOf[msg.sender] -= _value;            // Subtract from the sender        totalSupply -= _value;                      // Updates totalSupply        Burn(msg.sender, _value);        return true;    }    /**        * 销毁用户账户中指定个代币        *        * Remove `_value` tokens from the system irreversibly on behalf of `_from`.        *        * @param _from the address of the sender        * @param _value the amount of money to burn        */    function burnFrom(address _from, uint256 _value) public returns (bool success) {        require(balanceOf[_from] >= _value);                // Check if the targeted balance is enough        require(_value <= allowance[_from][msg.sender]);    // Check allowance        balanceOf[_from] -= _value;                         // Subtract from the targeted balance        allowance[_from][msg.sender] -= _value;             // Subtract from the sender's allowance        totalSupply -= _value;                              // Update totalSupply        Burn(_from, _value);        return true;    }}

三、存储合约代码实现

存储合约里实现了,upload上传(实现了赠送代币以及花费代币)、download下载、获取用户所存储文件的个数等函数。具体代码如下:

SaveCloud.sol:

pragma solidity ^0.4.16;import "./TokenERC20.sol";contract SaveCloud {    enum IsFinish {Yes, No}    IsFinish isFinish; // 用于判断代币是否已经赠送完    address private owner; // 合约的创建者    uint public saveIndex; // 合约存储文件的总数量    mapping (address=>string[]) public ipfs; // 用于存放每个用户所存储到ipfs的ipfs地址 address => 用户地址,string[] => ipfs地址    TokenERC20 public token; // 代币合约的实例化对象        event Upload(string ipfsAddress );    event Download(uint number);    modifier onlyOwner() {        require(msg.sender == owner);        _;    }    function SaveCloud() public {    /* 构造函数,部署合约时便会调用这个函数 */        owner = msg.sender;        saveIndex = 0;        isFinish = IsFinish.No;    }    function initToken(address tokenAddr) public onlyOwner {         /* 实例化代币合约对象,只有创建合约者才能调用此函数 */        token = TokenERC20(tokenAddr);    }    function upload(string ipfsAddress) public payable returns (bool success) {        if (token.balanceOf(address(this)) >= 10 && isFinish == IsFinish.No) {            token.transfer(msg.sender, 10); // 合约给sender转代币,存东西送代币        }else {            isFinish = IsFinish.Yes;        }        if (isFinish == IsFinish.Yes) {            token.transferFrom(msg.sender, owner, 2);            // 需要aprove(授权),如果不加aprove的话,谁都可以来调这个函数花别人的代币了,存东西,花代币,代币转给合约账户        }        ipfs[msg.sender].push(ipfsAddress);        saveIndex += 1;        Upload(ipfsAddress);        return true;    }    function download(uint number) external returns (string ipfsAddr) {        /* 获取用户的所存放文件的位置(ipfs地址) */        Download(number);        return ipfs[msg.sender][number];    }    function getLength() external view returns (uint length) {        /* 获取用户所存放文件的个数 */        return ipfs[msg.sender].length;    }}

三、部署代码:

笔者使用的是truffle框架,其本地部署代码如下:

2_deploy_contracts.js:

var TokenERC20 = artifacts.require("./TokenERC20.sol");var SaveCloud = artifacts.require("./SaveCloud.sol");module.exports = function(deployer) {  deployer.deploy(SaveCloud).then(function() {    /* 实例化SaveCloud合约后实例化代币合约 */    return deployer.deploy(TokenERC20, 10000000, "cloudb", "CloudB", SaveCloud.address);  })};

四、js调用代码:

js调用代码、以及前端页面代码,后续将会发送到github,还请大家自行查看。

原文出处

0人推荐
随时随地看视频
慕课网APP