Skip to content

Commit

Permalink
add support for RHP4 accounts
Browse files Browse the repository at this point in the history
  • Loading branch information
n8maninger authored and ChrisSchinnerl committed Nov 25, 2024
1 parent de44551 commit 8e67014
Show file tree
Hide file tree
Showing 8 changed files with 452 additions and 18 deletions.
5 changes: 5 additions & 0 deletions .changeset/add_support_for_rhp4_accounts.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
default: minor
---

# Add support for RHP4 accounts
22 changes: 22 additions & 0 deletions host/contracts/accounts.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package contracts

import (
proto4 "go.sia.tech/core/rhp/v4"
"go.sia.tech/core/types"
)

// AccountBalance returns the balance of an account.
func (cm *Manager) AccountBalance(account proto4.Account) (types.Currency, error) {
return cm.store.RHP4AccountBalance(account)
}

// CreditAccountsWithContract atomically revises a contract and credits the accounts
// returning the new balance of each account.
func (cm *Manager) CreditAccountsWithContract(deposits []proto4.AccountDeposit, contractID types.FileContractID, revision types.V2FileContract) ([]types.Currency, error) {
return cm.store.RHP4CreditAccounts(deposits, contractID, revision)
}

// DebitAccount debits an account.
func (cm *Manager) DebitAccount(account proto4.Account, usage proto4.Usage) error {
return cm.store.RHP4DebitAccount(account, usage)
}
7 changes: 7 additions & 0 deletions host/contracts/persist.go
Original file line number Diff line number Diff line change
Expand Up @@ -55,5 +55,12 @@ type (
// ExpireV2ContractSectors removes sector roots for any v2 contracts that are
// rejected or past their proof window.
ExpireV2ContractSectors(height uint64) error

// RHP4AccountBalance returns the balance of an account.
RHP4AccountBalance(proto4.Account) (types.Currency, error)
// RHP4CreditAccounts atomically revises a contract and credits the accounts
RHP4CreditAccounts([]proto4.AccountDeposit, types.FileContractID, types.V2FileContract) (balances []types.Currency, err error)
// RHP4DebitAccount debits an account.
RHP4DebitAccount(proto4.Account, proto4.Usage) error
}
)
225 changes: 210 additions & 15 deletions persist/sqlite/accounts.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,128 @@ import (
"time"

rhp3 "go.sia.tech/core/rhp/v3"
proto4 "go.sia.tech/core/rhp/v4"
"go.sia.tech/core/types"
"go.sia.tech/hostd/host/accounts"
"go.sia.tech/hostd/host/contracts"
"go.uber.org/zap"
)

// RHP4AccountBalance returns the balance of the account with the given ID.
func (s *Store) RHP4AccountBalance(account proto4.Account) (balance types.Currency, err error) {
err = s.transaction(func(tx *txn) error {
return tx.QueryRow(`SELECT balance FROM accounts WHERE account_id=$1`, encode(account)).Scan(decode(&balance))
})
return
}

// RHP4DebitAccount debits the account with the given ID.
func (s *Store) RHP4DebitAccount(account proto4.Account, usage proto4.Usage) error {
return s.transaction(func(tx *txn) error {
var dbID int64
var balance types.Currency
err := tx.QueryRow(`SELECT id, balance FROM accounts WHERE account_id=$1`, encode(account)).Scan(&dbID, decode(&balance))
if err != nil {
return fmt.Errorf("failed to query balance: %w", err)
}

total := usage.RenterCost()
balance, underflow := balance.SubWithUnderflow(total)
if underflow {
return fmt.Errorf("insufficient balance")
}

_, err = tx.Exec(`UPDATE accounts SET balance=$1 WHERE id=$2`, encode(balance), dbID)
if err != nil {
return fmt.Errorf("failed to update balance: %w", err)
} else if err := updateV2ContractFunding(tx, dbID, usage); err != nil {
return fmt.Errorf("failed to update contract funding: %w", err)
}
return nil
})
}

