diff --git a/packages/vats/src/orch-purse/mock-orch-chain.js b/packages/vats/src/orch-purse/mock-orch-chain.js index 189edd797fa0..e8e671cc1486 100644 --- a/packages/vats/src/orch-purse/mock-orch-chain.js +++ b/packages/vats/src/orch-purse/mock-orch-chain.js @@ -1,3 +1,5 @@ +import { Fail } from '@endo/errors'; +import { Far } from '@endo/pass-style'; import { M } from '@endo/patterns'; import { provideLazy } from '@agoric/store'; @@ -16,14 +18,6 @@ import { * @import {Zone} from '@agoric/base-zone' */ -const makeMinDenomInfo = (chain, _denom) => { - const brand = {}; // TODO the whole thing - return harden({ - brand, - chain, - }); -}; - /** * @param {Zone} zone */ @@ -31,28 +25,37 @@ export const prepareMinOrchestrator = zone => { const makeMinOrchAccount = zone.exoClass( 'MinOrchAccount', MinOrchAccountI, - (chain, ledger) => { - const addrValue = `${ledger.size()}`; - return { chain, ledger, addrValue }; + (chains, denoms, accounts, chain) => { + const acctAddrValue = `${accounts.size()}`; + return { chains, denoms, accounts, chain, acctAddrValue }; }, { getAddress() { - const { chain, addrValue: value } = this.state; + const { chain, acctAddrValue } = this.state; const { chainId } = chain.getChainInfo(); - return harden({ chainId, value }); + return harden({ chainId, value: acctAddrValue }); }, - getBalances() { - const { ledger, addrValue } = this.state; - const { balances } = ledger.get(addrValue); + async getBalances() { + const { accounts, acctAddrValue } = this.state; + const { balances } = accounts.get(acctAddrValue); return [...balances.values()]; }, - getBalance(denom) { - const { ledger, addrValue } = this.state; - const { balances } = ledger.get(addrValue); + async getBalance(denom) { + const { accounts, acctAddrValue } = this.state; + const { balances } = accounts.get(acctAddrValue); return balances.get(denom); }, - transfer(destAddr, denomAmount) { + async transfer(destAddr, denomAmount) { + const { denoms, accounts, acctAddrValue } = this.state; + const { balances } = accounts.get(acctAddrValue); + + const { chainId: destChainId, value: destAcctAddrValue } = destAddr; + const { denom, value: deltaValue } = denomAmount; + const { value: myOldBalanceValue } = balances.get(denom); + myOldBalanceValue >= deltaValue || + Fail`overdrawn ${myOldBalanceValue} - ${deltaValue}`; + const myNewBalanceValue = myOldBalanceValue - deltaValue; }, }, ); @@ -60,32 +63,25 @@ export const prepareMinOrchestrator = zone => { const makeMinChain = zone.exoClass( 'MinChain', MinChainI, - chainName => ({ + (chains, denoms, chainName) => ({ + chains, + denoms, chainId: chainName, - denoms: zone.mapStore('denoms', { - keyShape: M.string(), // denom - valueShape: MinDenomInfoShape, - }), - ledger: zone.mapStore('accounts', { - keyShape: M.string(), // addrValue - valueShape: { - account: MinOrchAccountShape, - balances: M.remotable('balances'), - }, - }), }), { - getChainInfo() { + async getChainInfo() { const { chainId } = this.state; return harden({ chainId }); }, - makeAccount() { - const { ledger } = this.state; + async makeAccount() { const { self } = this; - const account = makeMinOrchAccount(self, ledger); - const { value: addrValue } = account.getAddress(); - ledger.init( - addrValue, + const { chains, denoms, chainId } = this.state; + const { accounts } = chains.get(chainId); + + const account = makeMinOrchAccount(chains, denoms, accounts, self); + const { value: acctAddrValue } = account.getAddress(); + accounts.init( + acctAddrValue, harden({ account, balances: zone.mapStore('balances', { @@ -96,16 +92,19 @@ export const prepareMinOrchestrator = zone => { ); return account; }, - getDenomInfo(denom) { - const { denoms } = this.state; - const { self } = this; - return provideLazy(denoms, denom, d => makeMinDenomInfo(self, d)); - }, - asAmount(denomAmount) { + async makeDenom(denom) { const { self } = this; - const { denom, value } = denomAmount; - const { brand } = self.getDenomInfo(denom); - return AmountMath.make(brand, value); + const { denoms } = this.state; + // TODO make a full OrchIssuerKit + // denoms.init will error if denom is already taken + denoms.init( + denom, + harden({ + brand: Far(`fake ${denom} brand`, {}), + chain: self, + }), + ); + // TODO return orchIssuerKit }, }, ); @@ -115,14 +114,48 @@ export const prepareMinOrchestrator = zone => { MinOrchestratorI, () => ({ chains: zone.mapStore('chains', { - keyShape: M.string(), - valueShape: MinChainShape, + keyShape: M.string(), // chainName === chainId + valueShape: { + chain: MinChainShape, + denoms: M.remotable('denoms'), + accounts: M.remotable('accounts'), + }, + }), + denoms: zone.mapStore('denoms', { + keyShape: M.string(), // denom + valueShape: MinDenomInfoShape, }), }), { getChain(chainName) { - const { chains } = this.state; - return provideLazy(chains, chainName, makeMinChain); + const { chains, denoms } = this.state; + const { chain } = provideLazy(chains, chainName, cName => { + const ch = makeMinChain(chains, denoms, cName); + chains.init( + chainName, + harden({ + ch, + accounts: zone.mapStore('accounts', { + keyShape: M.string(), // acctAddrValue + valueShape: { + account: MinOrchAccountShape, + balances: M.remotable('balances'), + }, + }), + }), + ); + }); + return chain; + }, + getDenomInfo(denom) { + const { denoms } = this.state; + return denoms.get(denom); + }, + asAmount(denomAmount) { + const { self } = this; + const { denom, value } = denomAmount; + const { brand } = self.getDenomInfo(denom); + return AmountMath.make(brand, value); }, }, ); diff --git a/packages/vats/src/orch-purse/typeGuards.js b/packages/vats/src/orch-purse/typeGuards.js index 8ad2dc75e6eb..9ed6794074a2 100644 --- a/packages/vats/src/orch-purse/typeGuards.js +++ b/packages/vats/src/orch-purse/typeGuards.js @@ -28,7 +28,7 @@ export const MinOrchestratorShape = M.remotable('MinOrchestrator'); */ export const MinChainAcctAddrShape = harden({ chainId: M.string(), - value: M.string(), + value: M.string(), // acctAddrValue }); /** @@ -62,9 +62,8 @@ export const MinChainI = M.interface('MinChain', { getChainInfo: M.call().returns(M.eref(ChainInfoShape)), makeAccount: M.call().returns(M.eref(MinOrchAccountShape)), - // In the real API, these are on OrchestratorI - getDenomInfo: M.call(DenomShape).returns(MinDenomInfoShape), - asAmount: M.call(DenomAmountShape).returns(AmountShape), + // TODO returns(M.eref(OrchIssuerKitShape)) + makeDenom: M.call(DenomShape).returns(M.eref(M.any())), }); /** @@ -73,6 +72,8 @@ export const MinChainI = M.interface('MinChain', { */ export const MinOrchestratorI = M.interface('MinOrchestrator', { getChain: M.call(M.string()).returns(M.eref(MinChainShape)), + getDenomInfo: M.call(DenomShape).returns(MinDenomInfoShape), + asAmount: M.call(DenomAmountShape).returns(AmountShape), }); // //////////////////////// ERTP-like ////////////////////////////////////////// diff --git a/packages/vats/src/orch-purse/types.ts b/packages/vats/src/orch-purse/types.ts index 41bb0e566275..cb097d10ed0c 100644 --- a/packages/vats/src/orch-purse/types.ts +++ b/packages/vats/src/orch-purse/types.ts @@ -13,7 +13,7 @@ import type { Denom, DenomAmount, ChainAddress } from '@agoric/orchestration'; */ export type MinChainAcctAddr = { chainId: string; - value: string; + value: string; // acctAddrValue }; /** @@ -50,9 +50,7 @@ export type MinChain = { getChainInfo(): ERef; makeAccount(): ERef; - // In the real API, these are on Orchestrator - getDenomInfo(denom: Denom): MinDenomInfo; - asAmount(denomAmount: DenomAmount): Amount; + makeDenom(denom: Denom): ERef; // TODO OrchIssuerKit }; /** @@ -60,6 +58,8 @@ export type MinChain = { */ export type MinOrchestrator = { getChain(chainName: string): ERef; + getDenomInfo(denom: Denom): MinDenomInfo; + asAmount(denomAmount: DenomAmount): Amount; }; // //////////////////////// ERTP-like //////////////////////////////////////////