Documentation
¶
Index ¶
- Constants
- func NewMsgServer(keeper *Keeper) types.MsgServer
- func NewQueryServer(keeper *Keeper) types.QueryServer
- type Keeper
- func (k *Keeper) BeginBlocker(ctx sdk.Context) error
- func (k Keeper) CalculateVaultTotalAssets(ctx sdk.Context, vault *types.VaultAccount, principal sdk.Coin) (sdkmath.Int, error)
- func (k *Keeper) CanPayoutDuration(ctx sdk.Context, vault *types.VaultAccount, duration int64) (bool, error)
- func (k Keeper) ConvertDepositToSharesInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount, in sdk.Coin) (sdk.Coin, error)
- func (k Keeper) ConvertSharesToRedeemCoin(ctx sdk.Context, vault types.VaultAccount, shares math.Int, redeemDenom string) (sdk.Coin, error)
- func (k *Keeper) CreateVault(ctx sdk.Context, attributes VaultAttributer) (*types.VaultAccount, error)
- func (k *Keeper) EndBlocker(ctx sdk.Context) error
- func (k Keeper) EstimateTotalVaultValue(ctx sdk.Context, vault *types.VaultAccount) (sdk.Coin, error)
- func (k Keeper) ExportGenesis(ctx sdk.Context) *types.GenesisState
- func (k *Keeper) FindVaultAccount(ctx sdk.Context, id string) (*types.VaultAccount, error)
- func (k Keeper) GetAuthority() []byte
- func (k Keeper) GetNAVPerShareInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount) (math.Int, error)
- func (k Keeper) GetTVVInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount) (math.Int, error)
- func (k Keeper) GetVault(ctx sdk.Context, address sdk.AccAddress) (*types.VaultAccount, error)
- func (k *Keeper) GetVaults(ctx context.Context) ([]sdk.AccAddress, error)
- func (k Keeper) InitGenesis(ctx sdk.Context, genState *types.GenesisState)
- func (k Keeper) MigrateVaultAccountPaymentDenomDefaults(ctx sdk.Context) error
- func (k *Keeper) PerformVaultInterestTransfer(ctx sdk.Context, vault *types.VaultAccount) error
- func (k Keeper) SafeAddVerification(ctx sdk.Context, vault *types.VaultAccount) error
- func (k Keeper) SafeEnqueueTimeout(ctx sdk.Context, vault *types.VaultAccount) error
- func (k *Keeper) SetMaxInterestRate(ctx sdk.Context, vault *types.VaultAccount, maxRate string) error
- func (k *Keeper) SetMinInterestRate(ctx sdk.Context, vault *types.VaultAccount, minRate string) error
- func (k *Keeper) SetSwapInEnable(ctx sdk.Context, vault *types.VaultAccount, enabled bool) error
- func (k *Keeper) SetSwapOutEnable(ctx sdk.Context, vault *types.VaultAccount, enabled bool) error
- func (k *Keeper) SetVaultAccount(ctx sdk.Context, vault *types.VaultAccount) error
- func (k *Keeper) SetVaultLookup(ctx context.Context, vault *types.VaultAccount) error
- func (k *Keeper) SetWithdrawalDelay(ctx sdk.Context, vault *types.VaultAccount, delaySeconds uint64, ...) error
- func (k *Keeper) SwapIn(ctx sdk.Context, vaultAddr, recipient sdk.AccAddress, asset sdk.Coin) (*sdk.Coin, error)
- func (k *Keeper) SwapOut(ctx sdk.Context, vaultAddr, owner sdk.AccAddress, shares sdk.Coin, ...) (uint64, error)
- func (k Keeper) ToUnderlyingAssetAmount(ctx sdk.Context, vault types.VaultAccount, in sdk.Coin) (math.Int, error)
- func (k Keeper) UnitPriceFraction(ctx sdk.Context, srcDenom string, vault types.VaultAccount) (num, den math.Int, err error)
- func (k *Keeper) UpdateInterestRates(ctx sdk.Context, vault *types.VaultAccount, currentRate, desiredRate string) error
- func (k Keeper) ValidateInterestRateLimits(minRateStr, maxRateStr string) error
- type VaultAttributer
Constants ¶
const ( Supply = 0 NoFixedSupply = false NoForceTransfer = false NoGovControl = false )
const ( // AutoReconcilePayoutDuration is the time period (in seconds) used to forecast if a vault has // sufficient funds to cover future interest payments. AutoReconcilePayoutDuration = 24 * interest.SecondsPerHour )
const ( // AutoReconcileTimeout is the duration (in seconds) that a vault is considered // recently reconciled and is exempt from automatic interest checks. AutoReconcileTimeout = 20 * interest.SecondsPerHour )
const ( // MaxSwapOutBatchSize is the maximum number of pending swap-out requests // to process in a single EndBlocker invocation. This prevents a large queue // from consuming excessive block time and memory. This is a temporary value // and we will need to do more analysis on a proper batch size. // See https://github.com/ProvLabs/vault/issues/75. MaxSwapOutBatchSize = 100 )
Variables ¶
This section is empty.
Functions ¶
func NewMsgServer ¶
NewMsgServer creates a new MsgServer for the module.
func NewQueryServer ¶
func NewQueryServer(keeper *Keeper) types.QueryServer
Types ¶
type Keeper ¶
type Keeper struct {
AuthKeeper types.AccountKeeper
MarkerKeeper types.MarkerKeeper
BankKeeper types.BankKeeper
Vaults collections.Map[sdk.AccAddress, []byte]
PayoutVerificationSet collections.KeySet[sdk.AccAddress]
PayoutTimeoutQueue *queue.PayoutTimeoutQueue
PendingSwapOutQueue *queue.PendingSwapOutQueue
// contains filtered or unexported fields
}
func NewKeeper ¶
func NewKeeper( cdc codec.Codec, storeService store.KVStoreService, eventService event.Service, addressCodec address.Codec, authority []byte, authKeeper types.AccountKeeper, markerkeeper types.MarkerKeeper, bankkeeper types.BankKeeper, ) *Keeper
NewMsgServer creates a new Keeper for the module.
func (*Keeper) BeginBlocker ¶
BeginBlocker is a hook that is called at the beginning of every block.
func (Keeper) CalculateVaultTotalAssets ¶
func (k Keeper) CalculateVaultTotalAssets(ctx sdk.Context, vault *types.VaultAccount, principal sdk.Coin) (sdkmath.Int, error)
CalculateVaultTotalAssets returns the total value of the vault's assets, including the interest that would have accrued from PeriodStart to the current block time, without mutating state.
If no rate is set or accrual has not started, it returns the provided principal unchanged.
func (*Keeper) CanPayoutDuration ¶
func (k *Keeper) CanPayoutDuration(ctx sdk.Context, vault *types.VaultAccount, duration int64) (bool, error)
CanPayoutDuration determines whether the vault can fulfill the projected interest payment or refund over the given duration based on current reserves and principal TVV.
It returns true when duration <= 0, when accrued interest is zero, when positive interest can be paid from vault reserves, or when negative interest can be fully covered by the principal marker's underlying balance. Otherwise it returns false.
func (Keeper) ConvertDepositToSharesInUnderlyingAsset ¶
func (k Keeper) ConvertDepositToSharesInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount, in sdk.Coin) (sdk.Coin, error)
ConvertSharesToRedeemCoin converts a share amount into a payout coin in redeemDenom using the current TVV and total share supply (pro-rata, single-floor arithmetic).
Steps:
- Look up the unit price fraction for redeemDenom → underlying via UnitPriceFraction.
- Compute the payout in one step using CalculateRedeemProRataFraction(shares, totalShares, TVV, priceNum, priceDen) where TVV is from principal (marker) balances.
Returns a coin in redeemDenom. This function performs calculation only; callers must enforce liquidity/policy. If shares <= 0, returns a zero-amount coin.
func (Keeper) ConvertSharesToRedeemCoin ¶
func (k Keeper) ConvertSharesToRedeemCoin(ctx sdk.Context, vault types.VaultAccount, shares math.Int, redeemDenom string) (sdk.Coin, error)
ConvertSharesToRedeemCoin converts a share amount into a payout coin in redeemDenom using the current TVV and total share supply (both pro-rata, floor arithmetic).
Steps:
- Convert shares → underlying via CalculateAssetsFromShares(shares, totalShares, TVV) where TVV is from principal (marker) balances.
- Convert the resulting underlying amount to redeemDenom via FromUnderlyingAssetAmount (identity fast-path if redeemDenom == vault.UnderlyingAsset).
Returns a coin in redeemDenom. This function performs calculation only; callers must enforce liquidity/policy. If shares <= 0, returns a zero-amount coin.
func (*Keeper) CreateVault ¶
func (k *Keeper) CreateVault(ctx sdk.Context, attributes VaultAttributer) (*types.VaultAccount, error)
CreateVault creates the vault based on the provided attributes.
func (*Keeper) EndBlocker ¶
EndBlocker is a hook that is called at the end of every block.
func (Keeper) EstimateTotalVaultValue ¶ added in v1.0.9
func (k Keeper) EstimateTotalVaultValue(ctx sdk.Context, vault *types.VaultAccount) (sdk.Coin, error)
EstimateTotalVaultValue returns an estimated Total Vault Value (TVV) as a Coin denominated in the vault's underlying asset. It composes two steps without mutating state:
- Reads the current principal-only TVV from on-chain balances at the principal (marker) account (excludes reserves and unpaid interest).
- Applies the vault's interest model to estimate unpaid interest through CalculateVaultTotalAssets, producing a best-effort TVV as of the query block. The result is floor-rounded and suitable for pro-rata calculations.
If the vault is paused, the estimation honors the keeper’s paused logic inside GetTVVInUnderlyingAsset.
Returns an sdk.Coin { Denom: vault.UnderlyingAsset, Amount: ... }.
func (Keeper) ExportGenesis ¶
func (k Keeper) ExportGenesis(ctx sdk.Context) *types.GenesisState
ExportGenesis exports the current state of the vault module.
func (*Keeper) FindVaultAccount ¶
FindVaultAccount retrieves a vault by its address or share denomination.
func (Keeper) GetAuthority ¶
GetAuthority returns the module's authority.
func (Keeper) GetNAVPerShareInUnderlyingAsset ¶
func (k Keeper) GetNAVPerShareInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount) (math.Int, error)
GetNAVPerShareInUnderlyingAsset returns the floor NAV per share in units of vault.UnderlyingAsset.
Paused fast-path:
- If vault.Paused is true, this function short-circuits and returns vault.PausedBalance.Amount (ignores live TVV and share supply).
Computation (when not paused):
- TVV(underlying) is obtained from GetTVVInUnderlyingAsset (includes current vault holdings in underlying units).
- totalShareSupply is taken from vault.TotalShares.Amount (the recorded share supply).
- If total shares == 0, returns 0. Otherwise returns TVV / totalShareSupply (floor).
func (Keeper) GetTVVInUnderlyingAsset ¶
func (k Keeper) GetTVVInUnderlyingAsset(ctx sdk.Context, vault types.VaultAccount) (math.Int, error)
GetTVVInUnderlyingAsset returns the Total Vault Value (TVV) expressed in vault.UnderlyingAsset using floor arithmetic.
Paused fast-path:
- If vault.Paused is true, this function short-circuits and returns vault.PausedBalance.Amount (no balance iteration or NAV conversion).
Source of truth (when not paused):
- TVV sums the balances held at the vault’s *principal* account, i.e. the marker address for vault.PrincipalMarkerAddress().
- The vault account’s own balances are treated as *reserves* and are not included here.
Computation (when not paused):
- Iterate all non-share-denom balances at the marker (principal) account.
- Convert each balance to underlying units via ToUnderlyingAssetAmount.
- Sum the converted amounts (floor at each multiplication/division step).
func (Keeper) GetVault ¶
func (k Keeper) GetVault(ctx sdk.Context, address sdk.AccAddress) (*types.VaultAccount, error)
GetVault finds a vault by a given address.
This function will return nil if nothing exists at this address.
func (Keeper) InitGenesis ¶
func (k Keeper) InitGenesis(ctx sdk.Context, genState *types.GenesisState)
InitGenesis initializes the vault module state from genesis.
func (Keeper) MigrateVaultAccountPaymentDenomDefaults ¶ added in v1.0.14
MigrateVaultAccountPaymentDenomDefaults updates legacy VaultAccount state created prior to v1.0.13 by normalizing empty payment denom fields.
In versions <= v1.0.13, VaultAccount instances could be persisted with an empty PaymentDenom. Newer versions require a valid payment denom, which by default should be the vault’s underlying asset denom.
This migration iterates all accounts in the auth store, identifies VaultAccount instances, and for any vault with an empty PaymentDenom sets it to UnderlyingAsset. Updated vault accounts are validated and re-persisted using the auth keeper.
This function is intended to be executed once from an upgrade handler and is idempotent; running it multiple times will not modify already-migrated state.
func (*Keeper) PerformVaultInterestTransfer ¶
PerformVaultInterestTransfer applies accrued interest between the vault and the marker account if the current block time is beyond PeriodStart.
Interest is settled exclusively in the vault's defined UnderlyingAsset.
- Positive Interest: Paid from vault reserves to the marker. Fails if reserves are insufficient.
- Negative Interest: Refunded from marker principal to the vault. This is bounded by the available balance of the UnderlyingAsset in the marker account.
IMPORTANT: If the vault utilizes composite reserves (holding multiple token types), secondary assets are NOT liquidated or transferred to satisfy interest obligations. If the marker owes negative interest but lacks sufficient liquidity in the UnderlyingAsset, the transfer is capped at the available underlying balance, potentially resulting in a partial payment.
An EventVaultReconcile is emitted upon success. This method does not modify PeriodStart.
func (Keeper) SafeAddVerification ¶
SafeAddVerification clears any existing timeout entry for the given vault (if any), sets the vault's period start to the current block time, clears the period timeout, persists the vault, and stores the vault in the PayoutVerificationSet.
This ensures a vault is not present in both the verification set and timeout queues at the same time. Typically called after enabling interest or completing a reconciliation so the next accrual cycle begins cleanly.
func (Keeper) SafeEnqueueTimeout ¶
SafeEnqueueTimeout clears any existing timeout entry for the given vault (if any), sets the vault's period start to the current block time, sets a new period timeout at (now + AutoReconcileTimeout), persists the vault, and enqueues the timeout entry in the PayoutTimeoutQueue.
This ensures a vault is not present in both the timeout and verification queues at the same time. Typically called after a vault has been marked as payable so it will be revisited after the auto-reconcile window.
func (*Keeper) SetMaxInterestRate ¶
func (k *Keeper) SetMaxInterestRate(ctx sdk.Context, vault *types.VaultAccount, maxRate string) error
SetMaxInterestRate sets the maximum interest rate for a vault. An empty string disables the maximum rate check.
func (*Keeper) SetMinInterestRate ¶
func (k *Keeper) SetMinInterestRate(ctx sdk.Context, vault *types.VaultAccount, minRate string) error
SetMinInterestRate sets the minimum interest rate for a vault. An empty string disables the minimum rate check.
func (*Keeper) SetSwapInEnable ¶
SetSwapInEnable updates the SwapInEnabled flag for a given vault. It updates the vault account in the state and emits an EventToggleSwapIn event.
func (*Keeper) SetSwapOutEnable ¶
SetSwapOutEnable updates the SwapOutEnabled flag for a given vault. It updates the vault account in the state and emits an EventToggleSwapOut event.
func (*Keeper) SetVaultAccount ¶
SetVaultAccount validates and persists a VaultAccount using the auth keeper. Returns an error if validation fails.
func (*Keeper) SetVaultLookup ¶
SetVaultLookup stores a vault in the Vaults collection, keyed by its bech32 address. NOTE: should only be called by genesis and at vault creation. Returns an error if the vault is nil or the address cannot be parsed.
func (*Keeper) SetWithdrawalDelay ¶ added in v1.0.14
func (*Keeper) SwapIn ¶
func (k *Keeper) SwapIn(ctx sdk.Context, vaultAddr, recipient sdk.AccAddress, asset sdk.Coin) (*sdk.Coin, error)
SwapIn handles the process of depositing underlying assets into a vault in exchange for newly minted vault shares.
It performs the following steps:
- Retrieves the vault configuration for the given vault address.
- Verifies that swap-in is enabled for the vault.
- Reconciles any accrued interest from the vault to the marker module (if due).
- Resolves the vault share marker address.
- Validates that the provided underlying asset matches the vault’s configured underlying denom.
- Calculates the number of shares to mint based on the deposit, current supply, and vault balance.
- Mints the computed amount of shares under the vault’s admin authority.
- Withdraws the minted shares from the vault to the recipient address.
- Sends the underlying asset from the recipient to the vault’s marker account.
10. Emits a SwapIn event with metadata for indexing and audit.
Returns the minted share amount on success, or an error if any step fails.
func (*Keeper) SwapOut ¶
func (k *Keeper) SwapOut(ctx sdk.Context, vaultAddr, owner sdk.AccAddress, shares sdk.Coin, redeemDenom string) (uint64, error)
SwapOut validates a swap-out request, calculates the resulting assets, escrows the user's shares, and enqueues a pending withdrawal request to be processed by the EndBlocker. It returns the unique ID of the newly queued request.
func (Keeper) ToUnderlyingAssetAmount ¶
func (k Keeper) ToUnderlyingAssetAmount(ctx sdk.Context, vault types.VaultAccount, in sdk.Coin) (math.Int, error)
ToUnderlyingAssetAmount converts an input coin into its value expressed in vault.UnderlyingAsset using integer floor arithmetic.
Formula:
value_in_underlying = in.Amount * priceNumerator / priceDenominator
where (priceNumerator, priceDenominator) are from UnitPriceFraction(in.Denom → underlying). This performs a pure conversion based on NAV (or identity if denom==underlying). It does not enforce whether the denom is accepted by the vault; such policy checks are handled elsewhere.
func (Keeper) UnitPriceFraction ¶
func (k Keeper) UnitPriceFraction(ctx sdk.Context, srcDenom string, vault types.VaultAccount) (num, den math.Int, err error)
UnitPriceFraction returns the unit price of srcDenom expressed in underlyingAsset as an integer fraction (numerator, denominator) using marker Net Asset Value (NAV).
Semantics
- Forward NAV (srcDenom → underlyingAsset): NAV.Price is total underlying units for NAV.Volume units of srcDenom. 1 srcDenom = NAV.Price.Amount / NAV.Volume underlyingAsset → (num, den) = (NAV.Price.Amount, NAV.Volume).
- Reverse NAV (underlyingAsset → srcDenom): NAV.Price is total srcDenom units for NAV.Volume units of underlyingAsset. 1 srcDenom = NAV.Volume / NAV.Price.Amount underlyingAsset → (num, den) = (NAV.Volume, NAV.Price.Amount).
The result is integer (floor-safe) arithmetic.
Source selection - Identity/peg fast-paths return (1, 1):
- If srcDenom == underlyingAsset
- If underlyingAsset == "uylds.fcc" (temporary 1:1 peg; see https://github.com/ProvLabs/vault/issues/73)
- If vault.PaymentDenom == "uylds.fcc" (temporary 1:1 peg; see https://github.com/ProvLabs/vault/issues/73)
- Otherwise read both forward and reverse NAVs:
- If only one exists, use it.
- If both exist, choose the one with the greater UpdatedBlockHeight (newest).
Errors - If neither NAV exists, return the lookup error (if any) or "nav not found for src/underlying". - For the selected NAV direction:
- Forward: error if NAV.Volume == 0 or NAV.Price.Amount == 0.
- Reverse: error if NAV.Price.Amount == 0 or NAV.Volume == 0.
Returns - (num, den) as math.Int, suitable for floor(x * num / den).
func (*Keeper) UpdateInterestRates ¶
func (k *Keeper) UpdateInterestRates(ctx sdk.Context, vault *types.VaultAccount, currentRate, desiredRate string) error
UpdateInterestRates sets the vault's current and desired interest rates and emits an EventVaultInterestChange. The modified account is persisted via the auth keeper.
func (Keeper) ValidateInterestRateLimits ¶
ValidateInterestRateLimits checks that the provided minimum and maximum interest rates are valid decimal values and that the minimum rate is not greater than the maximum rate. Empty values are treated as unset and pass validation.