Skip to content

Commit 9e90738

Browse files
committed
feat(fast-usdc): settler handles OBSERVED tx
1 parent 9ce5a1c commit 9e90738

File tree

3 files changed

+100
-9
lines changed

3 files changed

+100
-9
lines changed

packages/fast-usdc/src/exos/settler.js

+32-6
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,17 @@
1+
import { AmountMath } from '@agoric/ertp';
12
import { assertAllDefined, makeTracer } from '@agoric/internal';
23
import { atob } from '@endo/base64';
34
import { makeError, q } from '@endo/errors';
5+
import { E } from '@endo/far';
46
import { M } from '@endo/patterns';
57

6-
import { AmountMath } from '@agoric/ertp';
8+
import { PendingTxStatus } from '../constants.js';
79
import { addressTools } from '../utils/address.js';
810
import { makeFeeTools } from '../utils/fees.js';
911

1012
/**
1113
* @import {FungibleTokenPacketData} from '@agoric/cosmic-proto/ibc/applications/transfer/v2/packet.js';
12-
* @import {Denom, OrchestrationAccount, LocalAccountMethods} from '@agoric/orchestration';
14+
* @import {Denom, OrchestrationAccount, LocalAccountMethods, ChainHub} from '@agoric/orchestration';
1315
* @import {WithdrawToSeat} from '@agoric/orchestration/src/utils/zoe-tools'
1416
* @import {IBCChannelID, VTransferIBCEvent} from '@agoric/vats';
1517
* @import {Zone} from '@agoric/zone';
@@ -30,17 +32,21 @@ const trace = makeTracer('Settler');
3032
* @param {FeeConfig} caps.feeConfig
3133
* @param {HostOf<WithdrawToSeat>} caps.withdrawToSeat
3234
* @param {import('@agoric/vow').VowTools} caps.vowTools
35+
* @param {ChainHub} caps.chainHub
3336
*/
3437
export const prepareSettler = (
3538
zone,
36-
{ statusManager, USDC, zcf, feeConfig, withdrawToSeat, vowTools },
39+
{ statusManager, USDC, zcf, feeConfig, withdrawToSeat, vowTools, chainHub },
3740
) => {
3841
assertAllDefined({ statusManager });
3942
return zone.exoClass(
4043
'Fast USDC Settler',
4144
M.interface('SettlerI', {
4245
monitorTransfers: M.callWhen().returns(M.any()),
4346
receiveUpcall: M.call(M.record()).returns(M.promise()),
47+
settleSansFees: M.call(M.string(), M.string(), M.nat()).returns(
48+
M.promise(),
49+
),
4450
}),
4551
/**
4652
* @param {{
@@ -111,6 +117,11 @@ export const prepareSettler = (
111117
);
112118
}
113119

120+
const pending = statusManager.lookupPending(sender, amountInt);
121+
if (pending.find(it => it.status === PendingTxStatus.Observed)) {
122+
return this.self.settleSansFees(sender, EUD, amountInt);
123+
}
124+
114125
// Disperse funds
115126

116127
const { repayer, settlementAccount } = this.state;
@@ -137,10 +148,25 @@ export const prepareSettler = (
137148
repayer.repay(settlingSeat, split);
138149

139150
// update status manager, marking tx `SETTLED`
140-
statusManager.settle(
141-
/** @type {NobleAddress} */ (tx.sender),
142-
amountInt,
151+
statusManager.settle(sender, amountInt);
152+
},
153+
/**
154+
* @param {NobleAddress} sender
155+
* @param {string} EUD
156+
* @param {bigint} amountInt
157+
*/
158+
async settleSansFees(sender, EUD, amountInt) {
159+
const { settlementAccount } = this.state;
160+
161+
const dest = chainHub.makeChainAddress(EUD);
162+
163+
const txfrV = E(settlementAccount).transfer(
164+
dest,
165+
AmountMath.make(USDC, amountInt),
143166
);
167+
await vowTools.when(txfrV); // TODO: watch, handle failure
168+
169+
statusManager.settle(sender, amountInt);
144170
},
145171
},
146172
{

packages/fast-usdc/test/exos/settler.test.ts

+64-1
Original file line numberDiff line numberDiff line change
@@ -64,13 +64,17 @@ const makeTestContext = async t => {
6464
return vowTools.asVow(() => {});
6565
};
6666

67+
const { chainHub } = common.facadeServices;
68+
chainHub.registerChain('dydx', fetchedChainInfo.dydx);
69+
chainHub.registerChain('osmosis', fetchedChainInfo.osmosis);
6770
const makeSettler = prepareSettler(zone.subZone('settler'), {
6871
statusManager,
6972
USDC: usdc.brand,
7073
zcf,
7174
withdrawToSeat: mockWithdrawToSeat,
7275
feeConfig: common.commonPrivateArgs.feeConfig,
7376
vowTools: common.bootstrap.vowTools,
77+
chainHub,
7478
});
7579

7680
const defaultSettlerParams = harden({
@@ -126,7 +130,7 @@ const makeTestContext = async t => {
126130

127131
const test = anyTest as TestFn<Awaited<ReturnType<typeof makeTestContext>>>;
128132

129-
test.before(async t => (t.context = await makeTestContext(t)));
133+
test.beforeEach(async t => (t.context = await makeTestContext(t)));
130134

131135
test('happy path: disburse to LPs; StatusManager removes tx', async t => {
132136
const {
@@ -218,6 +222,65 @@ test('happy path: disburse to LPs; StatusManager removes tx', async t => {
218222
// TODO, confirm vstorage write for TxStatus.SETTLED
219223
});
220224

225+
test('slow path: disburse to LPs; StatusManager removes tx', async t => {
226+
const {
227+
common,
228+
makeSettler,
229+
statusManager,
230+
defaultSettlerParams,
231+
repayer,
232+
simulate,
233+
accounts,
234+
peekCalls,
235+
} = t.context;
236+
const { usdc } = common.brands;
237+
const { feeConfig } = common.commonPrivateArgs;
238+
239+
const settler = makeSettler({
240+
repayer,
241+
settlementAccount: accounts.settlement.account,
242+
...defaultSettlerParams,
243+
});
244+
245+
const cctpTxEvidence = simulate.observe();
246+
t.deepEqual(
247+
statusManager.lookupPending(
248+
cctpTxEvidence.tx.forwardingAddress,
249+
cctpTxEvidence.tx.amount,
250+
),
251+
[{ ...cctpTxEvidence, status: PendingTxStatus.Observed }],
252+
'statusManager shows this tx is only observed',
253+
);
254+
255+
t.log('Simulate incoming IBC settlement');
256+
void settler.receiveUpcall(MockVTransferEvents.AGORIC_PLUS_OSMO());
257+
await eventLoopIteration();
258+
259+
t.log('review settler interactions with other components');
260+
t.deepEqual(peekCalls(), []);
261+
t.deepEqual(accounts.settlement.callLog, [
262+
[
263+
'transfer',
264+
{
265+
chainId: 'osmosis-1',
266+
encoding: 'bech32',
267+
value: 'osmo183dejcnmkka5dzcu9xw6mywq0p2m5peks28men',
268+
},
269+
usdc.units(150),
270+
],
271+
]);
272+
273+
t.deepEqual(
274+
statusManager.lookupPending(
275+
cctpTxEvidence.tx.forwardingAddress,
276+
cctpTxEvidence.tx.amount,
277+
),
278+
[],
279+
'SETTLED entry removed from StatusManger',
280+
);
281+
// TODO, confirm vstorage write for TxStatus.SETTLED
282+
});
283+
221284
test.todo("StatusManager does not receive update when we can't settle");
222285

223286
test.todo('Observed -> Settle flow');

packages/fast-usdc/test/mocks.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -44,9 +44,10 @@ export const prepareMockOrchAccounts = (
4444
OrchestrationAccount<{ chainId: 'agoric' }>
4545
>;
4646

47+
const settlementCallLog = [] as any[];
4748
const settlementAccountMock = zone.exo('Mock Settlement Account', undefined, {
48-
someMethod() {
49-
throw Error('todo');
49+
transfer(...args) {
50+
settlementCallLog.push(harden(['transfer', ...args]));
5051
},
5152
});
5253
const settlementAccount = settlementAccountMock as unknown as HostInterface<
@@ -59,6 +60,7 @@ export const prepareMockOrchAccounts = (
5960
},
6061
settlement: {
6162
account: settlementAccount,
63+
callLog: settlementCallLog,
6264
},
6365
};
6466
};

0 commit comments

Comments
 (0)