dagger-contracts/test/Marketplace.test.js

1422 lines
51 KiB
JavaScript
Raw Permalink Normal View History

const { ethers } = require("hardhat")
const { AddressZero } = ethers.constants
const { BigNumber } = ethers
const { expect } = require("chai")
const {
exampleConfiguration,
exampleRequest,
exampleProof,
invalidProof,
} = require("./examples")
2023-01-16 16:31:24 +01:00
const { periodic, hours } = require("./time")
2022-07-19 11:33:54 +02:00
const { requestId, slotId, askToArray } = require("./ids")
const {
RequestState,
SlotState,
enableRequestAssertions,
} = require("./requests")
const {
waitUntilCancelled,
waitUntilStarted,
waitUntilFinished,
waitUntilFailed,
waitUntilSlotFailed,
patchOverloads,
} = require("./marketplace")
const { price, pricePerSlot } = require("./price")
2022-07-19 17:09:35 +02:00
const {
snapshot,
revert,
mine,
2022-07-19 17:09:35 +02:00
ensureMinimumBlockHeight,
advanceTimeForNextBlock,
advanceTimeToForNextBlock,
currentTime,
2022-07-19 17:09:35 +02:00
} = require("./evm")
const { arrayify } = require("ethers/lib/utils")
const ACCOUNT_STARTING_BALANCE = 1_000_000_000
2023-05-30 14:42:59 +02:00
describe("Marketplace constructor", function () {
let Marketplace, token, verifier, config
beforeEach(async function () {
await snapshot()
await ensureMinimumBlockHeight(256)
const TestToken = await ethers.getContractFactory("TestToken")
token = await TestToken.deploy()
const TestVerifier = await ethers.getContractFactory("TestVerifier")
verifier = await TestVerifier.deploy()
Marketplace = await ethers.getContractFactory("TestMarketplace")
config = exampleConfiguration()
})
afterEach(async function () {
await revert()
})
function testPercentageOverflow(property) {
it(`should reject for ${property} overflowing percentage values`, async () => {
config.collateral[property] = 101
2023-05-30 14:42:59 +02:00
await expect(
Marketplace.deploy(config, token.address, verifier.address)
2023-05-30 14:42:59 +02:00
).to.be.revertedWith("Must be less than 100")
})
}
2023-05-30 14:42:59 +02:00
testPercentageOverflow("repairRewardPercentage")
testPercentageOverflow("slashPercentage")
2023-05-30 14:42:59 +02:00
it("should reject when total slash percentage exceeds 100%", async () => {
config.collateral.slashPercentage = 1
config.collateral.maxNumberOfSlashes = 101
await expect(
Marketplace.deploy(config, token.address, verifier.address)
).to.be.revertedWith("Maximum slashing exceeds 100%")
})
})
describe("Marketplace", function () {
const proof = exampleProof()
const config = exampleConfiguration()
let marketplace
let token
let verifier
let client,
clientWithdrawRecipient,
host,
host1,
host2,
host3,
hostRewardRecipient,
hostCollateralRecipient
let request
2022-07-19 17:09:35 +02:00
let slot
enableRequestAssertions()
beforeEach(async function () {
await snapshot()
await ensureMinimumBlockHeight(256)
;[
client,
clientWithdrawRecipient,
host1,
host2,
host3,
hostRewardRecipient,
hostCollateralRecipient,
] = await ethers.getSigners()
2022-02-21 11:31:37 +01:00
host = host1
2022-02-17 11:00:18 +01:00
const TestToken = await ethers.getContractFactory("TestToken")
token = await TestToken.deploy()
for (let account of [
client,
clientWithdrawRecipient,
host1,
host2,
host3,
hostRewardRecipient,
hostCollateralRecipient,
]) {
await token.mint(account.address, ACCOUNT_STARTING_BALANCE)
2022-02-21 11:31:37 +01:00
}
2022-02-17 11:00:18 +01:00
const TestVerifier = await ethers.getContractFactory("TestVerifier")
verifier = await TestVerifier.deploy()
const Marketplace = await ethers.getContractFactory("TestMarketplace")
marketplace = await Marketplace.deploy(
config,
token.address,
verifier.address
)
patchOverloads(marketplace)
2022-02-17 11:00:18 +01:00
request = await exampleRequest()
2022-02-17 11:00:18 +01:00
request.client = client.address
2022-07-19 17:09:35 +02:00
slot = {
request: requestId(request),
index: request.ask.slots / 2,
2022-07-19 17:09:35 +02:00
}
})
afterEach(async function () {
await revert()
})
2022-02-17 11:00:18 +01:00
function switchAccount(account) {
token = token.connect(account)
marketplace = marketplace.connect(account)
patchOverloads(marketplace)
2022-02-17 11:00:18 +01:00
}
describe("requesting storage", function () {
2022-02-17 11:00:18 +01:00
beforeEach(function () {
switchAccount(client)
})
it("emits event when storage is requested", async function () {
await token.approve(marketplace.address, price(request))
// We +1 second to the expiry because the time will advance with the mined transaction for requestStorage because of Hardhat
const expectedExpiry = (await currentTime()) + request.expiry + 1
await expect(marketplace.requestStorage(request))
.to.emit(marketplace, "StorageRequested")
.withArgs(requestId(request), askToArray(request.ask), expectedExpiry)
})
it("allows retrieval of request details", async function () {
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
const id = requestId(request)
expect(await marketplace.getRequest(id)).to.be.request(request)
})
2022-02-17 11:00:18 +01:00
it("rejects request with invalid client address", async function () {
let invalid = { ...request, client: host.address }
await token.approve(marketplace.address, price(invalid))
2022-02-17 11:00:18 +01:00
await expect(marketplace.requestStorage(invalid)).to.be.revertedWith(
"Invalid client address"
)
})
it("rejects request with insufficient payment", async function () {
let insufficient = price(request) - 1
await token.approve(marketplace.address, insufficient)
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
2022-03-15 16:18:44 +01:00
"ERC20: insufficient allowance"
)
})
it("rejects request when expiry out of bounds", async function () {
await token.approve(marketplace.address, price(request))
request.expiry = request.ask.duration + 1
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
"Expiry not in range"
)
request.expiry = 0
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
"Expiry not in range"
)
})
it("is rejected with insufficient slots ", async function () {
request.ask.slots = 0
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
"Insufficient slots"
)
})
it("is rejected when maxSlotLoss exceeds slots", async function () {
request.ask.maxSlotLoss = request.ask.slots + 1
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
"maxSlotLoss exceeds slots"
)
})
it("rejects resubmission of request", async function () {
await token.approve(marketplace.address, price(request) * 2)
await marketplace.requestStorage(request)
await expect(marketplace.requestStorage(request)).to.be.revertedWith(
"Request already exists"
)
})
})
2023-03-08 12:02:34 +01:00
describe("filling a slot with collateral", function () {
2022-07-19 11:33:54 +02:00
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
2022-07-19 11:33:54 +02:00
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
2022-07-19 11:33:54 +02:00
})
it("emits event when slot is filled", async function () {
await expect(marketplace.fillSlot(slot.request, slot.index, proof))
.to.emit(marketplace, "SlotFilled")
.withArgs(slot.request, slot.index)
2022-07-19 11:33:54 +02:00
})
it("allows retrieval of host that filled slot", async function () {
expect(await marketplace.getHost(slotId(slot))).to.equal(AddressZero)
await marketplace.fillSlot(slot.request, slot.index, proof)
expect(await marketplace.getHost(slotId(slot))).to.equal(host.address)
})
it("fails to retrieve a request of an empty slot", async function () {
expect(marketplace.getActiveSlot(slotId(slot))).to.be.revertedWith(
"Slot is free"
)
})
it("allows retrieval of request of a filled slot", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
let activeSlot = await marketplace.getActiveSlot(slotId(slot))
expect(activeSlot.request).to.be.request(request)
expect(activeSlot.slotIndex).to.equal(slot.index)
})
2022-07-19 11:33:54 +02:00
it("is rejected when proof is incorrect", async function () {
await expect(
marketplace.fillSlot(slot.request, slot.index, invalidProof())
2022-07-19 11:33:54 +02:00
).to.be.revertedWith("Invalid proof")
})
it("is rejected when slot already filled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await expect(
marketplace.fillSlot(slot.request, slot.index, proof)
).to.be.revertedWith("Slot is not free")
2022-07-19 11:33:54 +02:00
})
it("is rejected when request is unknown", async function () {
let unknown = await exampleRequest()
2022-07-19 11:33:54 +02:00
await expect(
marketplace.fillSlot(requestId(unknown), 0, proof)
).to.be.revertedWith("Unknown request")
})
it("is rejected when request is cancelled", async function () {
2022-07-19 11:33:54 +02:00
switchAccount(client)
let expired = { ...request, expiry: hours(1) + 1 }
await token.approve(marketplace.address, price(request))
2022-07-19 11:33:54 +02:00
await marketplace.requestStorage(expired)
await waitUntilCancelled(expired)
2022-07-19 11:33:54 +02:00
switchAccount(host)
await expect(
marketplace.fillSlot(requestId(expired), slot.index, proof)
).to.be.revertedWith("Slot is not free")
2022-07-19 11:33:54 +02:00
})
it("is rejected when request is finished", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, slot.request)
await expect(
marketplace.fillSlot(slot.request, slot.index, proof)
).to.be.revertedWith("Slot is not free")
})
it("is rejected when request is failed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
await expect(
marketplace.fillSlot(slot.request, slot.index, proof)
).to.be.revertedWith("Slot is not free")
})
2022-07-19 11:33:54 +02:00
it("is rejected when slot index not in range", async function () {
const invalid = request.ask.slots
2022-07-19 11:33:54 +02:00
await expect(
marketplace.fillSlot(slot.request, invalid, proof)
).to.be.revertedWith("Invalid slot")
})
it("fails when all slots are already filled", async function () {
const lastSlot = request.ask.slots - 1
2023-05-30 14:42:59 +02:00
await token.approve(
marketplace.address,
request.ask.collateral * lastSlot
)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, price(request) * lastSlot)
for (let i = 0; i <= lastSlot; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
await expect(
marketplace.fillSlot(slot.request, lastSlot, proof)
).to.be.revertedWith("Slot is not free")
})
})
2023-03-08 12:02:34 +01:00
describe("filling slot without collateral", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
})
it("is rejected when approved collateral is insufficient", async function () {
let insufficient = request.ask.collateral - 1
await token.approve(marketplace.address, insufficient)
await expect(
marketplace.fillSlot(slot.request, slot.index, proof)
).to.be.revertedWith("ERC20: insufficient allowance")
})
it("collects only requested collateral and not more", async function () {
2023-05-30 14:42:59 +02:00
await token.approve(marketplace.address, request.ask.collateral * 2)
2023-03-08 12:02:34 +01:00
const startBalanace = await token.balanceOf(host.address)
await marketplace.fillSlot(slot.request, slot.index, proof)
const endBalance = await token.balanceOf(host.address)
2023-05-30 14:42:59 +02:00
expect(startBalanace - endBalance).to.eq(request.ask.collateral)
2023-03-08 12:02:34 +01:00
})
})
describe("submitting proofs when slot is filled", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
await token.approve(marketplace.address, request.ask.collateral)
await marketplace.fillSlot(slot.request, slot.index, proof)
await advanceTimeForNextBlock(config.proofs.period)
})
it("allows proofs to be submitted", async function () {
await marketplace.submitProof(slotId(slot), proof)
})
it("converts first 31 bytes of challenge to field element", async function () {
let challenge = arrayify(await marketplace.getChallenge(slotId(slot)))
let truncated = challenge.slice(0, 31)
let littleEndian = new Uint8Array(truncated).reverse()
let expected = BigNumber.from(littleEndian)
expect(await marketplace.challengeToFieldElement(challenge)).to.equal(
expected
)
})
it("converts merkle root to field element", async function () {
let merkleRoot = request.content.merkleRoot
let littleEndian = new Uint8Array(merkleRoot).reverse()
let expected = BigNumber.from(littleEndian)
expect(await marketplace.merkleRootToFieldElement(merkleRoot)).to.equal(
expected
)
})
})
describe("request end", function () {
var requestTime
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
requestTime = await currentTime()
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
it("sets the request end time to now + duration", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await expect(
(await marketplace.requestEnd(requestId(request))).toNumber()
).to.be.closeTo(requestTime + request.ask.duration, 1)
})
it("sets request end time to the past once failed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
let slot0 = { ...slot, index: request.ask.maxSlotLoss + 1 }
const now = await currentTime()
await expect(await marketplace.requestEnd(requestId(request))).to.be.eq(
now - 1
)
})
it("sets request end time to the past once cancelled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await mine()
const now = await currentTime()
await expect(await marketplace.requestEnd(requestId(request))).to.be.eq(
now - 1
)
})
it("checks that request end time is in the past once finished", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await mine()
const now = await currentTime()
// in the process of calling currentTime and requestEnd,
// block.timestamp has advanced by 1, so the expected proof end time will
// be block.timestamp - 1.
await expect(await marketplace.requestEnd(requestId(request))).to.be.eq(
now - 1
)
})
})
describe("freeing a slot", function () {
2023-03-08 12:02:34 +01:00
let id
beforeEach(async function () {
slot.index = 0
id = slotId(slot)
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
it("fails to free finished slot when slot not filled", async function () {
slot.index = 5
let nonExistentId = slotId(slot)
await expect(
marketplace.freeFinishedSlot(nonExistentId, proof)
).to.be.revertedWith("Slot is free")
})
it("fails to free cancelled slot when slot not filled", async function () {
slot.index = 5
let nonExistentId = slotId(slot)
await expect(
marketplace.freeCancelledSlot(nonExistentId, proof)
).to.be.revertedWith("Slot is free")
})
it("fails to free failed slot when slot not filled", async function () {
slot.index = 5
let nonExistentId = slotId(slot)
await expect(
marketplace.freeFailedSlot(nonExistentId)
).to.be.revertedWith("Slot is free")
})
it("fails to free finished slot when contract is new", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot not finished"
)
})
it("fails to free cancelled slot when contract is new", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot not cancelled"
)
})
it("fails to free failed slot when contract is new", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await expect(marketplace.freeFailedSlot(id)).to.be.revertedWith(
"Slot not failed"
)
})
it("fails to free finished slot when contract is started", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot not finished"
)
})
it("fails to free cancelled slot when contract is started", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot not cancelled"
)
})
it("fails to free failed slot when contract is started", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await expect(marketplace.freeFailedSlot(id)).to.be.revertedWith(
"Slot not failed"
)
})
it("fails to free finished slot when contract is cancelled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot not finished"
)
})
it("fails to free failed slot when contract is cancelled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await expect(marketplace.freeFailedSlot(id)).to.be.revertedWith(
"Slot not failed"
)
})
it("fails to free cancelled slot when contract is finished", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot not cancelled"
)
})
it("fails to free failed slot when contract is finished", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await expect(marketplace.freeFailedSlot(id)).to.be.revertedWith(
"Slot not failed"
)
})
it("fails to free finished slot when contract is failed", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
slot.index = request.ask.maxSlotLoss + 1
id = slotId(slot)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot not finished"
)
})
it("fails to free cancelled slot when contract is failed", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
slot.index = request.ask.maxSlotLoss + 1
id = slotId(slot)
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot not cancelled"
)
})
it("can only be freed by the host occupying the slot when finished", async function () {
await waitUntilStarted(marketplace, request, proof, token)
switchAccount(client)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot filled by other host"
)
})
it("can only be freed by the host occupying the slot when cancelled", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
switchAccount(client)
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot filled by other host"
)
})
it("can only be freed by the host occupying the slot when failed", async function () {
await waitUntilStarted(marketplace, request, proof, token)
switchAccount(client)
await expect(marketplace.freeFailedSlot(id)).to.be.revertedWith(
"Slot filled by other host"
)
})
it("successfully frees finished slot", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await expect(marketplace.freeFinishedSlot(id, proof)).not.to.be.reverted
})
it("successfully frees cancelled slot", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await expect(marketplace.freeCancelledSlot(id, proof)).not.to.be.reverted
})
it("successfully frees failed slot", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
slot.index = request.ask.maxSlotLoss + 1
id = slotId(slot)
await expect(marketplace.freeFailedSlot(id)).not.to.be.reverted
})
it("can only free finished slot once", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(id, proof)
await expect(marketplace.freeFinishedSlot(id, proof)).to.be.revertedWith(
"Slot not finished"
)
})
it("can only free cancelled slot once", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await marketplace.freeCancelledSlot(id, proof)
await expect(marketplace.freeCancelledSlot(id, proof)).to.be.revertedWith(
"Slot not cancelled"
)
})
it("emits event once slot is freed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await expect(marketplace.forciblyFreeSlot(id))
.to.emit(marketplace, "SlotFreed")
.withArgs(slot.request, slot.index)
})
})
2022-07-19 17:09:35 +02:00
describe("paying out a slot", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
2022-07-19 17:09:35 +02:00
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
2022-07-19 17:09:35 +02:00
})
2023-03-08 12:02:34 +01:00
it("pays the host when contract has finished and returns collateral", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
2022-07-19 17:09:35 +02:00
const startBalance = await token.balanceOf(host.address)
await marketplace.freeFinishedSlot(slotId(slot), proof)
2022-07-19 17:09:35 +02:00
const endBalance = await token.balanceOf(host.address)
2023-05-30 14:42:59 +02:00
expect(endBalance - startBalance).to.equal(
pricePerSlot(request) + request.ask.collateral
)
2022-07-19 17:09:35 +02:00
})
it("pays to host reward address when contract has finished and returns collateral to host collateral address", async function () {
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
const startBalanceHost = await token.balanceOf(host.address)
const startBalanceReward = await token.balanceOf(
hostRewardRecipient.address
)
const startBalanceCollateral = await token.balanceOf(
hostCollateralRecipient.address
)
await marketplace.freeFinishedSlot(
slotId(slot),
proof,
hostRewardRecipient.address,
hostCollateralRecipient.address
)
const endBalanceHost = await token.balanceOf(host.address)
const endBalanceReward = await token.balanceOf(
hostRewardRecipient.address
)
const endBalanceCollateral = await token.balanceOf(
hostCollateralRecipient.address
)
expect(endBalanceHost).to.equal(startBalanceHost)
expect(endBalanceCollateral - startBalanceCollateral).to.equal(
request.ask.collateral
)
expect(endBalanceReward - startBalanceReward).to.equal(
pricePerSlot(request)
)
})
it("pays the host when contract was cancelled", async function () {
// Lets advance the time more into the expiry window
const filledAt = (await currentTime()) + Math.floor(request.expiry / 3)
const expiresAt = (
await marketplace.requestExpiry(requestId(request))
).toNumber()
await advanceTimeToForNextBlock(filledAt)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await marketplace.freeCancelledSlot(slotId(slot), proof)
const expectedPartialPayout = (expiresAt - filledAt) * request.ask.reward
const endBalance = await token.balanceOf(host.address)
2024-01-10 15:12:18 +01:00
expect(endBalance - ACCOUNT_STARTING_BALANCE).to.be.equal(
expectedPartialPayout
)
})
it("pays to host reward address when contract was cancelled, and returns collateral to host address", async function () {
// Lets advance the time more into the expiry window
const filledAt = (await currentTime()) + Math.floor(request.expiry / 3)
const expiresAt = (
await marketplace.requestExpiry(requestId(request))
).toNumber()
await advanceTimeToForNextBlock(filledAt)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
const startBalanceHost = await token.balanceOf(host.address)
const startBalanceReward = await token.balanceOf(
hostRewardRecipient.address
)
const startBalanceCollateral = await token.balanceOf(
hostCollateralRecipient.address
)
await marketplace.freeCancelledSlot(
slotId(slot),
proof,
hostRewardRecipient.address,
hostCollateralRecipient.address
)
const expectedPartialPayout = (expiresAt - filledAt) * request.ask.reward
const endBalanceReward = await token.balanceOf(
hostRewardRecipient.address
)
expect(endBalanceReward - startBalanceReward).to.be.equal(
expectedPartialPayout
)
const endBalanceHost = await token.balanceOf(host.address)
expect(endBalanceHost).to.be.equal(startBalanceHost)
const endBalanceCollateral = await token.balanceOf(
hostCollateralRecipient.address
)
expect(endBalanceCollateral - startBalanceCollateral).to.be.equal(
request.ask.collateral
)
})
2022-07-19 17:09:35 +02:00
it("cannot be filled again", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(slotId(slot), proof)
2022-07-19 17:09:35 +02:00
await expect(marketplace.fillSlot(slot.request, slot.index, proof)).to.be
.reverted
})
})
describe("fulfilling a request", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
it("emits event when all slots are filled", async function () {
const lastSlot = request.ask.slots - 1
2023-05-30 14:42:59 +02:00
await token.approve(
marketplace.address,
request.ask.collateral * lastSlot
)
for (let i = 0; i < lastSlot; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
await expect(marketplace.fillSlot(slot.request, lastSlot, proof))
.to.emit(marketplace, "RequestFulfilled")
.withArgs(requestId(request))
})
it("sets state when all slots are filled", async function () {
2023-03-08 12:02:34 +01:00
const slots = request.ask.slots
await token.approve(marketplace.address, request.ask.collateral * slots)
for (let i = 0; i < slots; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
await expect(await marketplace.requestState(slot.request)).to.equal(
RequestState.Started
)
})
it("fails when all slots are already filled", async function () {
const lastSlot = request.ask.slots - 1
2023-05-30 14:42:59 +02:00
await token.approve(
marketplace.address,
request.ask.collateral * (lastSlot + 1)
)
for (let i = 0; i <= lastSlot; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
await expect(
marketplace.fillSlot(slot.request, lastSlot, proof)
).to.be.revertedWith("Slot is not free")
})
})
describe("withdrawing funds", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
it("rejects withdraw when request not yet timed out", async function () {
switchAccount(client)
await expect(
marketplace.withdrawFunds(slot.request, clientWithdrawRecipient.address)
).to.be.revertedWith("Request not yet timed out")
})
it("rejects withdraw when wrong account used", async function () {
await waitUntilCancelled(request)
await expect(
marketplace.withdrawFunds(slot.request, clientWithdrawRecipient.address)
).to.be.revertedWith("Invalid client address")
})
it("rejects withdraw when in wrong state", async function () {
// fill all slots, should change state to RequestState.Started
const lastSlot = request.ask.slots - 1
2023-05-30 14:42:59 +02:00
await token.approve(
marketplace.address,
request.ask.collateral * (lastSlot + 1)
)
for (let i = 0; i <= lastSlot; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
await waitUntilCancelled(request)
switchAccount(client)
await expect(
marketplace.withdrawFunds(slot.request, clientWithdrawRecipient.address)
).to.be.revertedWith("Invalid state")
})
it("emits event once request is cancelled", async function () {
await waitUntilCancelled(request)
switchAccount(client)
await expect(
marketplace.withdrawFunds(slot.request, clientWithdrawRecipient.address)
)
.to.emit(marketplace, "RequestCancelled")
.withArgs(requestId(request))
})
it("withdraws to the client payout address", async function () {
await waitUntilCancelled(request)
switchAccount(client)
const startBalanceClient = await token.balanceOf(client.address)
const startBalancePayout = await token.balanceOf(
clientWithdrawRecipient.address
)
await marketplace.withdrawFunds(
slot.request,
clientWithdrawRecipient.address
)
const endBalanceClient = await token.balanceOf(client.address)
const endBalancePayout = await token.balanceOf(
clientWithdrawRecipient.address
)
expect(endBalanceClient).to.equal(startBalanceClient)
expect(endBalancePayout - startBalancePayout).to.equal(price(request))
})
it("withdraws to the client payout address for cancelled requests lowered by hosts payout", async function () {
// Lets advance the time more into the expiry window
const filledAt = (await currentTime()) + Math.floor(request.expiry / 3)
const expiresAt = (
await marketplace.requestExpiry(requestId(request))
).toNumber()
await advanceTimeToForNextBlock(filledAt)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
const expectedPartialhostRewardRecipient =
(expiresAt - filledAt) * request.ask.reward
switchAccount(client)
await marketplace.withdrawFunds(
slot.request,
clientWithdrawRecipient.address
)
const endBalance = await token.balanceOf(clientWithdrawRecipient.address)
expect(endBalance - ACCOUNT_STARTING_BALANCE).to.equal(
price(request) - expectedPartialhostRewardRecipient
2024-01-10 15:12:18 +01:00
)
})
})
describe("request state", function () {
const { New, Cancelled, Started, Failed, Finished } = RequestState
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
it("is 'New' initially", async function () {
expect(await marketplace.requestState(slot.request)).to.equal(New)
})
it("changes to 'Cancelled' once request is cancelled", async function () {
await waitUntilCancelled(request)
await mine()
expect(await marketplace.requestState(slot.request)).to.equal(Cancelled)
})
it("remains 'Cancelled' when client withdraws funds", async function () {
await waitUntilCancelled(request)
switchAccount(client)
await marketplace.withdrawFunds(
slot.request,
clientWithdrawRecipient.address
)
expect(await marketplace.requestState(slot.request)).to.equal(Cancelled)
})
it("changes to 'Started' once all slots are filled", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
expect(await marketplace.requestState(slot.request)).to.equal(Started)
})
it("changes to 'Failed' once too many slots are freed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
await mine()
expect(await marketplace.requestState(slot.request)).to.equal(Failed)
2022-09-13 17:20:07 +10:00
})
it("does not change to 'Failed' before it is started", async function () {
2023-05-30 14:42:59 +02:00
await token.approve(
marketplace.address,
request.ask.collateral * (request.ask.maxSlotLoss + 1)
)
for (let i = 0; i <= request.ask.maxSlotLoss; i++) {
await marketplace.fillSlot(slot.request, i, proof)
}
for (let i = 0; i <= request.ask.maxSlotLoss; i++) {
slot.index = i
let id = slotId(slot)
await marketplace.forciblyFreeSlot(id)
}
expect(await marketplace.requestState(slot.request)).to.equal(New)
})
it("changes to 'Finished' when the request ends", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await mine()
expect(await marketplace.requestState(slot.request)).to.equal(Finished)
})
it("remains 'Finished' once a slot is paid out", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(slotId(slot), proof)
expect(await marketplace.requestState(slot.request)).to.equal(Finished)
})
})
2023-01-18 15:26:21 +01:00
describe("slot state", function () {
const { Free, Filled, Finished, Failed, Paid, Cancelled } = SlotState
2023-01-18 15:26:21 +01:00
let period, periodEnd
beforeEach(async function () {
period = config.proofs.period
;({ periodOf, periodEnd } = periodic(period))
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
2023-01-18 15:26:21 +01:00
})
async function waitUntilProofIsRequired(id) {
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
await mine()
2023-01-18 15:26:21 +01:00
while (
!(
(await marketplace.isProofRequired(id)) &&
(await marketplace.getPointer(id)) < 250
)
) {
await advanceTimeForNextBlock(period)
await mine()
2023-01-18 15:26:21 +01:00
}
}
it("is 'Free' initially", async function () {
expect(await marketplace.slotState(slotId(slot))).to.equal(Free)
})
it("changes to 'Filled' when slot is filled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
expect(await marketplace.slotState(slotId(slot))).to.equal(Filled)
})
it("changes to 'Finished' when request finishes", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
2023-01-18 15:26:21 +01:00
await waitUntilFinished(marketplace, slot.request)
await mine()
2023-01-18 15:26:21 +01:00
expect(await marketplace.slotState(slotId(slot))).to.equal(Finished)
})
it("changes to 'Cancelled' when request is cancelled", async function () {
2023-01-18 15:26:21 +01:00
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await mine()
expect(await marketplace.slotState(slotId(slot))).to.equal(Cancelled)
2023-01-18 15:26:21 +01:00
})
it("changes to 'Free' when host frees the slot", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await marketplace.forciblyFreeSlot(slotId(slot))
2023-01-18 15:26:21 +01:00
expect(await marketplace.slotState(slotId(slot))).to.equal(Free)
})
it("changes to 'Free' when too many proofs are missed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
2023-01-18 15:26:21 +01:00
while ((await marketplace.slotState(slotId(slot))) === Filled) {
await waitUntilProofIsRequired(slotId(slot))
const missedPeriod = periodOf(await currentTime())
await advanceTimeForNextBlock(period)
await mine()
2023-01-18 15:26:21 +01:00
await marketplace.markProofAsMissing(slotId(slot), missedPeriod)
}
expect(await marketplace.slotState(slotId(slot))).to.equal(Free)
})
it("changes to 'Failed' when request fails", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
2023-01-18 15:26:21 +01:00
await waitUntilSlotFailed(marketplace, request, slot)
await mine()
2023-01-18 15:26:21 +01:00
expect(await marketplace.slotState(slotId(slot))).to.equal(Failed)
})
it("changes to 'Paid' when host has been paid", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
2023-01-18 15:26:21 +01:00
await waitUntilFinished(marketplace, slot.request)
await marketplace.freeFinishedSlot(slotId(slot), proof)
2023-01-18 15:26:21 +01:00
expect(await marketplace.slotState(slotId(slot))).to.equal(Paid)
})
})
describe("proof requirements", function () {
let period, periodOf, periodEnd
beforeEach(async function () {
period = config.proofs.period
;({ periodOf, periodEnd } = periodic(period))
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
async function waitUntilProofWillBeRequired(id) {
while (!(await marketplace.willProofBeRequired(id))) {
await mine()
}
}
async function waitUntilProofIsRequired(id) {
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
while (
!(
(await marketplace.isProofRequired(id)) &&
(await marketplace.getPointer(id)) < 250
)
2023-05-30 14:42:59 +02:00
) {
await advanceTimeForNextBlock(period)
await mine()
}
}
it("requires proofs when slot is filled", async function () {
const id = slotId(slot)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilProofWillBeRequired(id)
})
it("will not require proofs once cancelled", async function () {
const id = slotId(slot)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilProofWillBeRequired(id)
await expect(await marketplace.willProofBeRequired(id)).to.be.true
await waitUntilCancelled(request)
await mine()
await expect(await marketplace.willProofBeRequired(id)).to.be.false
})
it("does not require proofs once cancelled", async function () {
const id = slotId(slot)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilProofIsRequired(id)
await expect(await marketplace.isProofRequired(id)).to.be.true
await waitUntilCancelled(request)
await mine()
await expect(await marketplace.isProofRequired(id)).to.be.false
})
it("does not provide challenges once cancelled", async function () {
const id = slotId(slot)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilProofIsRequired(id)
await mine()
const challenge1 = await marketplace.getChallenge(id)
expect(BigNumber.from(challenge1).gt(0))
await waitUntilCancelled(request)
await mine()
const challenge2 = await marketplace.getChallenge(id)
expect(BigNumber.from(challenge2).isZero())
})
it("does not provide pointer once cancelled", async function () {
const id = slotId(slot)
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilProofIsRequired(id)
await mine()
const challenge1 = await marketplace.getChallenge(id)
expect(BigNumber.from(challenge1).gt(0))
await waitUntilCancelled(request)
await mine()
const challenge2 = await marketplace.getChallenge(id)
expect(BigNumber.from(challenge2).isZero())
})
})
describe("missing proofs", function () {
let period, periodOf, periodEnd
beforeEach(async function () {
period = config.proofs.period
;({ periodOf, periodEnd } = periodic(period))
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
})
async function waitUntilProofIsRequired(id) {
await advanceTimeToForNextBlock(periodEnd(periodOf(await currentTime())))
await mine()
while (
!(
(await marketplace.isProofRequired(id)) &&
(await marketplace.getPointer(id)) < 250
)
2023-05-30 14:42:59 +02:00
) {
await advanceTimeForNextBlock(period)
await mine()
}
}
it("fails to mark proof as missing when cancelled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
let missedPeriod = periodOf(await currentTime())
await expect(
marketplace.markProofAsMissing(slotId(slot), missedPeriod)
).to.be.revertedWith("Slot not accepting proofs")
})
describe("slashing when missing proofs", function () {
it("reduces collateral when too many proofs are missing", async function () {
const id = slotId(slot)
2023-03-08 12:02:34 +01:00
const { slashCriterion, slashPercentage } = config.collateral
await marketplace.fillSlot(slot.request, slot.index, proof)
for (let i = 0; i < slashCriterion; i++) {
await waitUntilProofIsRequired(id)
let missedPeriod = periodOf(await currentTime())
2024-01-10 15:12:18 +01:00
await advanceTimeForNextBlock(period + 1)
await marketplace.markProofAsMissing(id, missedPeriod)
}
2023-05-30 14:42:59 +02:00
const expectedBalance =
(request.ask.collateral * (100 - slashPercentage)) / 100
2023-03-08 12:02:34 +01:00
2023-05-30 14:42:59 +02:00
expect(
BigNumber.from(expectedBalance).eq(
await marketplace.getSlotCollateral(id)
)
)
})
})
it("frees slot when collateral slashed below minimum threshold", async function () {
2023-05-30 14:42:59 +02:00
const minimum =
request.ask.collateral -
(request.ask.collateral *
config.collateral.maxNumberOfSlashes *
config.collateral.slashPercentage) /
100
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
while ((await marketplace.slotState(slotId(slot))) === SlotState.Filled) {
2023-05-30 14:42:59 +02:00
expect(await marketplace.getSlotCollateral(slotId(slot))).to.be.gt(
minimum
)
await waitUntilProofIsRequired(slotId(slot))
const missedPeriod = periodOf(await currentTime())
2024-01-10 15:12:18 +01:00
await advanceTimeForNextBlock(period + 1)
await marketplace.markProofAsMissing(slotId(slot), missedPeriod)
}
expect(await marketplace.slotState(slotId(slot))).to.equal(SlotState.Free)
2023-05-30 14:42:59 +02:00
expect(await marketplace.getSlotCollateral(slotId(slot))).to.be.lte(
minimum
)
})
it("free slot when minimum reached and resets missed proof counter", async function () {
2023-05-30 14:42:59 +02:00
const minimum =
request.ask.collateral -
(request.ask.collateral *
config.collateral.maxNumberOfSlashes *
config.collateral.slashPercentage) /
100
await waitUntilStarted(marketplace, request, proof, token)
let missedProofs = 0
while ((await marketplace.slotState(slotId(slot))) === SlotState.Filled) {
2023-05-30 14:42:59 +02:00
expect(await marketplace.getSlotCollateral(slotId(slot))).to.be.gt(
minimum
)
await waitUntilProofIsRequired(slotId(slot))
const missedPeriod = periodOf(await currentTime())
2024-01-10 15:12:18 +01:00
await advanceTimeForNextBlock(period + 1)
2023-05-30 14:42:59 +02:00
expect(await marketplace.missingProofs(slotId(slot))).to.equal(
missedProofs
)
await marketplace.markProofAsMissing(slotId(slot), missedPeriod)
missedProofs += 1
}
expect(await marketplace.slotState(slotId(slot))).to.equal(SlotState.Free)
expect(await marketplace.missingProofs(slotId(slot))).to.equal(0)
2023-05-30 14:42:59 +02:00
expect(await marketplace.getSlotCollateral(slotId(slot))).to.be.lte(
minimum
)
})
})
describe("list of active requests", function () {
beforeEach(async function () {
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
switchAccount(client)
await token.approve(marketplace.address, price(request))
})
it("adds request to list when requesting storage", async function () {
await marketplace.requestStorage(request)
expect(await marketplace.myRequests()).to.deep.equal([requestId(request)])
})
2022-11-23 12:39:49 +01:00
it("keeps request in list when cancelled", async function () {
await marketplace.requestStorage(request)
await waitUntilCancelled(request)
await mine()
2022-11-23 12:39:49 +01:00
expect(await marketplace.myRequests()).to.deep.equal([requestId(request)])
})
it("removes request from list when funds are withdrawn", async function () {
await marketplace.requestStorage(request)
await waitUntilCancelled(request)
await marketplace.withdrawFunds(
requestId(request),
clientWithdrawRecipient.address
)
expect(await marketplace.myRequests()).to.deep.equal([])
})
2022-11-23 12:39:49 +01:00
it("keeps request in list when request fails", async function () {
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
await mine()
switchAccount(client)
2022-11-23 12:39:49 +01:00
expect(await marketplace.myRequests()).to.deep.equal([requestId(request)])
})
it("removes request from list when request finishes", async function () {
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(slotId(slot), proof)
switchAccount(client)
expect(await marketplace.myRequests()).to.deep.equal([])
})
})
2022-11-11 13:33:02 +11:00
describe("list of active slots", function () {
beforeEach(async function () {
switchAccount(client)
await token.approve(marketplace.address, price(request))
await marketplace.requestStorage(request)
switchAccount(host)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
2022-11-11 13:33:02 +11:00
})
it("adds slot to list when filling slot", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
let slot1 = { ...slot, index: slot.index + 1 }
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
2022-11-11 13:33:02 +11:00
await marketplace.fillSlot(slot.request, slot1.index, proof)
expect(await marketplace.mySlots()).to.have.members([
2022-11-11 13:33:02 +11:00
slotId(slot),
slotId(slot1),
])
})
it("removes slot from list when finished slot is freed", async function () {
slot.index = 0
2022-11-11 13:33:02 +11:00
let slot1 = { ...slot, index: slot.index + 1 }
let slot2 = { ...slot, index: slot.index + 2 }
let slot3 = { ...slot, index: slot.index + 3 }
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(slotId(slot), proof)
expect(await marketplace.mySlots()).to.have.members([
slotId(slot1),
slotId(slot2),
slotId(slot3),
])
})
it("removes slot from list when cancelled slot is freed", async function () {
slot.index = 0
let slot1 = { ...slot, index: slot.index + 1 }
await marketplace.fillSlot(slot.request, slot.index, proof)
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
await marketplace.fillSlot(slot1.request, slot1.index, proof)
await waitUntilCancelled(request)
await marketplace.freeCancelledSlot(slotId(slot), proof)
expect(await marketplace.mySlots()).to.have.members([slotId(slot1)])
2022-11-11 13:33:02 +11:00
})
it("removes slot from list when failed slot is freed", async function () {
slot.index = 0
let slot3 = { ...slot, index: slot.index + 3 }
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilFailed(marketplace, request)
await marketplace.freeFailedSlot(slotId(slot3))
expect(await marketplace.mySlots()).to.have.members([])
})
it("keeps slots when cancelled", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
let slot1 = { ...slot, index: slot.index + 1 }
2023-03-08 12:02:34 +01:00
await token.approve(marketplace.address, request.ask.collateral)
await marketplace.fillSlot(slot.request, slot1.index, proof)
await waitUntilCancelled(request)
await mine()
expect(await marketplace.mySlots()).to.have.members([
slotId(slot),
slotId(slot1),
])
})
it("removes slot when finished slot is freed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
2022-11-11 13:33:02 +11:00
await waitUntilFinished(marketplace, requestId(request))
await marketplace.freeFinishedSlot(slotId(slot), proof)
expect(await marketplace.mySlots()).to.not.contain(slotId(slot))
2022-11-11 13:33:02 +11:00
})
it("removes slot when cancelled slot is freed", async function () {
await marketplace.fillSlot(slot.request, slot.index, proof)
await waitUntilCancelled(request)
await marketplace.freeCancelledSlot(slotId(slot), proof)
expect(await marketplace.mySlots()).to.not.contain(slotId(slot))
})
it("removes slot when failed slot is freed", async function () {
2023-03-08 12:02:34 +01:00
await waitUntilStarted(marketplace, request, proof, token)
await waitUntilSlotFailed(marketplace, request, slot)
await marketplace.freeFailedSlot(slotId(slot))
expect(await marketplace.mySlots()).to.not.contain(slotId(slot))
})
2022-11-11 13:33:02 +11:00
})
})