Skip to content

Commit

Permalink
[FLB-01M] Potentially Incorrect Dispute Case Handling (#877)
Browse files Browse the repository at this point in the history
* Revert if invalid final state

* Test revert reasons
  • Loading branch information
zajck authored Jan 9, 2024
1 parent 634c838 commit 3892e75
Show file tree
Hide file tree
Showing 8 changed files with 144 additions and 6 deletions.
4 changes: 4 additions & 0 deletions contracts/domain/BosonErrors.sol
Original file line number Diff line number Diff line change
Expand Up @@ -214,6 +214,8 @@ interface BosonErrors {
error ExchangeAlreadyExists();
// Range length is 0, is more than quantity available or it would cause an overflow
error InvalidRangeLength();
// Exchange is being finalized into an invalid state
error InvalidTargeExchangeState();

// Twin related
// Twin does not exist
Expand Down Expand Up @@ -300,6 +302,8 @@ interface BosonErrors {
error InvalidDisputeTimeout();
// Absolute zero offers cannot be escalated
error EscalationNotAllowed();
// Dispute is being finalized into an invalid state
error InvalidTargeDisputeState();

// Config related
// Percentage exceeds 100%
Expand Down
23 changes: 23 additions & 0 deletions contracts/mock/MockDisputeHandlerFacet.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
// SPDX-License-Identifier: GPL-3.0-or-later
pragma solidity 0.8.22;

import { DisputeHandlerFacet } from "../protocol/facets/DisputeHandlerFacet.sol";

/**
* @title TestDisputeHandlerFacet
*
* @notice Extended DisputeHandlerFacet with additional external functions for testing
*/
contract TestDisputeHandlerFacet is DisputeHandlerFacet {
/**
* @notice Test function to test invalid final dispute state
*
* @param _exchangeId - the id of the associated exchange
* @param _targetState - target final state
*/
function finalizeDispute(uint256 _exchangeId, DisputeState _targetState) external {
(, Exchange storage exchange) = fetchExchange(_exchangeId);
(, Dispute storage dispute, DisputeDates storage disputeDates) = fetchDispute(_exchangeId);
finalizeDispute(_exchangeId, exchange, dispute, disputeDates, _targetState, 1000);
}
}
22 changes: 22 additions & 0 deletions contracts/mock/MockExchangeHandlerFacet.sol
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import { DisputeBase } from "../protocol/bases/DisputeBase.sol";
import { FundsLib } from "../protocol/libs/FundsLib.sol";
import "../domain/BosonConstants.sol";
import "@openzeppelin/contracts/utils/Address.sol";
import { ExchangeHandlerFacet } from "../protocol/facets/ExchangeHandlerFacet.sol";

/**
* @title MockExchangeHandlerFacet
Expand Down Expand Up @@ -537,3 +538,24 @@ contract MockExchangeHandlerFacetWithDefect is MockExchangeHandlerFacet {
emit VoucherCanceled2(exchange.offerId, _exchangeId, msgSender());
}
}

/**
* @title TestExchangeHandlerFacet
*
* @notice Extended ExchangeHandlerFacet with additional external functions for testing
*/
contract TestExchangeHandlerFacet is ExchangeHandlerFacet {
//solhint-disable-next-line
constructor(uint256 _firstExchangeId2_2_0) ExchangeHandlerFacet(_firstExchangeId2_2_0) {}

/**
* @notice Test function to test invalid final exchange state
*
* @param _exchangeId - the id of the exchange to finalize
* @param _targetState - the target state to which the exchange should be transitioned
*/
function finalizeExchange(uint256 _exchangeId, ExchangeState _targetState) external {
(, Exchange storage exchange) = fetchExchange(_exchangeId);
finalizeExchange(exchange, _targetState);
}
}
3 changes: 3 additions & 0 deletions contracts/protocol/facets/DisputeHandlerFacet.sol
Original file line number Diff line number Diff line change
Expand Up @@ -439,6 +439,9 @@ contract DisputeHandlerFacet is DisputeBase, IBosonDisputeHandler {
DisputeState _targetState,
uint256 _buyerPercent
) internal disputesNotPaused {
if (_targetState == DisputeState.Escalated || _targetState == DisputeState.Resolving)
revert InvalidTargeDisputeState();

// update dispute and exchange
_disputeDates.finalized = block.timestamp;
_dispute.state = _targetState;
Expand Down
10 changes: 5 additions & 5 deletions contracts/protocol/facets/ExchangeHandlerFacet.sol
Original file line number Diff line number Diff line change
Expand Up @@ -947,11 +947,11 @@ contract ExchangeHandlerFacet is DisputeBase, BuyerBase, IBosonExchangeHandler,
*/
function finalizeExchange(Exchange storage _exchange, ExchangeState _targetState) internal {
// Make sure target state is a final state
require(
_targetState == ExchangeState.Completed ||
_targetState == ExchangeState.Revoked ||
_targetState == ExchangeState.Canceled
);
if (
_targetState != ExchangeState.Completed &&
_targetState != ExchangeState.Revoked &&
_targetState != ExchangeState.Canceled
) revert InvalidTargeExchangeState();

// Set the exchange state to the target state
_exchange.state = _targetState;
Expand Down
2 changes: 2 additions & 0 deletions scripts/config/revert-reasons.js
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,7 @@ exports.RevertReasons = {
EXCHANGE_IS_NOT_IN_A_FINAL_STATE: "ExchangeIsNotInAFinalState",
INVALID_RANGE_LENGTH: "InvalidRangeLength",
EXCHANGE_ALREADY_EXISTS: "ExchangeAlreadyExists",
INVALID_TARGET_EXCHANGE_STATE: "InvalidTargeExchangeState",

// Voucher related
EXCHANGE_ID_IN_RESERVED_RANGE: "ExchangeIdInReservedRange",
Expand Down Expand Up @@ -190,6 +191,7 @@ exports.RevertReasons = {
DISPUTE_STILL_VALID: "DisputeStillValid",
INVALID_DISPUTE_TIMEOUT: "InvalidDisputeTimeout",
ESCALATION_NOT_ALLOWED: "EscalationNotAllowed",
INVALID_TARGET_DISPUTE_STATE: "InvalidTargeDisputeState",

// Config related
FEE_PERCENTAGE_INVALID: "InvalidFeePercentage",
Expand Down
45 changes: 44 additions & 1 deletion test/protocol/DisputeHandlerTest.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
const { ethers } = require("hardhat");
const { ZeroAddress, provider, zeroPadBytes, MaxUint256, parseUnits, getContractAt } = ethers;
const { ZeroAddress, provider, zeroPadBytes, MaxUint256, parseUnits, getContractAt, id } = ethers;
const { expect, assert } = require("chai");
const Exchange = require("../../scripts/domain/Exchange");
const Dispute = require("../../scripts/domain/Dispute");
Expand Down Expand Up @@ -29,6 +29,7 @@ const {
mockBuyer,
accountId,
} = require("../util/mock");
const { FacetCutAction } = require("../../scripts/util/diamond-utils.js");

/**
* Test the Boson Dispute Handler interface
Expand Down Expand Up @@ -2580,4 +2581,46 @@ describe("IBosonDisputeHandler", function () {
});
});
});

// Internal functions, tested with TestDisputeHandlerFacet
context("📋 Internal Dispute Handler Methods", async function () {
let testDisputeHandler;
beforeEach(async function () {
// Deploy test facet and cut the test functions
const TestDisputeHandlerFacet = await ethers.getContractFactory("TestDisputeHandlerFacet");
const testDisputeHandlerFacet = await TestDisputeHandlerFacet.deploy();
await testDisputeHandlerFacet.waitForDeployment();

const protocolDiamondAddress = await disputeHandler.getAddress();
const cutFacetViaDiamond = await getContractAt("DiamondCutFacet", protocolDiamondAddress);

// Define the facet cut
const facetCuts = [
{
facetAddress: await testDisputeHandlerFacet.getAddress(),
action: FacetCutAction.Add,
functionSelectors: [id("finalizeDispute(uint256,uint8)").slice(0, 10)],
},
];

// Send the DiamondCut transaction
await cutFacetViaDiamond.diamondCut(facetCuts, ZeroAddress, "0x");

testDisputeHandler = await getContractAt("TestDisputeHandlerFacet", protocolDiamondAddress);
});

context("👉 finalizeDispute()", async function () {
const invalidFinalStates = ["Resolving", "Escalated"];

invalidFinalStates.forEach((finalState) => {
it(`final state is ${finalState}`, async function () {
const exchangeId = 1;

await expect(
testDisputeHandler.finalizeDispute(exchangeId, DisputeState[finalState])
).to.revertedWithCustomError(bosonErrors, RevertReasons.INVALID_TARGET_DISPUTE_STATE);
});
});
});
});
});
41 changes: 41 additions & 0 deletions test/protocol/ExchangeHandlerTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -8132,4 +8132,45 @@ describe("IBosonExchangeHandler", function () {
});
});
});

// Internal functions, tested with TestExchangeHandlerFacet
context("📋 Internal Exchange Handler Methods", async function () {
let testExchangeHandler;
beforeEach(async function () {
// Deploy test facet and cut the test functions
const TestExchangeHandlerFacet = await ethers.getContractFactory("TestExchangeHandlerFacet");
const testExchangeHandlerFacet = await TestExchangeHandlerFacet.deploy(0);
await testExchangeHandlerFacet.waitForDeployment();

const cutFacetViaDiamond = await getContractAt("DiamondCutFacet", protocolDiamondAddress);

// Define the facet cut
const facetCuts = [
{
facetAddress: await testExchangeHandlerFacet.getAddress(),
action: FacetCutAction.Add,
functionSelectors: [id("finalizeExchange(uint256,uint8)").slice(0, 10)],
},
];

// Send the DiamondCut transaction
await cutFacetViaDiamond.connect(deployer).diamondCut(facetCuts, ZeroAddress, "0x");

testExchangeHandler = await getContractAt("TestExchangeHandlerFacet", protocolDiamondAddress);
});

context("👉 finalizeExchange()", async function () {
const invalidFinalStates = ["Committed", "Redeemed", "Disputed"];

invalidFinalStates.forEach((finalState) => {
it(`final state is ${finalState}`, async function () {
const exchangeId = 1;

await expect(
testExchangeHandler.finalizeExchange(exchangeId, ExchangeState[finalState])
).to.revertedWithCustomError(bosonErrors, RevertReasons.INVALID_TARGET_EXCHANGE_STATE);
});
});
});
});
});

0 comments on commit 3892e75

Please sign in to comment.