木星链 木星链
Ctrl+D收藏木星链
首页 > 瑞波币 > 正文

PAY:创建 ERC20 代币支付拆分智能合约

作者:

时间:1900/1/1 0:00:00

在加密货币的几乎每个领域,支付都是一个反复出现的话题,特别是向多个质押者提供支付。例如,DAO希望为多个计划提供资金,DEX希望合并向某些参与者分配交易费用,或者团队希望将代币作为月薪分发给团队成员。

智能合约使我们能够自动化这些类型的支付功能,这就限制了人工管理支付所导致的潜在错误,并允许我们将宝贵的时间花在其他生产性任务上。

今天,我们将学习如何创建自己的ERC20代币支付拆分器,它可以合并到任何项目中!

下面的内容要求你对Solidity有点熟悉,不过任何人都可以学习。

我们将创建两个合约。第一个将是ERC20代币支付拆分智能合约,第二个将是模拟池智能合约。ERC20代币支付拆分器智能合约将是抽象的,并持有用于管理收付方及其各自支付部分的逻辑和数据。模拟池将继承ERC20代币支付拆分器,以便我们可以自动将支付分发给多个质押者。在两个合约中拆分支付功能的原因有两个:

展示在真实世界的用例中代币支付拆分合约的使用

确保代币支付拆分合约足够灵活,任何人都可以选择并集成到自己的项目中

OpenZeppelin已有一个名为PaymentSplitter

SafeERC20

paymentToken是我们用于支付的ERC20代币的地址。

_totalShares提供来自所有收款人的份额相加。

_totalTokenReleased是已支付给所有收款人的支付代币总额。

_payees提供了当前所有收款人地址的数组。

_shares是收款人地址与分配给他们的份额数量的映射。

_tokenReleased是收款人地址到支付代币数量的映射。

现在放置一个接受三个参数的构造函数。第一个参数是我们希望在合约部署中初始化的收款人的数组。第二个参数是每个收款人的份额数组。第三个是将用于支付的ERC20代币的地址。

pragma solidity 0

俄罗斯为国际支付创建了一个区块链平台以取代Swift系统:金色财经报道,Watcher

function shares(address account) public view returns (uint256) { ? ?return _shares[account];}function payee(uint256 index) public view returns (address) { ? ?return _payees[index];}

现在我们将创建用于添加收款人的函数。

pragma solidity 0

_addPayee是我们在构造函数中调用的用于设置收款人数组的函数。这个函数有两个参数,收款人的帐户和与其相关的份额数量。然后它会检查账户是否为零地址,份额是否大于零,以及该账户是否已经注册为收款人。如果所有检查都通过,那么我们将数据添加到各自的变量中。

现在让我们添加一个函数来支持将代币分发给收款人。

pragma solidity 0

Release是一个任何人都可以调用的函数,它接受一个现有收款人帐户的参数。来分析一下这个函数中发生了什么。首先,它检查帐户是否有分配给它的份额。然后,它创建一个名为tokenTotalReceived的变量,该变量将合约的当前代币余额与之前释放的代币总数相加。创建另一个称为payment的变量,该变量确定收到的代币总额中有多少是欠账户的,然后减去多少已经释放到账户。然后,一个require语句检查当前支付金额是否大于零(即,当前是否欠下了更多代币)。如果该检查通过,则更新账户的tokenReleased,并更新totalTokenReleased。最后,支付给账户的代币金额被转账。

Nerve网络新增UNI跨链,并在NerveDEX创建UNI/USDT交易对:据官方微博消息,NerveNetwork新增加UNI(Uniswap)跨链,并且在NerveDEX上创建UNI/USDT交易对。NULS生态ENVT、ALEPH、OBEE、DATT、MXT等均已上线Uniswap,参加过Uniswap流动性交易的用户请记得登陆Uniswap领取UNI空投。[2020/9/17]

现在函数已经就位了!但是这个合约还有一件事要做....事件!

我们将在合约中添加两个事件,将事件添加到合约顶部是一个良好的实践。

pragma solidity 0

