Collecting Orders
The Catalyst order server is the recommended integration surface. However, if you are interested in the lowest latency, decentralized, and permissionless order discovery, you can read some orders on-chain.
Types
type OutputDescription = { bytes32 remoteOracle: string; bytes32 remoteFiller: string; uint256 chainId: number; bytes32 token: string; uint256 amount: string; bytes32 recipient: string; bytes remoteCall: string; bytes fulfillmentContext: string;};
type CatalystCompactOrder = { user: string; nonce: number; originChainId: number; fillDeadline: number; localOracle: string; inputs: [string, string][]; outputs: OutputDescription[];};
type OrderDto = { order_type: "CatalystCompactOrder" | ...; order: CatalystCompactOrder | ...; quote: { fromAsset: string; toAsset: string; toPrice: string; fromPrice: string; intermediary: "USD" | "EUR" | "BTC" | string; // explicit types as example discount: string; }[]; sponsorSignature: string | ""; allocatorSignature: string | ""; submitTime: number;};
WebSocket Subscription
The Catalyst order server delivers orders through a WebSocket subscription, allowing for real-time updates without the need to continuously poll the order server. By leveraging WebSocket, the Catalyst order server broadcasts new orders as they arrive, offering a significant reduction in latency but at the cost of solvers having to locally filter incoming data.
Below is a simplified implementation in pure JavaScript that demonstrates how to connect to the WebSocket server, handle incoming messages, respond to ping events, and automatically attempt to reconnect if the connection is lost.
(TODO: update TypeScript block)
const WebSocket = require("ws");
// Configuration variablesconst wsUri = process.env.ORDER_SERVER_WS_URI; // Set your WebSocket server URIconst apiKey = process.env.ORDER_SERVER_API_KEY; // Set your API keyconst reconnectInterval = 5000; // Reconnect interval in milliseconds
// Function to connect to the WebSocket serverfunction connectToOrderServer() { const ws = new WebSocket(wsUri, { headers: { "x-api-key": apiKey, }, });
ws.on("open", () => { console.log("Connected to WebSocket server"); });
ws.on("message", (data) => { try { const parsedData = JSON.parse(data.toString()); console.log("Received message:", parsedData);
switch (parsedData.event) { case "ping": // You will be automatically disconnected if you don't respond to ping messages ws.send(JSON.stringify({ event: "pong" })); break; case "order": handleReceiveOrder(parsedData, ws); break; default: console.log("Unknown message type:", parsedData); } } catch (error) { console.error("Error parsing JSON:", error); } });
ws.on("error", (error) => { console.error("WebSocket error:", error); });
ws.on("close", () => { console.log("Disconnected from WebSocket");
console.log("Attempting to reconnect..."); setTimeout(() => { ws.close(); // Close any existing connection connectToOrderServer(); // Attempt to reconnect }, reconnectInterval); });}
// Function to handle ordersfunction handleReceiveOrder(data, ws) { console.log("Handling order:", data); // Add your custom handling logic here}
// Start listening to the order serverconnectToOrderServer();
Evaluating Orders
After fetching an order, the solver must thoroughly evaluate it to determine its viability and potential execution. To facilitate this evaluation, several contextual pointers are available within the returned order data. Key aspects to consider include:
- Quote Validation: Use the
OrderDto.quote
field to access the price context, which provides the pricing details for the inputs and outputs of the order. If you trust the order server, you can primarily rely on this quote to validate the order’s pricing. However, it’s crucial to verify that the solver supports the specific origin chain (OrderDto.order.originChainId
) and output chains (OrderDto.order.orderData.outputs[...].chainId
) as well as their respective tokens (input[].token
andoutput[].token
). These parameters are guaranteed to be present across all order types.
Evaluating orders carefully ensures that solvers can accurately determine the feasibility of executing an order, adhere to exclusivity rules, and avoid conflicts, thereby maintaining the integrity and efficiency of the order fulfillment process.
On-chain Order Broadcast
Output settlement schemes supporting on-chain orders like CompactSettlerWithDeposit.sol
allow anyone to broadcast and collect orders on-chain through the Deposit
event:
event Deposited(bytes32 orderId, CatalystCompactOrder order);