一个简单的以太坊合约让imtoken支持多签
熟悉比特币和以太坊的人应该都知道,在比特币中有2种类型的地址,1开头的是P2PKH,就是个人地址,3开头的是P2SH,一般是一个多签地址。所以在原生上比特币就支持多签。多签的一个优势就是可以多方对一笔付款达成共识,才能支付成功。比如3个人合伙开公司,他们的对外付款是比特币,为了防止管理财务的人作恶,于是他们可以创建2/3多签的地址,每个人持有一个私钥,对于每一笔付款,必须任意2个人都签名了才能支付出去。
比特币上的这个多签地址在以太坊上是没有原生支持的!以太坊最大的优点是支持图灵完备的智能合约,所以多签功能需要靠智能合约来实现。
为了简化代码,我们的需求是这样的:创建一个AB两个用户创建2/2的多签合约,该合约支持指定的ERC20 Token的支付。当某需要对外付款时,A用户调用合约,发起对C的转账n个Token,B用户也必须调用合约,发起对C的转账n个Token,只有A和B都调用了合约后,合约才会真的付款。其他用户发起转账无效。
根据以上需求,我改了一款极其简单的多签合约。代码如下:
- pragma solidity ^0.4.24;
- interface Token {
- function balanceOf(address _owner) public view returns (uint256 );
- function transfer(address _to, uint256 _value) public ;
- }
- contract MultiSig {
- address private addrA;
- address private addrB;
- address private addrToken;
- struct Permit {
- bool addrAYes;
- bool addrBYes;
- }
- mapping (address => mapping (uint => Permit)) private permits;
- event Transfer(address indexed from, address indexed to, uint256 value);
- constructor(address a, address b, address tokenAddress) public{
- addrA = a;
- addrB = b;
- addrToken = tokenAddress;
- }
- function getAddrs() public view returns(address, address,address) {
- return (addrA, addrB,addrToken);
- }
- function transferTo(address to, uint amount) public{
- Token token = Token(addrToken);
- require(token.balanceOf(this) >= amount);
- if (msg.sender == addrA) {
- permits[to][amount].addrAYes = true;
- } else if (msg.sender == addrB) {
- permits[to][amount].addrBYes = true;
- } else {
- require(false);
- }
- if (permits[to][amount].addrAYes == true && permits[to][amount].addrBYes == true) {
- token.transfer(to, amount);
- permits[to][amount].addrAYes = false;
- permits[to][amount].addrBYes = false;
- }
- emit Transfer(msg.sender, to, amount);
- }
- }
以上代码十分简陋,功能十分有限,而且需要在etherscan或者remix上调用,对用户来说十分不友好,于是我想到了可以按ERC20标准接口对这个多签合约进行改造。改造后的合约看起来像是一个Token,但是本质上是一个多签地址。A B用户都可以使用imtoken或者KCash之类的支持ERC20的钱包APP进行多签,而不需要任何复杂的技能。所以我改写后的多签合约如下:
- pragma solidity ^0.4.24;
- interface IERC20 {
- function transfer(address to, uint256 value) external returns (bool);
- function approve(address spender, uint256 value) external returns (bool);
- function transferFrom(address from, address to, uint256 value) external returns (bool);
- function totalSupply() external view returns (uint256);
- function balanceOf(address who) external view returns (uint256);
- function allowance(address owner, address spender) external view returns (uint256);
- event Transfer(address indexed from, address indexed to, uint256 value);
- event Approval(address indexed owner, address indexed spender, uint256 value);
- }
- contract MultiSig is IERC20 {
- address private addrA;
- address private addrB;
- address private addrToken;
- struct Permit {
- bool addrAYes;
- bool addrBYes;
- }
- mapping (address => mapping (uint => Permit)) private permits;
- event Transfer(address indexed from, address indexed to, uint256 value);
- event Approval(address indexed owner, address indexed spender, uint256 value);
- uint public totalSupply = 10*10**26;
- uint8 constant public decimals = 18;
- string constant public name = "MutiSigPTN";
- string constant public symbol = "MPTN";
- function approve(address spender, uint256 value) external returns (bool){
- return false;
- }
- function transferFrom(address from, address to, uint256 value) external returns (bool){
- return false;
- }
- function totalSupply() external view returns (uint256){
- IERC20 token = IERC20(addrToken);
- return token.totalSupply();
- }
- function allowance(address owner, address spender) external view returns (uint256){
- return 0;
- }
- constructor(address a, address b, address tokenAddress) public{
- addrA = a;
- addrB = b;
- addrToken = tokenAddress;
- }
- function getAddrs() public view returns(address, address,address) {
- return (addrA, addrB,addrToken);
- }
- function transfer(address to, uint amount) public returns (bool){
- IERC20 token = IERC20(addrToken);
- require(token.balanceOf(this) >= amount);
- if (msg.sender == addrA) {
- permits[to][amount].addrAYes = true;
- } else if (msg.sender == addrB) {
- permits[to][amount].addrBYes = true;
- } else {
- require(false);
- }
- if (permits[to][amount].addrAYes == true && permits[to][amount].addrBYes == true) {
- token.transfer(to, amount);
- permits[to][amount].addrAYes = false;
- permits[to][amount].addrBYes = false;
- }
- emit Transfer(msg.sender, to, amount);
- return true;
- }
- function balanceOf(address _owner) public view returns (uint) {
- IERC20 token = IERC20(addrToken);
- if (_owner==addrA || _owner==addrB){
- return token.balanceOf(this);
- }
- return 0;
- }
- }
这里我需要特别指出的是decimals这个属性必须与所支持的ERC20 Token一致,这样钱包才会算出正确的转账金额。另外imtoken的缓存刷新比较慢,并不是部署了合约后马上就能搜索到的。以上代码都实测没问题。
一个简单的以太坊合约让imtoken支持多签的更多相关文章
- go打造以太坊合约测试框架
传送门: 柏链项目学院 1 以太坊智能合约编译 以太坊智能合约编写使用solidity语言,一般情况下我们会在remix环境下进行编译测试,在线环境相对比较稳定.如果不想用在线环境,那我们就需要自己动 ...
- 以太坊智能合约Hello World示例程序
简介 以太坊(Ethereum)是一提供个智能合约(smart contract)功能的公共区块链(BlockChain)平台. 本文介绍了一个简单的以太坊智能合约的开发过程. 开发环境 在以太坊上开 ...
- [币严区块链]简单易懂的以太坊(ETH)智能合约开发入门教程
以太坊(Ethereum)是一提供个智能合约(smart contract)功能的公共区块链(BlockChain)平台. 本文介绍了一个简单的以太坊智能合约的开发过程. 开发环境 在以太坊上开发应用 ...
- c#实战开发:用.net core开发一个简单的Web以太坊钱包 (六)
今天就来开发一个C# 版的简易钱包 先回顾以前的内容 c#实战开发:以太坊Geth 命令发布智能合约 (五) c#实战开发:以太坊Geth 常用命令 (四) c#实战开发:以太坊钱包快速同步区块和钱包 ...
- 如何在以太坊上搭建一个Dapp?
原创: 前哨小兵甲 区块链前哨 昨天 策划|Tina作者|Mahesh Murthy俗话说,实践出真知!对于开发人员来说,最好的学习办法就是亲自动手做一个小项目.所以,接下来我们将会以一个投票程序为例 ...
- 区块链入门到实战(27)之以太坊(Ethereum) – 智能合约开发
智能合约的优点 与传统合同相比,智能合约有一些显著优点: 不需要中间人 费用低 代码就是规则 区块链网络中有多个备份,不用担心丢失 避免人工错误 无需信任,就可履行协议 匿名履行协议 以太坊(Ethe ...
- 基于以太坊开发的类似58同城的DApp开发与应用案例
今天,Origin开发团队很高兴地宣布在以太坊Rinkeby测试网络上推出Origin Protocol Demo DApp ! 在这个DApp中,你可以在不同垂直行业的solidarity econ ...
- C#以太坊基础入门
在这一部分,我们将使用C#开发一个最简单的.Net控制台应用,来接入以太坊节点,并打印 所连接节点旳版本信息.通过这一部分的学习,你将掌握以下技能: 如何使用节点仿真器 如何在命令行访问以太坊节点 如 ...
- Geth命令用法-参数详解 and 以太坊源码文件目录
本文是对以太坊客户端geth命令的解析 命令用法 geth [选项] 命令 [命令选项] [参数-] 版本 1.7.3-stable 命令 account 管理账户 attach 启动交互式JavaS ...
随机推荐
- 吴恩达机器学习笔记58-协同过滤算法(Collaborative Filtering Algorithm)
在之前的基于内容的推荐系统中,对于每一部电影,我们都掌握了可用的特征,使用这些特征训练出了每一个用户的参数.相反地,如果我们拥有用户的参数,我们可以学习得出电影的特征. 但是如果我们既没有用户的参数, ...
- 找不到servlet对应的class
javax.servlet.ServletException: Wrapper cannot find servlet class com.suntomor.lewan.pay.NotifyRecei ...
- html 和css 效果--整理集合篇
一.如何用一张图片做背景图,并且图片自适应div的大小 background: url("../stu_wengu.png") center center no-repeat; b ...
- 深度解密HTTP通信细节
目录 HTTP报文截获 背景介绍 抓包 mac本地 远程docker 请求 && 分析 关闭服务进程 关闭docker 重启docker 正常请求 HTTP协议分析 整体介绍 编码 M ...
- 爸爸又给Spring MVC生了个弟弟叫Spring WebFlux
情景引入 很早之前,Java就火起来了,是因为它善于开发和处理网络方面的应用. Java有一个爱好,就是喜欢制定规范标准,但自己又不善于去实现. 反倒是一些服务提供商使用它的规范标准来制造应用服务器而 ...
- SLAM+语音机器人DIY系列:(四)差分底盘设计——2.stm32主控软件设计
摘要 运动底盘是移动机器人的重要组成部分,不像激光雷达.IMU.麦克风.音响.摄像头这些通用部件可以直接买到,很难买到通用的底盘.一方面是因为底盘的尺寸结构和参数是要与具体机器人匹配的:另一方面是因为 ...
- 类和对象,以及 LeetCode 每日一题
所有类都是引用类型. 1 定义类 类是某一批对象的抽象. 1.1 定义类的语法: [修饰符] class 类名{ 零到多个构造器定义 零到多个成员变量 零到多个方法 } 对于一个类定义而言,可以包含三 ...
- Odoo 菜单美化的扩展模块
详见: http://www.oejia.net/blog/2018/07/07/oejia_menu_about.html Odoo 菜单美化主题,odoo默认的菜单算是简洁好用的,如果您觉得还少了 ...
- 自己动手写事件总线(EventBus)
本文由云+社区发表 事件总线核心逻辑的实现. EventBus的作用 Android中存在各种通信场景,如Activity之间的跳转,Activity与Fragment以及其他组件之间的交互,以及在某 ...
- ElasticSearch、Logstash、Kibana 搭建高效率日志管理系统
ELK (ElasticSearch.LogStash以及Kibana)三者组合是一个非常强大的工具,这里我们来实现监控日志文件并且收到日志到ElasticSearch搜索引擎,利用Kibana可视化 ...