|
|
|
@ -2,97 +2,104 @@ const { ethGetBalance, ethSendTransaction } = require('./helpers/web3'); |
|
|
|
|
const expectEvent = require('./helpers/expectEvent'); |
|
|
|
|
const { assertRevert } = require('./helpers/assertRevert'); |
|
|
|
|
|
|
|
|
|
const SecureInvariantTargetBounty = artifacts.require('SecureInvariantTargetBounty'); |
|
|
|
|
const InsecureInvariantTargetBounty = artifacts.require('InsecureInvariantTargetBounty'); |
|
|
|
|
const BreakInvariantBountyMock = artifacts.require('BreakInvariantBountyMock'); |
|
|
|
|
const TargetMock = artifacts.require('TargetMock'); |
|
|
|
|
|
|
|
|
|
require('chai') |
|
|
|
|
.use(require('chai-bignumber')(web3.BigNumber)) |
|
|
|
|
.should(); |
|
|
|
|
|
|
|
|
|
const sendReward = async (from, to, value) => ethSendTransaction({ |
|
|
|
|
from, |
|
|
|
|
to, |
|
|
|
|
value, |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
const reward = new web3.BigNumber(web3.toWei(1, 'ether')); |
|
|
|
|
|
|
|
|
|
contract('BreakInvariantBounty', function ([_, owner, researcher, nonTarget]) { |
|
|
|
|
context('against secure contract', function () { |
|
|
|
|
contract('BreakInvariantBounty', function ([_, owner, researcher, anyone, nonTarget]) { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
this.bounty = await SecureInvariantTargetBounty.new({ from: owner }); |
|
|
|
|
this.bounty = await BreakInvariantBountyMock.new({ from: owner }); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
it('can set reward', async function () { |
|
|
|
|
await sendReward(owner, this.bounty.address, reward); |
|
|
|
|
|
|
|
|
|
const balance = await ethGetBalance(this.bounty.address); |
|
|
|
|
balance.should.be.bignumber.equal(reward); |
|
|
|
|
await ethSendTransaction({ from: owner, to: this.bounty.address, value: reward }); |
|
|
|
|
(await ethGetBalance(this.bounty.address)).should.be.bignumber.equal(reward); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
context('with reward', function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
const result = await this.bounty.createTarget({ from: researcher }); |
|
|
|
|
const event = expectEvent.inLogs(result.logs, 'TargetCreated'); |
|
|
|
|
|
|
|
|
|
this.targetAddress = event.args.createdAddress; |
|
|
|
|
await ethSendTransaction({ from: owner, to: this.bounty.address, value: reward }); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
await sendReward(owner, this.bounty.address, reward); |
|
|
|
|
describe('destroy', function () { |
|
|
|
|
it('returns all balance to the owner', async function () { |
|
|
|
|
const ownerPreBalance = await ethGetBalance(owner); |
|
|
|
|
await this.bounty.destroy({ from: owner, gasPrice: 0 }); |
|
|
|
|
const ownerPostBalance = await ethGetBalance(owner); |
|
|
|
|
|
|
|
|
|
const balance = await ethGetBalance(this.bounty.address); |
|
|
|
|
balance.should.be.bignumber.equal(reward); |
|
|
|
|
(await ethGetBalance(this.bounty.address)).should.be.bignumber.equal(0); |
|
|
|
|
ownerPostBalance.sub(ownerPreBalance).should.be.bignumber.equal(reward); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
it('cannot claim reward', async function () { |
|
|
|
|
await assertRevert( |
|
|
|
|
this.bounty.claim(this.targetAddress, { from: researcher }), |
|
|
|
|
); |
|
|
|
|
it('reverts when called by anyone', async function () { |
|
|
|
|
await assertRevert(this.bounty.destroy({ from: anyone })); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
describe('claim', function () { |
|
|
|
|
it('is initially unclaimed', async function () { |
|
|
|
|
(await this.bounty.claimed()).should.equal(false); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
context('against broken contract', function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
this.bounty = await InsecureInvariantTargetBounty.new(); |
|
|
|
|
it('can create claimable target', async function () { |
|
|
|
|
const { logs } = await this.bounty.createTarget({ from: researcher }); |
|
|
|
|
expectEvent.inLogs(logs, 'TargetCreated'); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
const result = await this.bounty.createTarget({ from: researcher }); |
|
|
|
|
const event = expectEvent.inLogs(result.logs, 'TargetCreated'); |
|
|
|
|
context('with target', async function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
const { logs } = await this.bounty.createTarget({ from: researcher }); |
|
|
|
|
const event = expectEvent.inLogs(logs, 'TargetCreated'); |
|
|
|
|
this.target = TargetMock.at(event.args.createdAddress); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
this.targetAddress = event.args.createdAddress; |
|
|
|
|
await sendReward(owner, this.bounty.address, reward); |
|
|
|
|
context('before exploiting vulnerability', async function () { |
|
|
|
|
it('reverts when claiming reward', async function () { |
|
|
|
|
await assertRevert(this.bounty.claim(this.target.address, { from: researcher })); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
it('can claim reward', async function () { |
|
|
|
|
await this.bounty.claim(this.targetAddress, { from: researcher }); |
|
|
|
|
const claim = await this.bounty.claimed(); |
|
|
|
|
context('after exploiting vulnerability', async function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
await this.target.exploitVulnerability({ from: researcher }); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
claim.should.equal(true); |
|
|
|
|
it('sends the reward to the researcher', async function () { |
|
|
|
|
await this.bounty.claim(this.target.address, { from: anyone }); |
|
|
|
|
|
|
|
|
|
const researcherPrevBalance = await ethGetBalance(researcher); |
|
|
|
|
const researcherPreBalance = await ethGetBalance(researcher); |
|
|
|
|
await this.bounty.withdrawPayments(researcher); |
|
|
|
|
const researcherPostBalance = await ethGetBalance(researcher); |
|
|
|
|
|
|
|
|
|
await this.bounty.withdrawPayments(researcher, { gasPrice: 0 }); |
|
|
|
|
const updatedBalance = await ethGetBalance(this.bounty.address); |
|
|
|
|
updatedBalance.should.be.bignumber.equal(0); |
|
|
|
|
researcherPostBalance.sub(researcherPreBalance).should.be.bignumber.equal(reward); |
|
|
|
|
(await ethGetBalance(this.bounty.address)).should.be.bignumber.equal(0); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
const researcherCurrBalance = await ethGetBalance(researcher); |
|
|
|
|
researcherCurrBalance.sub(researcherPrevBalance).should.be.bignumber.equal(reward); |
|
|
|
|
context('after claiming', async function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
await this.bounty.claim(this.target.address, { from: researcher }); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
it('cannot claim reward from non-target', async function () { |
|
|
|
|
await assertRevert( |
|
|
|
|
this.bounty.claim(nonTarget, { from: researcher }) |
|
|
|
|
); |
|
|
|
|
it('is claimed', async function () { |
|
|
|
|
(await this.bounty.claimed()).should.equal(true); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
context('reward claimed', function () { |
|
|
|
|
beforeEach(async function () { |
|
|
|
|
await this.bounty.claim(this.targetAddress, { from: researcher }); |
|
|
|
|
it('no longer accepts rewards', async function () { |
|
|
|
|
await assertRevert(ethSendTransaction({ from: owner, to: this.bounty.address, value: reward })); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
it('should no longer be payable', async function () { |
|
|
|
|
await assertRevert( |
|
|
|
|
sendReward(owner, this.bounty.address, reward) |
|
|
|
|
); |
|
|
|
|
context('with non-target', function () { |
|
|
|
|
it('reverts when claiming reward', async function () { |
|
|
|
|
await assertRevert(this.bounty.claim(nonTarget, { from: researcher })); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|
}); |
|
|
|
|