// RHP4CreditAccounts credits the accounts with the given deposits and revises
// the contract.
func (s *Store) RHP4CreditAccounts(deposits []proto4.AccountDeposit, contractID types.FileContractID, revision types.V2FileContract) (balances []types.Currency, err error) {
err = s.transaction(func(tx *txn) error {
getBalanceStmt, err := tx.Prepare(`SELECT balance FROM accounts WHERE account_id=$1`)
if err != nil {
return fmt.Errorf("failed to prepare get balance statement: %w", err)
}
defer getBalanceStmt.Close()

updateBalanceStmt, err := tx.Prepare(`INSERT INTO accounts (account_id, balance, expiration_timestamp) VALUES ($1, $2, $3) ON CONFLICT (account_id) DO UPDATE SET balance=EXCLUDED.balance, expiration_timestamp=EXCLUDED.expiration_timestamp RETURNING id`)
if err != nil {
return fmt.Errorf("failed to prepare update balance statement: %w", err)
}
defer updateBalanceStmt.Close()

getFundingAmountStmt, err := tx.Prepare(`SELECT amount FROM contract_v2_account_funding WHERE contract_id=$1 AND account_id=$2`)
if err != nil {
return fmt.Errorf("failed to prepare get funding amount statement: %w", err)
}
defer getFundingAmountStmt.Close()

updateFundingAmountStmt, err := tx.Prepare(`INSERT INTO contract_v2_account_funding (contract_id, account_id, amount) VALUES ($1, $2, $3) ON CONFLICT (contract_id, account_id) DO UPDATE SET amount=EXCLUDED.amount`)
if err != nil {
return fmt.Errorf("failed to prepare update funding amount statement: %w", err)
}
defer updateFundingAmountStmt.Close()

var contractDBID int64
err = tx.QueryRow(`SELECT id FROM contracts_v2 WHERE contract_id=$1`, encode(contractID)).Scan(&contractDBID)
if err != nil {
return fmt.Errorf("failed to get contract ID: %w", err)
}

var usage proto4.Usage
var createdAccounts int
for _, deposit := range deposits {
var balance types.Currency
err := getBalanceStmt.QueryRow(encode(deposit.Account)).Scan(decode(&balance))
if err != nil && !errors.Is(err, sql.ErrNoRows) {
return fmt.Errorf("failed to get balance: %w", err)
} else if errors.Is(err, sql.ErrNoRows) {
createdAccounts++
}

balance = balance.Add(deposit.Amount)

var accountDBID int64
err = updateBalanceStmt.QueryRow(encode(deposit.Account), encode(balance), encode(time.Now().Add(90*24*time.Hour))).Scan(&accountDBID)
if err != nil {
return fmt.Errorf("failed to update balance: %w", err)
}
balances = append(balances, balance)

var fundAmount types.Currency
if err := getFundingAmountStmt.QueryRow(contractDBID, accountDBID).Scan(decode(&fundAmount)); err != nil && !errors.Is(err, sql.ErrNoRows) {
return fmt.Errorf("failed to get funding amount: %w", err)
}
fundAmount = fundAmount.Add(deposit.Amount)
if _, err := updateFundingAmountStmt.Exec(contractDBID, accountDBID, encode(fundAmount)); err != nil {
return fmt.Errorf("failed to update funding amount: %w", err)
}
usage.AccountFunding = usage.AccountFunding.Add(deposit.Amount)
}

_, err = reviseV2Contract(tx, contractID, revision, usage)
if err != nil {
return fmt.Errorf("failed to revise contract: %w", err)
}

if err := incrementCurrencyStat(tx, metricAccountBalance, usage.AccountFunding, false, time.Now()); err != nil {
return fmt.Errorf("failed to increment balance metric: %w", err)
} else if err := incrementNumericStat(tx, metricActiveAccounts, createdAccounts, time.Now()); err != nil {
return fmt.Errorf("failed to increment active accounts metric: %w", err)
}

return nil
})
return
}

