From 792511165c95a8cfbd0a65e0af346da2f72394b3 Mon Sep 17 00:00:00 2001 From: cgewecke Date: Tue, 9 Jun 2020 21:48:10 -0700 Subject: [PATCH 1/2] Add failing transfer stipend test --- .../projects/solc-6/contracts/B_Wallet.sol | 24 +++++++++++++++ .../projects/solc-6/test/b_wallet.js | 30 +++++++++++++++++++ test/units/buidler/standard.js | 7 ++++- 3 files changed, 60 insertions(+), 1 deletion(-) create mode 100644 test/integration/projects/solc-6/contracts/B_Wallet.sol create mode 100644 test/integration/projects/solc-6/test/b_wallet.js diff --git a/test/integration/projects/solc-6/contracts/B_Wallet.sol b/test/integration/projects/solc-6/contracts/B_Wallet.sol new file mode 100644 index 00000000..fc818665 --- /dev/null +++ b/test/integration/projects/solc-6/contracts/B_Wallet.sol @@ -0,0 +1,24 @@ +pragma solidity ^0.6.0; + +contract B_Wallet { + + event Deposit(address indexed _sender, uint _value, bytes data); + + function transferPayment(uint payment, address payable recipient) public { + recipient.transfer(payment); + } + + function sendPayment(uint payment, address payable recipient) public { + require(recipient.send(payment), 'sendPayment failed'); + } + + function getBalance() public view returns(uint){ + return address(this).balance; + } + + receive() external payable + { + if (msg.value > 0) + emit Deposit(msg.sender, msg.value, msg.data); + } +} diff --git a/test/integration/projects/solc-6/test/b_wallet.js b/test/integration/projects/solc-6/test/b_wallet.js new file mode 100644 index 00000000..da91d33a --- /dev/null +++ b/test/integration/projects/solc-6/test/b_wallet.js @@ -0,0 +1,30 @@ +const Wallet = artifacts.require('B_Wallet'); + +contract('B_Wallet', accounts => { + it('should should allow transfers and sends', async () => { + const walletA = await Wallet.new(); + const walletB = await Wallet.new(); + + await walletA.sendTransaction({ + value: web3.utils.toBN(500), from: accounts[0], + }); + + await walletA.sendPayment(50, walletB.address, { + from: accounts[0], + }); + + await walletA.transferPayment(50, walletB.address, { + from: accounts[0], + }); + + // Also try transferring 0, for branch hit + await walletA.transferPayment(0, walletB.address, { + from: accounts[0], + }); + + // Throws invalid opcode if compiled w/ solc >= 0.5.14 & default EVM version + const balance = await walletB.getBalance(); + assert.equal(balance.toNumber(), 100); + }); +}); + diff --git a/test/units/buidler/standard.js b/test/units/buidler/standard.js index 1eb66fd3..be127d6b 100644 --- a/test/units/buidler/standard.js +++ b/test/units/buidler/standard.js @@ -289,7 +289,12 @@ describe('Buidler Plugin: standard use cases', function() { { file: mock.pathToContract(buidlerConfig, 'ContractB.sol'), pct: 0, - } + }, + { + file: mock.pathToContract(buidlerConfig, 'B_Wallet.sol'), + pct: 100, + }, + ]; verify.lineCoverage(expected); From ea5ed9aee636b8d6465d8ab50b4842f2df72d8a9 Mon Sep 17 00:00:00 2001 From: cgewecke Date: Wed, 10 Jun 2020 09:22:13 -0700 Subject: [PATCH 2/2] Stop instrumenting receive() for statements / fns (to save gas) --- lib/parse.js | 8 +++++++- lib/registrar.js | 8 +++++++- .../projects/solc-6/contracts/B_Wallet.sol | 12 ++++++------ 3 files changed, 20 insertions(+), 8 deletions(-) diff --git a/lib/parse.js b/lib/parse.js index f7bfca39..29706d75 100644 --- a/lib/parse.js +++ b/lib/parse.js @@ -105,9 +105,15 @@ parse.ForStatement = function(contract, expression) { parse.FunctionDefinition = function(contract, expression) { parse.Modifiers(contract, expression.modifiers); if (expression.body) { - register.functionDeclaration(contract, expression); + // Skip fn & statement instrumentation for `receive` methods to + // minimize gas distortion + (expression.name === null && expression.isReceiveEther) + ? register.trackStatements = false + : register.functionDeclaration(contract, expression); + parse[expression.body.type] && parse[expression.body.type](contract, expression.body); + register.trackStatements = true; } }; diff --git a/lib/registrar.js b/lib/registrar.js index aaa794ec..8ba78847 100644 --- a/lib/registrar.js +++ b/lib/registrar.js @@ -5,7 +5,11 @@ */ class Registrar { - constructor(){} + constructor(){ + // Sometimes we don't want to inject statements + // because they're an unnecessary expense. ex: `receive` + this.trackStatements = true; + } /** * Adds injection point to injection points map @@ -27,6 +31,8 @@ class Registrar { * @param {Object} expression AST node */ statement(contract, expression) { + if (!this.trackStatements) return; + const startContract = contract.instrumented.slice(0, expression.range[0]); const startline = ( startContract.match(/\n/g) || [] ).length + 1; const startcol = expression.range[0] - startContract.lastIndexOf('\n') - 1; diff --git a/test/integration/projects/solc-6/contracts/B_Wallet.sol b/test/integration/projects/solc-6/contracts/B_Wallet.sol index fc818665..d47470df 100644 --- a/test/integration/projects/solc-6/contracts/B_Wallet.sol +++ b/test/integration/projects/solc-6/contracts/B_Wallet.sol @@ -4,6 +4,12 @@ contract B_Wallet { event Deposit(address indexed _sender, uint _value, bytes data); + receive() external payable + { + if (msg.value > 0) + emit Deposit(msg.sender, msg.value, msg.data); + } + function transferPayment(uint payment, address payable recipient) public { recipient.transfer(payment); } @@ -15,10 +21,4 @@ contract B_Wallet { function getBalance() public view returns(uint){ return address(this).balance; } - - receive() external payable - { - if (msg.value > 0) - emit Deposit(msg.sender, msg.value, msg.data); - } }