在加密货币的几乎每个领域,支付都是一个反复出现的话题,特别是向多个质押者提供支付。例如,DAO希望为多个计划提供资金,DEX希望合并向某些参与者分配交易费用,或者团队希望将代币作为月薪分发给团队成员。
智能合约使我们能够自动化这些类型的支付功能,这就限制了人工管理支付所导致的潜在错误,并允许我们将宝贵的时间花在其他生产性任务上。
今天,我们将学习如何创建自己的ERC20代币支付拆分器,它可以合并到任何项目中!
先决条件和设置
下面的内容要求你对Solidity有点熟悉,不过任何人都可以学习。
项目架构
我们将创建两个合约。第一个将是ERC20代币支付拆分智能合约,第二个将是模拟池智能合约。ERC20代币支付拆分器智能合约将是抽象的,并持有用于管理收付方及其各自支付部分的逻辑和数据。模拟池将继承ERC20代币支付拆分器,以便我们可以自动将支付分发给多个质押者。在两个合约中拆分支付功能的原因有两个:
展示在真实世界的用例中代币支付拆分合约的使用
确保代币支付拆分合约足够灵活,任何人都可以选择并集成到自己的项目中
OpenZeppelin已有一个名为PaymentSplitter
SafeERC20
paymentToken是我们用于支付的ERC20代币的地址。
_totalShares提供来自所有收款人的份额相加。
_totalTokenReleased是已支付给所有收款人的支付代币总额。
_payees提供了当前所有收款人地址的数组。
_shares是收款人地址与分配给他们的份额数量的映射。
_tokenReleased是收款人地址到支付代币数量的映射。
现在放置一个接受三个参数的构造函数。第一个参数是我们希望在合约部署中初始化的收款人的数组。第二个参数是每个收款人的份额数组。第三个是将用于支付的ERC20代币的地址。
pragmasolidity0
第一部完全由BTC筹资创建的纪录片将于明日推出:9月15日消息,第一部完全由BTC筹资创建的纪录片将于明日9:00在Swan Bitcoin YouTube频道上推出。该纪录片名为“这台绿色机器:比特币和清洁能源的未来”,涉及了能源消耗问题和围绕比特币的影响。这部纪录片提出,比特币及其基于工作证明的共识算法比美元更好、更有效地利用能源。此外,该纪录片也试图回答比特币如何有助于实现“碳零”排放的问题。此外,纪录片背后的制作团队目前正在为其下一部名为TRUST的纪录片接受捐款。与其第一部纪录片类似,这个项目将完全由比特币和其社区提供资金。(Bitcoinist)[2021/9/15 23:26:05]
构造函数包含一个require语句,以确保两个数组具有相同的长度,以便每个收款人都有分配给他们的份额。还有另一个require语句,以确保合约初始化与至少有一个收款人。
还有一个for循环,它将每个收款人及其份额分配我们上面创建的变量。这是通过一个名为_addPayee的函数完成的,我们将很快创建这个函数。
构造函数就绪后,再添加几个函数来调用和获取合约变量。
pragmasolidity0
functionshares(addressaccount)publicviewreturns(uint256){??return_shares;}functionpayee(uint256index)publicviewreturns(address){??return_payees;}
现在我们将创建用于添加收款人的函数。
pragmasolidity0
_addPayee是我们在构造函数中调用的用于设置收款人数组的函数。这个函数有两个参数,收款人的帐户和与其相关的份额数量。然后它会检查账户是否为零地址,份额是否大于零,以及该账户是否已经注册为收款人。如果所有检查都通过,那么我们将数据添加到各自的变量中。
现在让我们添加一个函数来支持将代币分发给收款人。
pragmasolidity0
动态 | 比特币艺术家Bnoiit
functionrelease(addressaccount)publicvirtual{??///existingFunctionCode??emitPaymentReleased(account,payment);}
现在代币支付拆分合约已经建立!为了理解这在真实场景中是如何工作的,让我们创建一个模拟池合约,它将导入代币支付拆分器。
创建模拟池合约
这个合约不会很复杂,因为我们只是想演示如何集成代币支付拆分器。这个合约定期收到我们想分发给收款人列表的特定ERC20代币。这个ERC20代币可以通过不同的场景到达,比如用户存款或来自另一个智能合约的重定向费用。在现实生活中,根据不同的项目,可能会有一个更复杂的合约,包含更多的功能来满足用户的用例。
在合约文件夹中,创建一个名为MockPool.sol的新文件。然后添加以下代码。
pragmasolidity^0.8.0;import"??functiondrainTo(address_transferTo,address_token)publiconlyOwner{????require(????_token!=paymentToken,????"MockPool:TokentodrainisPaymentToken"????);????uint256balance=IERC20(_token).balanceOf(address(this));????require(balance>0,"MockPool:Tokentodrainbalanceis0");????IERC20(_token).safeTransfer(_transferTo,balance);??}}
动态 | Kraken交易所宣布收购Interchange 以创建加密交易平台:据Business Wire日消息,Kraken交易所今天宣布收购Interchange。Interchange是一家会计、对账和报告服务提供商,为加密货币对冲基金、资产经理和基金管理员提供服务。该交易创造了世界上第一个端到端的加密交易和对账平台,并将Interchange联合创始人和加密老兵Dan Held和Clark Moody带到Kraken。[2019/7/31]
在这份合约中,导入三样东西。首先是OpenZeppelin的Ownable实用程序,它在某些函数上使用唯一的onlyOwner修饰符。第二个是SafeERC20,它允许安全的ERC20代币转账,正如将在合约中看到。第三个是我们的TokenPaymentSplitter合约。
在MockPool构造函数中,我们需要TokenPaymentSplitter提供相同的三个参数,我们只是将它们传递给我们继承的合约。
在这个合约中添加了另一个函数,drainTo。它实际上与TokenPaymentSplitter合约没有任何关系。它只是在另一个没有设置为支付代币的ERC20代币被发送到池时的一种安全机制,然后有一种方法让合约所有者释放该代币。
测试合约
测试智能合约与创建它们同样重要。这些合约处理的资产通常是属于其他人的,所以作为开发人员,我们有责任确保这些资产按照他们应该的方式工作,并且我们的测试可以覆盖几乎所有的边缘情况。
将在这里进行的测试是一些示例,以显示TokenPaymentSplitter智能合约按照我们的预期工作。在处理自己的项目时,可能希望创建专门适合自己的用例的测试。
为了支持我们的测试,我们希望包含一个ERC20代币,为此,我们将创建一个新的solididity文件,该文件导入OpenZepplinERC20模板以供我们的测试使用。在合约文件夹中,创建一个名为Imports.sol的新文件,并包括以下代码:
pragmasolidity^0.8.0;import"
现在,在test文件夹中创建一个名为test.js的文件。在这个文件的顶部,我们将导入支持我们的测试的包。
const{expect}=require('chai')const{ethers}=require('hardhat')
现在,为了设置测试,我们将首先创建必要的变量,创建beforeEach函数,该函数在每次测试之前调用,并创建一个空的describe函数,该函数将很快包含我们的测试。
describe('TokenPaymentSplitterTests',()=>{letdeployerletaccount1letaccount2letaccount3letaccount4lettestPaymentTokenletmockPoolbeforeEach(async()=>{??=awaitethers.getSigners()??constTestPaymentToken=awaitethers.getContractFactory('ERC20PresetMinterPauser')??testPaymentToken=awaitTestPaymentToken.deploy('TestPaymentToken','TPT')??awaittestPaymentToken.deployed(。)describe('Addpayeeswithvaryingamountsanddistributepayments',async()=>{}}
动态 | 区块链创业公司Editional推出以太坊应用 允许用户创建和交易收藏品:据The Block消息,纽约区块链创业公司Editional推出以太坊应用程序,允许用户在区块链上创建、存储交易收藏品。[2019/5/1]
在这些部分就位后,让我们进入这些测试的核心部分!
支付代币平均分配给多个收款人
在我们的第一个测试中,我们想看看当我们部署一个包含平均分配份额的收款人列表的合约时会发生什么。下面是测试代码。
it('paymenttokenisdistributedevenlytomultiplepayees',async()=>{??payeeAddressArray=??payeeShareArray=??constMockPool=awaitethers.getContractFactory('MockPool')??mockPool=awaitMockPool.deploy(????payeeAddressArray,????payeeShareArray,????testPaymentToken.address??)??awaitmockPool.deployed()??awaittestPaymentToken.mint(mockPool.address,100000)??awaitmockPool????.connect(account1)????.release(account1.address)??awaitmockPool????.connect(account2)????.release(account2.address)??awaitmockPool????.connect(account3)????.release(account3.address)??awaitmockPool????.connect(account4)????.release(account4.address)??constaccount1TokenBalance=awaittestPaymentToken.balanceOf(account1.address)??constaccount2TokenBalance=awaittestPaymentToken.balanceOf(account2.address)??constaccount3TokenBalance=awaittestPaymentToken.balanceOf(account3.address)??constaccount4TokenBalance=awaittestPaymentToken.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('paymenttokenisdistributedunevenlytomultiplepayees',async()=>{??payeeAddressArray=??payeeShareArray=??constMockPool=awaitethers.getContractFactory('MockPool')??mockPool=awaitMockPool.deploy(????payeeAddressArray,????payeeShareArray,????testPaymentToken.address??)??awaitmockPool.deployed()??awaittestPaymentToken.mint(mockPool.address,100000)??awaitmockPool????.connect(account1)????.release(account1.address)??awaitmockPool????.connect(account2)????.release(account2.address)??awaitmockPool????.connect(account3)????.release(account3.address)??awaitmockPool????.connect(account4)????.release(account4.address)??constmockPoolTestPaymentTokenBalance=awaittestPaymentToken.balanceOf(????mockPool.address??)??constaccount1TokenBalance=awaittestPaymentToken.balanceOf(account1.address)??constaccount2TokenBalance=awaittestPaymentToken.balanceOf(account2.address)??constaccount3TokenBalance=awaittestPaymentToken.balanceOf(account3.address)??constaccount4TokenBalance=awaittestPaymentToken.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('paymenttokenisdistributedunevenlytomultiplepayeeswithadditionalpaymenttokensenttopool',async()=>{??payeeAddressArray=??payeeShareArray=??constMockPool=awaitethers.getContractFactory('MockPool')??mockPool=awaitMockPool.deploy(????payeeAddressArray,????payeeShareArray,????testPaymentToken.address??)??awaitmockPool.deployed()??awaittestPaymentToken.mint(mockPool.address,100000)??awaitmockPool????.connect(account1)????.release(account1.address)??awaitmockPool????.connect(account2)????.release(account2.address)??awaittestPaymentToken.mint(mockPool.address,100000)??awaitmockPool????.connect(account3)????.release(account3.address)??awaitmockPool????.connect(account4)????.release(account4.address)??awaitmockPool????.connect(account1)????.release(account1.address)??awaitmockPool????.connect(account2)????.release(account2.address)??constmockPoolTestPaymentTokenBalance=awaittestPaymentToken.balanceOf(????mockPool.address??????)??constaccount1TokenBalance=awaittestPaymentToken.balanceOf(account1.address)??constaccount2TokenBalance=awaittestPaymentToken.balanceOf(account2.address)??constaccount3TokenBalance=awaittestPaymentToken.balanceOf(account3.address)??constaccount4TokenBalance=awaittestPaymentToken.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。)
现在所有的测试都就绪了,是时候运行它们了,看看它们是否工作!在项目根文件夹中,使用npxhardhattest启动测试。如果一切都是正确的,那么你应该看到如下图所示的所有绿色格子。
如上所述,我们需要做更多的测试,以确保整个项目/协议按照预期工作,支付拆分器是它的集成部分。这将意味着更多的单元测试来覆盖所有可用的功能,以及更复杂的集成测试,这取决于具体用例。
总结
支付是许多加密协议的一个常见方面,有几种方法可以解决它们。今天我们学习了一种管理支付的方法,尽管用户甚至可以在此合约的基础上构建以满足您的特定需求,如跨多个代币启用支付,添加额外的收款人或移除收款人,或在一个函数调用中同时分发所有支付。
Source:https://medium.com/coinmonks/create-an-erc20-token-payment-splitting-smart-contract-c79436470ccc
标签:KENPAYTOKENTOKEkraken交易所平台币KodexPayfruittokenFitR Metaverse Token
近日,FTX创始人SamBankman-Fried接受了Cointelegraph的采访,在采访中,SBF谈到了FTX的下一步营销动作,对近期比特币下跌的看法,对DeFi和传统金融关系的看法.
1900/1/1 0:00:00FTX和FTX.US今宣布和NBA球星斯蒂芬库里达成长期合作伙伴关系。库里将获得FTXTradingLimited的股权,并担任FTX的品牌大使,此外,库里的部分薪资会以加密货币支付.
1900/1/1 0:00:00北京时间9月13日消息,据DuneAnalytics统计数据显示,以太坊Layer2网络Arbitrum的资产桥TVL已突破21亿美元,其7天的TVL增长率超过了3200%,而在同期.
1900/1/1 0:00:00当游戏中的真钱交易模式不起作用时,我们发现「边玩边赚」开始兴起。那么,从@AxieInfinity@Diablo@EveOnline@CSGO这些游戏中又可以相互学习到什么有价值的东西呢?我将在.
1900/1/1 0:00:00Rlink作为虎符智能链HSC上首个去中心化邀请和返佣协议,于2021年9月17日正式上线并进入首轮合约调用奖励通缩.
1900/1/1 0:00:00金色财经区块链9月23日讯“边玩边赚”游戏出现至今已经有一段时间了,但感觉它直到最近才真正走到我们面前.
1900/1/1 0:00:00