// AccountBalance returns the balance of the account with the given ID.
func (s *Store) AccountBalance(accountID rhp3.Account) (balance types.Currency, err error) {
err = s.transaction(func(tx *txn) error {
Expand All @@ -26,20 +142,6 @@ func (s *Store) AccountBalance(accountID rhp3.Account) (balance types.Currency,
return
}

func incrementContractAccountFunding(tx *txn, accountID, contractID int64, amount types.Currency) error {
var fundingValue types.Currency
err := tx.QueryRow(`SELECT amount FROM contract_account_funding WHERE contract_id=$1 AND account_id=$2`, contractID, accountID).Scan(decode(&fundingValue))
if err != nil && !errors.Is(err, sql.ErrNoRows) {
return fmt.Errorf("failed to get fund amount: %w", err)
}
fundingValue = fundingValue.Add(amount)
_, err = tx.Exec(`INSERT INTO contract_account_funding (contract_id, account_id, amount) VALUES ($1, $2, $3) ON CONFLICT (contract_id, account_id) DO UPDATE SET amount=EXCLUDED.amount`, contractID, accountID, encode(fundingValue))
if err != nil {
return fmt.Errorf("failed to update funding source: %w", err)
}
return nil
}

// CreditAccountWithContract adds the specified amount to the account with the given ID.
func (s *Store) CreditAccountWithContract(fund accounts.FundAccountWithContract) error {
return s.transaction(func(tx *txn) error {
Expand Down Expand Up @@ -181,6 +283,20 @@ func (s *Store) PruneAccounts(height uint64) error {
})
}

func incrementContractAccountFunding(tx *txn, accountID, contractID int64, amount types.Currency) error {
var fundingValue types.Currency
err := tx.QueryRow(`SELECT amount FROM contract_account_funding WHERE contract_id=$1 AND account_id=$2`, contractID, accountID).Scan(decode(&fundingValue))
if err != nil && !errors.Is(err, sql.ErrNoRows) {
return fmt.Errorf("failed to get fund amount: %w", err)
}
fundingValue = fundingValue.Add(amount)
_, err = tx.Exec(`INSERT INTO contract_account_funding (contract_id, account_id, amount) VALUES ($1, $2, $3) ON CONFLICT (contract_id, account_id) DO UPDATE SET amount=EXCLUDED.amount`, contractID, accountID, encode(fundingValue))
if err != nil {
return fmt.Errorf("failed to update funding source: %w", err)
}
return nil
}

func accountBalance(tx *txn, accountID rhp3.Account) (dbID int64, balance types.Currency, err error) {
err = tx.QueryRow(`SELECT id, balance FROM accounts WHERE account_id=$1`, encode(accountID)).Scan(&dbID, decode(&balance))
return
Expand All @@ -192,6 +308,26 @@ type fundAmount struct {
Amount types.Currency
}

// contractV2Funding returns all contracts that were used to fund the account.
func contractV2Funding(tx *txn, accountID int64) (fund []fundAmount, err error) {
rows, err := tx.Query(`SELECT id, contract_id, amount FROM contract_v2_account_funding WHERE account_id=$1`, accountID)
if err != nil {
return nil, err
}
defer rows.Close()

for rows.Next() {
var f fundAmount
if err := rows.Scan(&f.ID, &f.ContractID, decode(&f.Amount)); err != nil {
return nil, fmt.Errorf("failed to scan row: %w", err)
} else if f.Amount.IsZero() {
continue
}
fund = append(fund, f)
}
return
}

// contractFunding returns all contracts that were used to fund the account.
func contractFunding(tx *txn, accountID int64) (fund []fundAmount, err error) {
rows, err := tx.Query(`SELECT id, contract_id, amount FROM contract_account_funding WHERE account_id=$1`, accountID)
Expand All @@ -212,6 +348,65 @@ func contractFunding(tx *txn, accountID int64) (fund []fundAmount, err error) {
return
}

// updateV2ContractFunding distributes account usage to the contracts that funded
// the account.
func updateV2ContractFunding(tx *txn, accountID int64, usage proto4.Usage) error {
funding, err := contractV2Funding(tx, accountID)
if err != nil {
return fmt.Errorf("failed to get contract funding: %w", err)
}

distributeFunds := func(usage, additional, remainder *types.Currency) {
if remainder.IsZero() || usage.IsZero() {
return
}

v := *usage
if usage.Cmp(*remainder) > 0 {
v = *remainder
}
*usage = usage.Sub(v)
*remainder = remainder.Sub(v)
*additional = additional.Add(v)
}

// distribute account usage to the funding contracts
for _, f := range funding {
remainder := f.Amount

var additionalUsage proto4.Usage
distributeFunds(&usage.Storage, &additionalUsage.Storage, &remainder)
distributeFunds(&usage.Ingress, &additionalUsage.Ingress, &remainder)
distributeFunds(&usage.Egress, &additionalUsage.Egress, &remainder)
distributeFunds(&usage.RPC, &additionalUsage.RPC, &remainder)

if remainder.IsZero() {
if _, err := tx.Exec(`DELETE FROM contract_v2_account_funding WHERE id=$1`, f.ID); err != nil {
return fmt.Errorf("failed to delete account funding: %w", err)
}
} else {
_, err := tx.Exec(`UPDATE contract_v2_account_funding SET amount=$1 WHERE id=$2`, encode(remainder), f.ID)
if err != nil {
return fmt.Errorf("failed to update account funding: %w", err)
}
}

var contractExistingFunding types.Currency
if err := tx.QueryRow(`SELECT account_funding FROM contracts_v2 WHERE id=$1`, f.ContractID).Scan(decode(&contractExistingFunding)); err != nil {
return fmt.Errorf("failed to get contract usage: %w", err)
}
contractExistingFunding = contractExistingFunding.Sub(f.Amount.Sub(remainder))
if _, err := tx.Exec(`UPDATE contracts_v2 SET account_funding=$1 WHERE id=$2`, encode(contractExistingFunding), f.ContractID); err != nil {
return fmt.Errorf("failed to update contract account funding: %w", err)
}

if err := updateV2ContractUsage(tx, f.ContractID, additionalUsage); err != nil {
return fmt.Errorf("failed to update contract usage: %w", err)
}
}
return nil
}

// updateContractUsage distributes account usage to the contracts that funded
// the account.
func updateContractUsage(tx *txn, accountID int64, usage accounts.Usage, log *zap.Logger) error {
Expand Down Expand Up @@ -265,7 +460,7 @@ func updateContractUsage(tx *txn, accountID int64, usage accounts.Usage, log *za
return fmt.Errorf("failed to decrement account funding: %w", err)
}

if contract.Status == contracts.ContractStatusActive || contract.Status == contracts.ContractStatusPending {
if contract.Status == contracts.ContractStatusActive {
// increment potential revenue
if err := incrementPotentialRevenueMetrics(tx, additionalUsage, false); err != nil {
return fmt.Errorf("failed to increment contract potential revenue: %w", err)
Expand Down
Loading

0 comments on commit 8e67014

Please sign in to comment.