function release(address account) public virtual { ? ?///existingFunctionCode ? ?emit PaymentReleased(account, payment);}

现在代币支付拆分合约已经建立!为了理解这在真实场景中是如何工作的,让我们创建一个模拟池合约,它将导入代币支付拆分器。

这个合约不会很复杂,因为我们只是想演示如何集成代币支付拆分器。这个合约定期收到我们想分发给收款人列表的特定ERC20代币。这个ERC20代币可以通过不同的场景到达,比如用户存款或来自另一个智能合约的重定向费用。在现实生活中,根据不同的项目,可能会有一个更复杂的合约,包含更多的功能来满足用户的用例。

在合约文件夹中,创建一个名为 MockPool.sol 的新文件。然后添加以下代码。

pragma solidity ^0.8.0;import " ? ?function drainTo(address _transferTo, address _token) public onlyOwner { ? ? ? ?require( ? ? ? ?_token != paymentToken, ? ? ? ?"MockPool: Token to drain is PaymentToken" ? ? ? ?); ? ? ? ?uint256 balance = IERC20(_token).balanceOf(address(this)); ? ? ? ?require(balance > 0, "MockPool: Token to drain balance is 0"); ? ? ? ?IERC20(_token).safeTransfer(_transferTo, balance); ? ?}}

动态 | Perlin 将与星展银行、IMDA 合作创建基于区块链的贸易平台:总部位于新加坡的星展银行与跨国贸易公司 Trafigura Group 合作利用区块链技术促进全球贸易,智能合约平台 Perlin 将支持其创建 ICC TradeFlow 区块链贸易平台,ICC TradeFlow 平台将基于由新加坡信息通信媒体发展管理局(IMDA)提供的贸易信任网络基础设。星展集团(DBS)表示,新平台将连接来自不同国家的合作伙伴,将端到端的贸易文件传输时间从 45 天缩短至仅 20 天。该平台的首笔交易将处理非洲和中国的 2000 万美元的铁矿石贸易。[2019/11/9]

在这份合约中,导入三样东西。首先是OpenZeppelin的Ownable实用程序,它在某些函数上使用唯一的onlyOwner 修饰符。第二个是SafeERC20,它允许安全的ERC20代币转账,正如将在合约中看到。第三个是我们的TokenPaymentSplitter合约。

在MockPool构造函数中,我们需要TokenPaymentSplitter提供相同的三个参数,我们只是将它们传递给我们继承的合约。

在这个合约中添加了另一个函数,drainTo。它实际上与TokenPaymentSplitter合约没有任何关系。它只是在另一个没有设置为支付代币的ERC20代币被发送到池时的一种安全机制,然后有一种方法让合约所有者释放该代币。

测试智能合约与创建它们同样重要。这些合约处理的资产通常是属于其他人的,所以作为开发人员,我们有责任确保这些资产按照他们应该的方式工作,并且我们的测试可以覆盖几乎所有的边缘情况。

将在这里进行的测试是一些示例,以显示TokenPaymentSplitter智能合约按照我们的预期工作。在处理自己的项目时,可能希望创建专门适合自己的用例的测试。

为了支持我们的测试,我们希望包含一个ERC20代币,为此,我们将创建一个新的solididity文件,该文件导入OpenZepplin ERC20模板以供我们的测试使用。在合约文件夹中,创建一个名为Imports.sol 的新文件,并包括以下代码:

pragma solidity ^0.8.0;import "

现在,在test文件夹中创建一个名为test.js的文件。在这个文件的顶部,我们将导入支持我们的测试的包。

const { expect } = require('chai')const { ethers } = require('hardhat')

现在,为了设置测试,我们将首先创建必要的变量,创建beforeEach函数,该函数在每次测试之前调用,并创建一个空的 describe 函数,该函数将很快包含我们的测试。

describe('TokenPaymentSplitter Tests', () => {let deployerlet account1let account2let account3let account4let testPaymentTokenlet mockPoolbeforeEach(async () => { ? ?[deployer, account1, account2, account3, account4] = await ethers.getSigners() ? ?const TestPaymentToken = await ethers.getContractFactory('ERC20PresetMinterPauser') ? ?testPaymentToken = await TestPaymentToken.deploy('TestPaymentToken', 'TPT') ? ?await testPaymentToken.deployed(。)describe('Add payees with varying amounts and distribute payments', async () => {}}

声音 | Weiss Ratings:PayPal创建类似加密货币的在线货币但最终以失败告终:韦氏评级(Weiss Ratings)发推称:PayPal联合创始人承认,他们曾试图创建一种类似于加密货币的全球在线货币。他们不知道钱是怎么运作的,最后,PayPal变成了一个初级银行,一败涂地。这可能就是一些PayPal创始人如此喜欢BTC的原因吧。[2019/2/2]

在这些部分就位后,让我们进入这些测试的核心部分!

在我们的第一个测试中,我们想看看当我们部署一个包含平均分配份额的收款人列表的合约时会发生什么。下面是测试代码。

it('payment token is distributed evenly to multiple payees', async () => { ? ?payeeAddressArray = [account1.address, account2.address, account3.address, account4.address] ? ?payeeShareArray = [10, 10, 10, 10] ? ?const MockPool = await ethers.getContractFactory('MockPool') ? ?mockPool = await MockPool.deploy( ? ? ? ?payeeAddressArray, ? ? ? ?payeeShareArray, ? ? ? ?testPaymentToken.address ? ?) ? ?await mockPool.deployed() ? ?await testPaymentToken.mint(mockPool.address, 100000) ? ?await mockPool ? ? ? ?.connect(account1) ? ? ? ?.release(account1.address) ? ?await mockPool ? ? ? ?.connect(account2) ? ? ? ?.release(account2.address) ? ?await mockPool ? ? ? ?.connect(account3) ? ? ? ?.release(account3.address) ? ?await mockPool ? ? ? ?.connect(account4) ? ? ? ?.release(account4.address) ? ?const account1TokenBalance = await testPaymentToken.balanceOf(account1.address) ? ?const account2TokenBalance = await testPaymentToken.balanceOf(account2.address) ? ?const account3TokenBalance = await testPaymentToken.balanceOf(account3.address) ? ?const account4TokenBalance = await testPaymentToken.balanceOf(account4.address) ? ?expect(account1TokenBalance).to.equal(25000) ? ?expect(account2TokenBalance).to.equal(25000) ? ?expect(account3TokenBalance).to.equal(25000) ? ?expect(account4TokenBalance).to.equal(25000。)

在这个测试中,我们将合约分配给4个收款人,每个人都有10个相同的份额。然后我们向合约发送100000单位的testPaymentToken,并向每个收款人发放付款。在测试中可以注意到,每个收款人都在调用函数来向自己释放代币。

在第二个测试中,我们希望确保即使每个收款人的份额分配不均,数学计算仍然有效。

it('payment token is distributed unevenly to multiple payees', async () => { ? ?payeeAddressArray = [account1.address, account2.address, account3.address, account4.address] ? ?payeeShareArray = [10, 5, 11, 7] ? ?const MockPool = await ethers.getContractFactory('MockPool') ? ?mockPool = await MockPool.deploy( ? ? ? ?payeeAddressArray, ? ? ? ?payeeShareArray, ? ? ? ?testPaymentToken.address ? ?) ? ?await mockPool.deployed() ? ?await testPaymentToken.mint(mockPool.address, 100000) ? ?await mockPool ? ? ? ?.connect(account1) ? ? ? ?.release(account1.address) ? ?await mockPool ? ? ? ?.connect(account2) ? ? ? ?.release(account2.address) ? ?await mockPool ? ? ? ?.connect(account3) ? ? ? ?.release(account3.address) ? ?await mockPool ? ? ? ?.connect(account4) ? ? ? ?.release(account4.address) ? ?const mockPoolTestPaymentTokenBalance = await testPaymentToken.balanceOf( ? ? ? ?mockPool.address ? ?) ? ?const account1TokenBalance = await testPaymentToken.balanceOf(account1.address) ? ?const account2TokenBalance = await testPaymentToken.balanceOf(account2.address) ? ?const account3TokenBalance = await testPaymentToken.balanceOf(account3.address) ? ?const account4TokenBalance = await testPaymentToken.balanceOf(account4.address) ? ?expect(mockPoolTestPaymentTokenBalance).to.equal(1) ? ?expect(account1TokenBalance).to.equal(30303) ? ?expect(account2TokenBalance).to.equal(15151) ? ?expect(account3TokenBalance).to.equal(33333) ? ?expect(account4TokenBalance).to.equal(21212。)

看起来收款人还能拿到钱,但注意到什么了吗?合约中还剩下一个单位的支付代币!由于Solidity没有小数,当它达到最低单位时,它通常会四舍五入,这可能会导致合约尘埃飞扬,就像我们在这里看到的。不过不用担心,因为我们预计未来会有支付代币流入合约,所以它将继续分发。

这与之前的测试类似,不过在资金被释放给收款人之间增加了更多支付代币发送到池中。这表明,随着支付代币不断流入模拟池合约,数学仍然可以确保收款人收到正确的金额。

it('payment token is distributed unevenly to multiple payees with additional payment token sent to pool', async () => { ? ?payeeAddressArray = [account1.address, account2.address, account3.address, account4.address] ? ?payeeShareArray = [10, 5, 11, 7] ? ?const MockPool = await ethers.getContractFactory('MockPool') ? ?mockPool = await MockPool.deploy( ? ? ? ?payeeAddressArray, ? ? ? ?payeeShareArray, ? ? ? ?testPaymentToken.address ? ?) ? ?await mockPool.deployed() ? ?await testPaymentToken.mint(mockPool.address, 100000) ? ?await mockPool ? ? ? ?.connect(account1) ? ? ? ?.release(account1.address) ? ?await mockPool ? ? ? ?.connect(account2) ? ? ? ?.release(account2.address) ? ?await testPaymentToken.mint(mockPool.address, 100000) ? ?await mockPool ? ? ? ?.connect(account3) ? ? ? ?.release(account3.address) ? ?await mockPool ? ? ? ?.connect(account4) ? ? ? ?.release(account4.address) ? ?await mockPool ? ? ? ?.connect(account1) ? ? ? ?.release(account1.address) ? ?await mockPool ? ? ? ?.connect(account2) ? ? ? ?.release(account2.address) ? ?const mockPoolTestPaymentTokenBalance = await testPaymentToken.balanceOf( ? ? ? ?mockPool.address ? ? ? ? ? ?) ? ?const account1TokenBalance = await testPaymentToken.balanceOf(account1.address) ? ?const account2TokenBalance = await testPaymentToken.balanceOf(account2.address) ? ?const account3TokenBalance = await testPaymentToken.balanceOf(account3.address) ? ?const account4TokenBalance = await testPaymentToken.balanceOf(account4.address) ? ?expect(mockPoolTestPaymentTokenBalance).to.equal(1) ? ?expect(account1TokenBalance).to.equal(60606) ? ?expect(account2TokenBalance).to.equal(30303) ? ?expect(account3TokenBalance).to.equal(66666) ? ?expect(account4TokenBalance).to.equal(42424。)

现在所有的测试都就绪了,是时候运行它们了,看看它们是否工作!在项目根文件夹中,使用npx hardhat test启动测试。如果一切都是正确的,那么你应该看到如下图所示的所有绿色格子。

如上所述,我们需要做更多的测试,以确保整个项目/协议按照预期工作,支付拆分器是它的集成部分。这将意味着更多的单元测试来覆盖所有可用的功能,以及更复杂的集成测试,这取决于具体用例。

支付是许多加密协议的一个常见方面,有几种方法可以解决它们。今天我们学习了一种管理支付的方法,尽管用户甚至可以在此合约的基础上构建以满足您的特定需求,如跨多个代币启用支付,添加额外的收款人或移除收款人,或在一个函数调用中同时分发所有支付。

Source:https://medium.com/coinmonks/create-an-erc20-token-payment-splitting-smart-contract-c79436470ccc

标签:PAYTOKETOKKENPAYBITWrapped LUNA TokenJinbi TokenCryptocurrency Top 10 Tokens Index

瑞波币热门资讯
元宇宙:首发 | 作为乌托邦的元宇宙

本文由高承实原创,授权金色财经首发。元宇宙目前被炒得满热。但元宇宙仅仅是一个概念,甚至连一个概念都够不上,仅仅是一些稍微成点体系的想法。有人说元宇宙是世界观,但更合适的界定,我个人认为是乌托邦.

1900/1/1 0:00:00
ETH:OpenSea高管利用“老鼠仓”不当获利 吃瓜网友“链上开扒”

NFT 行业也有瓜吃了。北京时间 9 月 15 日,推特用户 Zuwu 发推指责 OpenSea 产品负责人 Nate Chastain?利用“老鼠仓”不当获利.

1900/1/1 0:00:00
NFT:NFT 估值的方法论:如何发现一个好的 NFT?

NFT 涉及艺术、游戏等等众多门类,是否购买一个 NFT 通常与个人爱好及品味息息相关。然而,作为一类数字资产,资产属性决定了必然有多种方式可以让我们对其进行主观或客观的评价和估值.

1900/1/1 0:00:00
区块链:中国政务区块链应用全球领先 并推动社会治理体系持续升级

从以比特币为代表的区块链1.0时代,到以智能合约为基础、以金融应用为核心的区块链2.0时代,再到区块链应用于政务服务和更广泛的各行业,10余年间,我们已经开始进入区块链3.0时代.

1900/1/1 0:00:00
DEF:2021 年 DeFi 趋势和黑客

人们在不知道潜在风险的情况下参与DeFi传奇!去中心化金融,俗称“DeFi”,旨在颠覆传统金融机构,如银行、加密货币交易所,其中大多数都运行在以太坊区块链上.

1900/1/1 0:00:00
元宇宙:虚实中的元宇宙 未到颠覆世界之时

在概念炒作之后,市场将审视相关技术革新、硬件设备和上游产业的进展速度,并开始对元宇宙蓝图具象化的漫长等待。资本追逐新而又玄的概念时永不眠.

1900/1/1 0:00:00