2021-11-01 12:30:35 +01:00
|
|
|
const { expect } = require("chai")
|
2024-01-23 13:56:52 +01:00
|
|
|
const { ethers, deployments } = require("hardhat")
|
2022-04-12 08:43:47 +02:00
|
|
|
const { hexlify, randomBytes } = ethers.utils
|
2022-03-08 15:58:08 +01:00
|
|
|
const {
|
|
|
|
snapshot,
|
|
|
|
revert,
|
2022-04-05 10:11:30 +02:00
|
|
|
mine,
|
2022-03-09 11:21:19 +01:00
|
|
|
ensureMinimumBlockHeight,
|
2022-03-08 15:58:08 +01:00
|
|
|
currentTime,
|
2023-10-16 11:14:02 +02:00
|
|
|
advanceTimeForNextBlock,
|
|
|
|
advanceTimeToForNextBlock,
|
2022-03-08 15:58:08 +01:00
|
|
|
} = require("./evm")
|
2023-01-09 14:21:23 +01:00
|
|
|
const { periodic } = require("./time")
|
2024-01-23 13:56:52 +01:00
|
|
|
const { loadProof, loadPublicInput } = require("../verifier/verifier")
|
2023-01-16 16:31:04 +01:00
|
|
|
const { SlotState } = require("./requests")
|
2023-01-23 17:58:42 +01:00
|
|
|
const binomialTest = require("@stdlib/stats-binomial-test")
|
2024-01-18 13:50:54 +01:00
|
|
|
const { exampleProof } = require("./examples")
|
2021-11-01 12:30:35 +01:00
|
|
|
|
|
|
|
describe("Proofs", function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
const slotId = hexlify(randomBytes(32))
|
2022-04-05 10:11:30 +02:00
|
|
|
const period = 30 * 60
|
2021-11-01 12:30:35 +01:00
|
|
|
const timeout = 5
|
2022-03-10 10:19:21 +01:00
|
|
|
const downtime = 64
|
2022-03-10 10:12:03 +01:00
|
|
|
const probability = 4 // require a proof roughly once every 4 periods
|
2024-08-14 07:50:32 +02:00
|
|
|
const downtimeProduct = 67
|
2022-04-05 10:11:30 +02:00
|
|
|
const { periodOf, periodEnd } = periodic(period)
|
2021-11-01 12:30:35 +01:00
|
|
|
|
|
|
|
let proofs
|
|
|
|
|
|
|
|
beforeEach(async function () {
|
2022-03-08 15:58:08 +01:00
|
|
|
await snapshot()
|
2022-03-09 11:21:19 +01:00
|
|
|
await ensureMinimumBlockHeight(256)
|
2021-11-01 12:30:35 +01:00
|
|
|
const Proofs = await ethers.getContractFactory("TestProofs")
|
2024-01-25 10:49:53 +01:00
|
|
|
await deployments.fixture(["Verifier"])
|
2024-01-23 13:56:52 +01:00
|
|
|
const verifier = await deployments.get("Groth16Verifier")
|
2024-01-10 15:12:18 +01:00
|
|
|
proofs = await Proofs.deploy(
|
2024-08-14 07:50:32 +02:00
|
|
|
{ period, timeout, downtime, zkeyHash: "", downtimeProduct },
|
2024-01-10 15:12:18 +01:00
|
|
|
verifier.address
|
|
|
|
)
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
2022-03-08 15:58:08 +01:00
|
|
|
afterEach(async function () {
|
|
|
|
await revert()
|
|
|
|
})
|
|
|
|
|
2022-09-29 20:07:55 +10:00
|
|
|
describe("general", function () {
|
|
|
|
beforeEach(async function () {
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Filled)
|
2022-09-29 20:07:55 +10:00
|
|
|
})
|
2021-11-01 15:28:22 +01:00
|
|
|
|
2022-09-29 20:07:55 +10:00
|
|
|
it("requires proofs with an agreed upon probability", async function () {
|
2023-01-23 17:58:42 +01:00
|
|
|
const samples = 256 // 256 samples avoids bias due to pointer downtime
|
|
|
|
|
2023-01-10 12:51:26 +01:00
|
|
|
await proofs.startRequiringProofs(slotId, probability)
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2022-09-29 20:07:55 +10:00
|
|
|
let amount = 0
|
2023-01-23 17:58:42 +01:00
|
|
|
for (let i = 0; i < samples; i++) {
|
2023-01-09 14:20:59 +01:00
|
|
|
if (await proofs.isProofRequired(slotId)) {
|
2022-09-29 20:07:55 +10:00
|
|
|
amount += 1
|
|
|
|
}
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2021-11-03 17:02:12 +01:00
|
|
|
}
|
2023-01-23 17:58:42 +01:00
|
|
|
|
|
|
|
const p = 1 / probability // expected probability
|
|
|
|
const alpha = 1 / 1000 // unit test can fail once every 1000 runs
|
|
|
|
|
|
|
|
// use binomial test to check that the measured amount is likely to occur
|
|
|
|
expect(binomialTest(amount, samples, { p, alpha }).rejected).to.be.false
|
2022-09-29 20:07:55 +10:00
|
|
|
})
|
2021-11-03 17:02:12 +01:00
|
|
|
|
2023-01-18 16:05:32 +01:00
|
|
|
it("supports probability 1 (proofs are always required)", async function () {
|
|
|
|
await proofs.startRequiringProofs(slotId, 1)
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2023-01-18 16:05:32 +01:00
|
|
|
while ((await proofs.getPointer(slotId)) < downtime) {
|
|
|
|
await mine()
|
|
|
|
}
|
|
|
|
expect(await proofs.isProofRequired(slotId)).to.be.true
|
|
|
|
})
|
|
|
|
|
2022-09-29 20:07:55 +10:00
|
|
|
it("requires no proofs in the start period", async function () {
|
|
|
|
const startPeriod = Math.floor((await currentTime()) / period)
|
|
|
|
const probability = 1
|
2023-01-10 12:51:26 +01:00
|
|
|
await proofs.startRequiringProofs(slotId, probability)
|
2022-09-29 20:07:55 +10:00
|
|
|
while (Math.floor((await currentTime()) / period) == startPeriod) {
|
2023-01-09 14:20:59 +01:00
|
|
|
expect(await proofs.isProofRequired(slotId)).to.be.false
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(Math.floor(period / 10))
|
|
|
|
await mine()
|
2022-09-29 20:07:55 +10:00
|
|
|
}
|
|
|
|
})
|
2022-03-08 15:58:08 +01:00
|
|
|
|
2022-09-29 20:07:55 +10:00
|
|
|
it("requires proofs for different ids at different times", async function () {
|
|
|
|
let id1 = hexlify(randomBytes(32))
|
|
|
|
let id2 = hexlify(randomBytes(32))
|
|
|
|
let id3 = hexlify(randomBytes(32))
|
2023-01-09 14:20:59 +01:00
|
|
|
for (let slotId of [id1, id2, id3]) {
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Filled)
|
2023-01-10 12:51:26 +01:00
|
|
|
await proofs.startRequiringProofs(slotId, probability)
|
2022-09-29 20:07:55 +10:00
|
|
|
}
|
|
|
|
let req1, req2, req3
|
|
|
|
while (req1 === req2 && req2 === req3) {
|
|
|
|
req1 = await proofs.isProofRequired(id1)
|
|
|
|
req2 = await proofs.isProofRequired(id2)
|
|
|
|
req3 = await proofs.isProofRequired(id3)
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2022-09-29 20:07:55 +10:00
|
|
|
}
|
|
|
|
})
|
2021-11-03 17:15:03 +01:00
|
|
|
|
2022-09-29 20:07:55 +10:00
|
|
|
it("moves pointer one block at a time", async function () {
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
|
|
|
|
await mine()
|
2022-09-29 20:07:55 +10:00
|
|
|
for (let i = 0; i < 256; i++) {
|
2023-01-09 14:20:59 +01:00
|
|
|
let previous = await proofs.getPointer(slotId)
|
2022-09-29 20:07:55 +10:00
|
|
|
await mine()
|
2023-01-09 14:20:59 +01:00
|
|
|
let current = await proofs.getPointer(slotId)
|
2022-09-29 20:07:55 +10:00
|
|
|
expect(current).to.equal((previous + 1) % 256)
|
|
|
|
}
|
|
|
|
})
|
2022-04-05 10:11:30 +02:00
|
|
|
})
|
2022-04-05 11:27:02 +02:00
|
|
|
|
|
|
|
describe("when proof requirement is upcoming", function () {
|
|
|
|
async function waitUntilProofWillBeRequired() {
|
2023-01-09 14:20:59 +01:00
|
|
|
while (!(await proofs.willProofBeRequired(slotId))) {
|
2022-04-05 11:27:02 +02:00
|
|
|
await mine()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
beforeEach(async function () {
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Filled)
|
2023-01-10 12:51:26 +01:00
|
|
|
await proofs.startRequiringProofs(slotId, probability)
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
|
2022-04-05 11:27:02 +02:00
|
|
|
await waitUntilProofWillBeRequired()
|
|
|
|
})
|
|
|
|
|
|
|
|
it("means the pointer is in downtime", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
expect(await proofs.getPointer(slotId)).to.be.lt(downtime)
|
|
|
|
while ((await proofs.getPointer(slotId)) < downtime) {
|
|
|
|
expect(await proofs.willProofBeRequired(slotId)).to.be.true
|
2022-04-05 11:27:02 +02:00
|
|
|
await mine()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
it("means that a proof is required after downtime", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
while ((await proofs.getPointer(slotId)) < downtime) {
|
2022-04-05 11:27:02 +02:00
|
|
|
await mine()
|
|
|
|
}
|
2023-01-09 14:20:59 +01:00
|
|
|
expect(await proofs.willProofBeRequired(slotId)).to.be.false
|
|
|
|
expect(await proofs.isProofRequired(slotId)).to.be.true
|
2022-04-05 11:27:02 +02:00
|
|
|
})
|
2022-09-13 17:18:55 +10:00
|
|
|
|
2023-01-16 16:31:04 +01:00
|
|
|
it("will not require proofs when slot is finished", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
expect(await proofs.getPointer(slotId)).to.be.lt(downtime)
|
|
|
|
expect(await proofs.willProofBeRequired(slotId)).to.be.true
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Finished)
|
2023-01-09 14:20:59 +01:00
|
|
|
expect(await proofs.willProofBeRequired(slotId)).to.be.false
|
2022-09-13 17:18:55 +10:00
|
|
|
})
|
2022-04-05 11:27:02 +02:00
|
|
|
})
|
2022-03-09 11:11:01 +01:00
|
|
|
|
2022-04-11 14:09:48 +02:00
|
|
|
describe("when proofs are required", function () {
|
2024-01-23 13:56:52 +01:00
|
|
|
const proof = loadProof("hardhat")
|
|
|
|
const pubSignals = loadPublicInput("hardhat")
|
2022-04-12 08:43:47 +02:00
|
|
|
|
2021-11-01 12:30:35 +01:00
|
|
|
beforeEach(async function () {
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Filled)
|
2023-01-10 12:51:26 +01:00
|
|
|
await proofs.startRequiringProofs(slotId, probability)
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
2023-01-09 14:20:59 +01:00
|
|
|
async function waitUntilProofIsRequired(slotId) {
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
|
|
|
|
await mine()
|
|
|
|
|
2022-03-10 13:35:41 +01:00
|
|
|
while (
|
|
|
|
!(
|
2023-01-09 14:20:59 +01:00
|
|
|
(await proofs.isProofRequired(slotId)) &&
|
|
|
|
(await proofs.getPointer(slotId)) < 250
|
2022-03-10 13:35:41 +01:00
|
|
|
)
|
|
|
|
) {
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-10 13:04:46 +01:00
|
|
|
it("provides different challenges per period", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
|
|
|
const challenge1 = await proofs.getChallenge(slotId)
|
|
|
|
await waitUntilProofIsRequired(slotId)
|
|
|
|
const challenge2 = await proofs.getChallenge(slotId)
|
2022-03-10 13:04:46 +01:00
|
|
|
expect(challenge2).not.to.equal(challenge1)
|
|
|
|
})
|
|
|
|
|
2023-01-09 14:20:59 +01:00
|
|
|
it("provides different challenges per slotId", async function () {
|
2022-04-12 08:43:47 +02:00
|
|
|
const id2 = hexlify(randomBytes(32))
|
|
|
|
const id3 = hexlify(randomBytes(32))
|
2023-01-09 14:20:59 +01:00
|
|
|
const challenge1 = await proofs.getChallenge(slotId)
|
2022-03-10 13:04:46 +01:00
|
|
|
const challenge2 = await proofs.getChallenge(id2)
|
|
|
|
const challenge3 = await proofs.getChallenge(id3)
|
|
|
|
expect(challenge1 === challenge2 && challenge2 === challenge3).to.be.false
|
|
|
|
})
|
|
|
|
|
2024-01-18 14:09:36 +01:00
|
|
|
it("handles a correct proof", async function () {
|
2024-01-22 16:43:03 +01:00
|
|
|
await proofs.proofReceived(slotId, proof, pubSignals)
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it("fails proof submission when proof is incorrect", async function () {
|
2024-01-18 13:50:54 +01:00
|
|
|
let invalid = exampleProof()
|
2024-01-30 11:17:04 +01:00
|
|
|
await expect(
|
|
|
|
proofs.proofReceived(slotId, invalid, pubSignals)
|
|
|
|
).to.be.revertedWith("Invalid proof")
|
2024-01-22 16:43:03 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it("fails proof submission when public input is incorrect", async function () {
|
|
|
|
let invalid = [1, 2, 3]
|
2024-01-30 11:17:04 +01:00
|
|
|
await expect(
|
|
|
|
proofs.proofReceived(slotId, proof, invalid)
|
|
|
|
).to.be.revertedWith("Invalid proof")
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
2022-04-12 08:43:47 +02:00
|
|
|
it("emits an event when proof was submitted", async function () {
|
2024-01-22 16:43:03 +01:00
|
|
|
await expect(proofs.proofReceived(slotId, proof, pubSignals))
|
2024-01-15 16:25:30 +01:00
|
|
|
.to.emit(proofs, "ProofSubmitted")
|
|
|
|
.withArgs(slotId)
|
2022-04-12 08:43:47 +02:00
|
|
|
})
|
|
|
|
|
2022-02-09 14:17:23 +01:00
|
|
|
it("fails proof submission when already submitted", async function () {
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
|
2024-01-22 16:43:03 +01:00
|
|
|
await proofs.proofReceived(slotId, proof, pubSignals)
|
|
|
|
await expect(
|
|
|
|
proofs.proofReceived(slotId, proof, pubSignals)
|
|
|
|
).to.be.revertedWith("Proof already submitted")
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it("marks a proof as missing", async function () {
|
2023-01-09 14:41:28 +01:00
|
|
|
expect(await proofs.missingProofs(slotId)).to.equal(0)
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2022-03-08 15:58:08 +01:00
|
|
|
let missedPeriod = periodOf(await currentTime())
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(missedPeriod))
|
|
|
|
await mine()
|
2023-01-09 14:20:59 +01:00
|
|
|
await proofs.markProofAsMissing(slotId, missedPeriod)
|
2023-01-09 14:41:28 +01:00
|
|
|
expect(await proofs.missingProofs(slotId)).to.equal(1)
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
2022-03-08 15:58:08 +01:00
|
|
|
it("does not mark a proof as missing before period end", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2022-03-08 15:58:08 +01:00
|
|
|
let currentPeriod = periodOf(await currentTime())
|
|
|
|
await expect(
|
2023-01-09 14:20:59 +01:00
|
|
|
proofs.markProofAsMissing(slotId, currentPeriod)
|
2022-03-08 15:58:08 +01:00
|
|
|
).to.be.revertedWith("Period has not ended yet")
|
|
|
|
})
|
|
|
|
|
|
|
|
it("does not mark a proof as missing after timeout", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2022-03-08 15:58:08 +01:00
|
|
|
let currentPeriod = periodOf(await currentTime())
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(currentPeriod) + timeout)
|
2021-11-01 12:30:35 +01:00
|
|
|
await expect(
|
2023-01-09 14:20:59 +01:00
|
|
|
proofs.markProofAsMissing(slotId, currentPeriod)
|
2022-03-08 15:58:08 +01:00
|
|
|
).to.be.revertedWith("Validation timed out")
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
|
2024-01-18 14:09:36 +01:00
|
|
|
it("does not mark a received proof as missing", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2024-01-18 14:09:36 +01:00
|
|
|
let receivedPeriod = periodOf(await currentTime())
|
2024-01-22 16:43:03 +01:00
|
|
|
await proofs.proofReceived(slotId, proof, pubSignals)
|
2024-01-18 14:09:36 +01:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(receivedPeriod))
|
2023-10-16 11:14:02 +02:00
|
|
|
await mine()
|
2021-11-01 12:30:35 +01:00
|
|
|
await expect(
|
2024-01-18 14:09:36 +01:00
|
|
|
proofs.markProofAsMissing(slotId, receivedPeriod)
|
2021-11-01 12:30:35 +01:00
|
|
|
).to.be.revertedWith("Proof was submitted, not missing")
|
|
|
|
})
|
|
|
|
|
|
|
|
it("does not mark proof as missing when not required", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
while (await proofs.isProofRequired(slotId)) {
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeForNextBlock(period)
|
|
|
|
await mine()
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
2022-03-08 15:58:08 +01:00
|
|
|
let currentPeriod = periodOf(await currentTime())
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(currentPeriod))
|
|
|
|
await mine()
|
2021-11-01 12:30:35 +01:00
|
|
|
await expect(
|
2023-01-09 14:20:59 +01:00
|
|
|
proofs.markProofAsMissing(slotId, currentPeriod)
|
2021-11-01 12:30:35 +01:00
|
|
|
).to.be.revertedWith("Proof was not required")
|
|
|
|
})
|
|
|
|
|
|
|
|
it("does not mark proof as missing twice", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2022-03-08 15:58:08 +01:00
|
|
|
let missedPeriod = periodOf(await currentTime())
|
2023-10-16 11:14:02 +02:00
|
|
|
await advanceTimeToForNextBlock(periodEnd(missedPeriod))
|
|
|
|
await mine()
|
2023-01-09 14:20:59 +01:00
|
|
|
await proofs.markProofAsMissing(slotId, missedPeriod)
|
2021-11-01 12:30:35 +01:00
|
|
|
await expect(
|
2023-01-09 14:20:59 +01:00
|
|
|
proofs.markProofAsMissing(slotId, missedPeriod)
|
2021-11-01 12:30:35 +01:00
|
|
|
).to.be.revertedWith("Proof already marked as missing")
|
|
|
|
})
|
2022-09-13 17:18:55 +10:00
|
|
|
|
2023-01-16 16:31:04 +01:00
|
|
|
it("requires no proofs when slot is finished", async function () {
|
2023-01-09 14:20:59 +01:00
|
|
|
await waitUntilProofIsRequired(slotId)
|
2023-01-16 16:31:04 +01:00
|
|
|
await proofs.setSlotState(slotId, SlotState.Finished)
|
|
|
|
expect(await proofs.isProofRequired(slotId)).to.be.false
|
2022-09-13 17:18:55 +10:00
|
|
|
})
|
2021-11-01 12:30:35 +01:00
|
|
|
})
|
|
|
|
})
|