Skip to content

Commit

Permalink
feat: Advancer exo behaviors
Browse files Browse the repository at this point in the history
- refs: #10390
  • Loading branch information
0xpatrickdev committed Nov 9, 2024
1 parent c1b9538 commit 9c4e291
Show file tree
Hide file tree
Showing 4 changed files with 373 additions and 147 deletions.
129 changes: 92 additions & 37 deletions packages/fast-usdc/src/exos/advancer.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { AmountMath, BrandShape } from '@agoric/ertp';
import { assertAllDefined } from '@agoric/internal';
import { ChainAddressShape } from '@agoric/orchestration';
import { VowShape } from '@agoric/vow';
Expand Down Expand Up @@ -28,7 +29,7 @@ import { addressTools } from '../utils/address.js';
*/
export const prepareAdvancer = (
zone,
{ chainHub, feed, log, statusManager, vowTools: { watch } },
{ chainHub, feed, log, statusManager, vowTools: { watch, when } },
) => {
assertAllDefined({ feed, statusManager, watch });

Expand All @@ -51,6 +52,7 @@ export const prepareAdvancer = (
* @param {{ destination: ChainAddress; amount: bigint; }} ctx
*/
onFulfilled(result, { destination, amount }) {
// TODO vstorage update?
log(
'Advance transfer fulfilled',
q({ amount, destination, result }).toString(),
Expand All @@ -67,12 +69,15 @@ export const prepareAdvancer = (
return zone.exoClass(
'Fast USDC Advancer',
M.interface('AdvancerI', {
handleTransactionEvent: M.call(CctpTxEvidenceShape).returns(VowShape),
handleTransactionEvent: M.callWhen(CctpTxEvidenceShape).returns(
M.or(M.undefined(), VowShape),
),
}),
/**
* @param {{
* localDenom: Denom;
* poolAccount: HostInterface<OrchestrationAccount<{ chainId: 'agoric' }>>;
* usdcBrand: Brand<'nat'>;
* }} config
*/
config => harden(config),
Expand All @@ -81,56 +86,106 @@ export const prepareAdvancer = (
* XXX For now, assume this is invoked when a new entry is
* observed via the EventFeed.
*
* Returns a Promise for a Vow , since we we `await vt.when()`the
* `poolAccount.getBalance()` call. `getBalance()` interfaces with
* `BankManager` and `ChainHub`, so we can expect the calls to resolve
* promptly. We also don't care how many time `.getBalance` is called,
* and watched vows are not dependent on its result.
*
* This also might return `undefined` (unwrapped) if precondition checks
* fail.
*
* We don't expect any callers to depend on the settlement of
* `handleTransactionEvent` - errors caught are communicated to the
* `StatusManager` - so we don't need to concern ourselves with
* preserving the vow chain for callers.
*
* @param {CctpTxEvidence} evidence
*/
handleTransactionEvent(evidence) {
// XXX assume EventFeed performs validation checks. Advancer will assert uniqueness.

const { recipientAddress } = evidence.aux;
const { EUD } = addressTools.getQueryParams(recipientAddress).params;
if (!EUD) {
statusManager.observe(evidence);
throw makeError(
`recipientAddress does not contain EUD param: ${q(recipientAddress)}`,
);
}
async handleTransactionEvent(evidence) {
await null;
try {
const { recipientAddress } = evidence.aux;
const { EUD } = addressTools.getQueryParams(recipientAddress).params;
// XXX assume EventFeed performs validation checks. Advancer will assert uniqueness.
if (!EUD) {
throw makeError(
`recipientAddress does not contain EUD param: ${q(recipientAddress)}`,
);
}

// TODO #10391 this can throw, and should make a status update in the catch
const { chainId } = chainHub.getChainInfoByAddress(EUD);
// this will throw if the bech32 prefix is not found
const { chainId } = chainHub.getChainInfoByAddress(EUD);

/** @type {DenomAmount} */
const requestedAmount = harden({
denom: this.state.localDenom,
value: BigInt(evidence.tx.amount),
});
/** @type {DenomAmount} */
const requestedAmount = harden({
denom: this.state.localDenom,
value: BigInt(evidence.tx.amount),
});
/**
* Ensure there's enough funds in poolAccount.
*
* It's safe to await here since we don't care how many
* times we call `getBalance`. Our later Vow call - `transferV` and
* its ctx - are also not reliant on the consistency of this value.
*/
const poolBalance = await when(
E(this.state.poolAccount).getBalance(this.state.localDenom),
);

// TODO #10391 ensure there's enough funds in poolAccount
if (
!AmountMath.isGTE(
AmountMath.make(this.state.usdcBrand, poolBalance.value),
AmountMath.make(this.state.usdcBrand, requestedAmount.value),
)
) {
log(
`Insufficient pool funds`,
`Requested ${q(requestedAmount)} but only have ${q(poolBalance)}`,
);
statusManager.observe(evidence);
return;
}

/** @type {ChainAddress} */
const destination = harden({
chainId,
value: EUD,
encoding: /** @type {const} */ ('bech32'),
});
/** @type {ChainAddress} */
const destination = harden({
chainId,
value: EUD,
encoding: /** @type {const} */ ('bech32'),
});

const transferV = E(this.state.poolAccount).transfer(
destination,
requestedAmount,
);
// mark as Advanced since we're initiating the advance
try {
// will throw if we've already .skipped or .advanced this evidence
statusManager.advance(evidence);
} catch (e) {
// only anticipated error is `assertNotSeen`, so
// intercept the catch so we don't call .skip which
// also performs this check
log('Advancer error:', q(e).toString());
return;
}

// mark as Advanced since we're initiating the advance
statusManager.advance(evidence);
const transferV = E(this.state.poolAccount).transfer(
destination,
requestedAmount,
);

return watch(transferV, transferHandler, {
destination,
amount: requestedAmount.value,
});
return watch(transferV, transferHandler, {
destination,
amount: requestedAmount.value,
});
} catch (e) {
log(`Advancer error:`, q(e).toString());
statusManager.observe(evidence);
}
},
},
{
stateShape: harden({
localDenom: M.string(),
poolAccount: M.remotable(),
usdcBrand: BrandShape,
}),
},
);
Expand Down
Loading

0 comments on commit 9c4e291

Please sign in to comment.