Skip to main content

Documentation Index

Fetch the complete documentation index at: https://kleros-mintlify-changelog-2026-05-12-1778458371.mintlify.app/llms.txt

Use this file to discover all available pages before exploring further.

Overview

Kleros Court V2 runs on Arbitrum One (the “home chain”). If your arbitrable contract is on a different chain (Ethereum Mainnet, Gnosis Chain, or another L2), you interact with Kleros through the Foreign Gateway deployed on your chain. The gateway handles cross-chain communication via Vea. From your contract’s perspective, the Foreign Gateway acts as the arbitrator you call createDispute() on it and receive rulings from it.

Gateway Architecture

Your Chain (Foreign)                    Arbitrum (Home)
┌──────────────┐                       ┌──────────────┐
│  Your        │──createDispute()──►   │              │
│  Arbitrable  │                       │  Kleros      │
│  Contract    │◄──rule()───────────   │  Core        │
└──────┬───────┘                       └──────┬───────┘
       │                                      │
┌──────▼───────┐                       ┌──────▼───────┐
│  Foreign     │◄───── Vea Bridge ────►│  Home        │
│  Gateway     │                       │  Gateway     │
└──────────────┘                       └──────────────┘
The Foreign Gateway:
  • Mirrors arbitrationCost() locally so your contract can check fees without cross-chain calls
  • Assigns a local disputeID for backward compatibility
  • Computes a disputeHash to uniquely identify disputes across chains
  • Receives rulings from the Home Gateway via Vea and calls rule() on your contract

Creating a Dispute from a Foreign Chain

Step 1: Implement IArbitrableV2

Your contract must implement the IArbitrableV2 interface:
import {IArbitrableV2, IArbitratorV2} from "@kleros/kleros-v2-contracts/interfaces/IArbitrableV2.sol";

contract MyArbitrable is IArbitrableV2 {
    IArbitratorV2 public immutable foreignGateway;

    constructor(address _foreignGateway) {
        foreignGateway = IArbitratorV2(_foreignGateway);
    }

    function createDispute() external payable {
        uint256 cost = foreignGateway.arbitrationCost(extraData);
        require(msg.value >= cost, "Insufficient fee");

        uint256 disputeID = foreignGateway.createDispute{value: msg.value}(
            numberOfChoices,
            extraData
        );
        // Store disputeID and map to your internal state
    }

    function rule(uint256 _disputeID, uint256 _ruling) external override {
        require(msg.sender == address(foreignGateway), "Only arbitrator");
        // Enforce the ruling
    }
}

Step 2: Check Arbitration Cost

The Foreign Gateway mirrors the fee schedule from Kleros Core, so arbitrationCost() returns the correct fee without a cross-chain call:
// extraData encodes: courtID (uint96) + numberOfJurors (uint256)
bytes memory extraData = abi.encodePacked(
    uint96(1),    // General Court
    uint256(3)    // 3 jurors
);

uint256 fee = foreignGateway.arbitrationCost(extraData);

Step 3: Receive the Ruling

When Kleros Court reaches a final ruling, the Home Gateway sends it via Vea to the Foreign Gateway. The Foreign Gateway calls rule() on your contract:
function rule(uint256 _disputeID, uint256 _ruling) external override {
    require(msg.sender == address(foreignGateway), "Only arbitrator");
    // _ruling: 0 = refused to rule, 1..N = ruling options
    enforceRuling(_disputeID, _ruling);
}

Cross-Chain Dispute Identifier

Disputes have different IDs on each chain:
IdentifierChainPurpose
localDisputeIDForeign chainIncremental ID assigned by Foreign Gateway for backward compatibility
disputeHashBoth chainsUnique hash computed from (chainId, blockHash, arbitrable, disputeID, choices, extraData)
homeDisputeIDArbitrumID assigned by KlerosCore
The Foreign Gateway maps between localDisputeID and disputeHash. The Home Gateway maps between disputeHash and homeDisputeID.

Evidence Submission

Evidence can be submitted on either the foreign chain or the home chain. The Court V2 frontend consolidates evidence from both chains by monitoring event logs from both gateways. On the foreign chain, emit evidence events from your arbitrable contract:
emit Evidence(foreignGateway, disputeID, msg.sender, evidenceURI);

Timing Considerations

PhaseTiming
Foreign → Home (dispute creation)5–10 minutes via Simple Bridge
Home → Foreign (ruling relay, happy path)Configurable challenge period, typically faster than native bridge
Home → Foreign (ruling relay, unhappy path)~7 days (native Arbitrum → Ethereum bridge)
Design your application to handle the delay between dispute creation and ruling delivery. The evidence period on Kleros Court provides sufficient buffer in most cases.

Foreign Gateway Contract

The Foreign Gateway implements IArbitratorV2 from your contract’s perspective:
interface IForeignGateway is IArbitratorV2 {
    function homeChainID() external view returns (uint256);
    function homeGateway() external view returns (address);

    // Inherited from IArbitratorV2:
    function createDispute(uint256 _choices, bytes calldata _extraData)
        external payable returns (uint256 disputeID);
    function arbitrationCost(bytes calldata _extraData)
        external view returns (uint256 cost);
}
Foreign Gateway contract addresses may change during the V2 beta period as contracts are upgraded. Always verify the current deployment before integrating.