mirror of
https://github.com/status-im/dagger-contracts.git
synced 2025-01-25 22:09:55 +00:00
c3e85c675a
Adds nonce to storage request to ensure uniqueness.
350 lines
10 KiB
JavaScript
350 lines
10 KiB
JavaScript
const { expect } = require("chai")
|
|
const { ethers } = require("hardhat")
|
|
const { hashRequest, hashBid, sign } = require("./marketplace")
|
|
|
|
describe("Storage Contracts", function () {
|
|
|
|
const duration = 31 * 24 * 60 * 60 // 31 days
|
|
const size = 1 * 1024 * 1024 * 1024 // 1 Gigabyte
|
|
const contentHash = ethers.utils.sha256("0xdeadbeef") // hash of content
|
|
const proofPeriod = 8 // 8 blocks ≈ 2 minutes
|
|
const proofTimeout = 4 // 4 blocks ≈ 1 minute
|
|
const price = 42
|
|
const nonce = ethers.utils.randomBytes(32)
|
|
|
|
var contracts
|
|
var client, host
|
|
var bidExpiry
|
|
var requestHash, bidHash
|
|
var id
|
|
|
|
beforeEach(async function () {
|
|
[client, host] = await ethers.getSigners()
|
|
let StorageContracts = await ethers.getContractFactory("StorageContracts")
|
|
contracts = await StorageContracts.deploy()
|
|
requestHash = hashRequest(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce
|
|
)
|
|
bidExpiry = Math.round(Date.now() / 1000) + 60 * 60 // 1 hour from now
|
|
bidHash = hashBid(requestHash, bidExpiry, price)
|
|
id = bidHash
|
|
})
|
|
|
|
describe("when properly instantiated", function () {
|
|
|
|
beforeEach(async function () {
|
|
await contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash)
|
|
)
|
|
})
|
|
|
|
it("has a duration", async function () {
|
|
expect(await contracts.duration(id)).to.equal(duration)
|
|
})
|
|
|
|
it("contains the size of the data that is to be stored", async function () {
|
|
expect(await contracts.size(id)).to.equal(size)
|
|
})
|
|
|
|
it("contains the hash of the data that is to be stored", async function () {
|
|
expect(await contracts.contentHash(id)).to.equal(contentHash)
|
|
})
|
|
|
|
it("has a price", async function () {
|
|
expect(await contracts.price(id)).to.equal(price)
|
|
})
|
|
|
|
it("knows the host that provides the storage", async function () {
|
|
expect(await contracts.host(id)).to.equal(await host.getAddress())
|
|
})
|
|
|
|
it("has an average time between proofs (in blocks)", async function (){
|
|
expect(await contracts.proofPeriod(id)).to.equal(proofPeriod)
|
|
})
|
|
|
|
it("has a proof timeout (in blocks)", async function (){
|
|
expect(await contracts.proofTimeout(id)).to.equal(proofTimeout)
|
|
})
|
|
})
|
|
|
|
it("cannot be created when contract id already used", async function () {
|
|
await contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash)
|
|
)
|
|
await expect(contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash)
|
|
)).to.be.revertedWith("A contract with this id already exists")
|
|
})
|
|
|
|
it("cannot be created when client signature is invalid", async function () {
|
|
let invalidHash = hashRequest(
|
|
duration + 1,
|
|
size,
|
|
contentHash,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce
|
|
)
|
|
let invalidSignature = await sign(client, invalidHash)
|
|
await expect(contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
invalidSignature,
|
|
await sign(host, bidHash)
|
|
)).to.be.revertedWith("Invalid signature")
|
|
})
|
|
|
|
it("cannot be created when host signature is invalid", async function () {
|
|
let invalidBid = hashBid(requestHash, bidExpiry, price - 1)
|
|
let invalidSignature = await sign(host, invalidBid)
|
|
await expect(contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
invalidSignature
|
|
)).to.be.revertedWith("Invalid signature")
|
|
})
|
|
|
|
it("cannot be created when proof timeout is too large", async function () {
|
|
let invalidTimeout = 129 // max proof timeout is 128 blocks
|
|
requestHash = hashRequest(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
proofPeriod,
|
|
invalidTimeout,
|
|
nonce
|
|
)
|
|
bidHash = hashBid(requestHash, bidExpiry, price)
|
|
await expect(contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
invalidTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash),
|
|
)).to.be.revertedWith("Invalid proof timeout")
|
|
})
|
|
|
|
it("cannot be created when bid has expired", async function () {
|
|
let expired = Math.round(Date.now() / 1000) - 60 // 1 minute ago
|
|
let bidHash = hashBid(requestHash, expired, price)
|
|
await expect(contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
expired,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash),
|
|
)).to.be.revertedWith("Bid expired")
|
|
})
|
|
|
|
describe("proofs", function () {
|
|
|
|
async function mineBlock() {
|
|
await ethers.provider.send("evm_mine")
|
|
}
|
|
|
|
async function minedBlockNumber() {
|
|
return await ethers.provider.getBlockNumber() - 1
|
|
}
|
|
|
|
async function mineUntilProofIsRequired(id) {
|
|
while (!await contracts.isProofRequired(id, await minedBlockNumber())) {
|
|
mineBlock()
|
|
}
|
|
}
|
|
|
|
async function mineUntilProofTimeout() {
|
|
for (let i=0; i<proofTimeout; i++) {
|
|
mineBlock()
|
|
}
|
|
}
|
|
|
|
beforeEach(async function () {
|
|
await contracts.newContract(
|
|
duration,
|
|
size,
|
|
contentHash,
|
|
price,
|
|
proofPeriod,
|
|
proofTimeout,
|
|
nonce,
|
|
bidExpiry,
|
|
await host.getAddress(),
|
|
await sign(client, requestHash),
|
|
await sign(host, bidHash)
|
|
)
|
|
})
|
|
|
|
it("requires on average a proof every period", async function () {
|
|
let blocks = 400
|
|
let proofs = 0
|
|
for (i=0; i<blocks; i++) {
|
|
await mineBlock()
|
|
if (await contracts.isProofRequired(id, await minedBlockNumber())) {
|
|
proofs += 1
|
|
}
|
|
}
|
|
let average = blocks / proofs
|
|
expect(average).to.be.closeTo(proofPeriod, proofPeriod / 2)
|
|
})
|
|
|
|
it("requires no proof for blocks that are unavailable", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
for (i=0; i<256; i++) { // only last 256 blocks are available in solidity
|
|
mineBlock()
|
|
}
|
|
expect(await contracts.isProofRequired(id, blocknumber)).to.be.false
|
|
})
|
|
|
|
it("submits a correct proof", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await contracts.submitProof(id, blocknumber, true)
|
|
})
|
|
|
|
it("fails proof submission when proof is incorrect", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await expect(
|
|
contracts.submitProof(id, blocknumber, false)
|
|
).to.be.revertedWith("Invalid proof")
|
|
})
|
|
|
|
it("fails proof submission when proof was not required", async function () {
|
|
while (await contracts.isProofRequired(id, await minedBlockNumber())) {
|
|
await mineBlock()
|
|
}
|
|
let blocknumber = await minedBlockNumber()
|
|
await expect(
|
|
contracts.submitProof(id, blocknumber, true)
|
|
).to.be.revertedWith("No proof required")
|
|
})
|
|
|
|
it("fails proof submission when proof is too late", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await mineUntilProofTimeout()
|
|
await expect(
|
|
contracts.submitProof(id, blocknumber, true)
|
|
).to.be.revertedWith("Proof not allowed after timeout")
|
|
})
|
|
|
|
it("fails proof submission when already submitted", async function() {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await contracts.submitProof(id, blocknumber, true)
|
|
await expect(
|
|
contracts.submitProof(id, blocknumber, true)
|
|
).to.be.revertedWith("Proof already submitted")
|
|
})
|
|
|
|
it("marks a proof as missing", async function () {
|
|
expect(await contracts.missingProofs(id)).to.equal(0)
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await mineUntilProofTimeout()
|
|
await contracts.markProofAsMissing(id, blocknumber)
|
|
expect(await contracts.missingProofs(id)).to.equal(1)
|
|
})
|
|
|
|
it("does not mark a proof as missing before timeout", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await mineBlock()
|
|
await expect(
|
|
contracts.markProofAsMissing(id, blocknumber)
|
|
).to.be.revertedWith("Proof has not timed out yet")
|
|
})
|
|
|
|
it("does not mark a submitted proof as missing", async function () {
|
|
await mineUntilProofIsRequired(id)
|
|
let blocknumber = await minedBlockNumber()
|
|
await contracts.submitProof(id, blocknumber, true)
|
|
await mineUntilProofTimeout()
|
|
await expect(
|
|
contracts.markProofAsMissing(id, blocknumber)
|
|
).to.be.revertedWith("Proof was submitted, not missing")
|
|
})
|
|
|
|
it("does not mark proof as missing when not required", async function () {
|
|
while (await contracts.isProofRequired(id, await minedBlockNumber())) {
|
|
mineBlock()
|
|
}
|
|
let blocknumber = await minedBlockNumber()
|
|
await mineUntilProofTimeout()
|
|
await expect(
|
|
contracts.markProofAsMissing(id, blocknumber)
|
|
).to.be.revertedWith("Proof was not required")
|
|
})
|
|
})
|
|
})
|
|
|
|
// TODO: implement checking of actual proofs of storage, instead of dummy bool
|
|
// TODO: payment on constructor
|
|
// TODO: contract start and timeout
|
|
// TODO: only allow proofs after start of contract
|
|
// TODO: payout
|
|
// TODO: stake
|