Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update estimateBalancedTxBody to account for required deposits #527

Merged
merged 1 commit into from
May 3, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
102 changes: 68 additions & 34 deletions cardano-api/internal/Cardano/Api/Fees.hs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ import Cardano.Api.Eon.ShelleyBasedEra
import Cardano.Api.Eras.Case
import Cardano.Api.Eras.Core
import Cardano.Api.Error
import Cardano.Api.Feature
import qualified Cardano.Api.Ledger.Lens as A
import Cardano.Api.Pretty
import Cardano.Api.ProtocolParameters
Expand All @@ -73,6 +74,8 @@ import qualified Cardano.Ledger.Alonzo.Plutus.Context as Plutus
import qualified Cardano.Ledger.Alonzo.Scripts as Alonzo
import qualified Cardano.Ledger.Api as L
import qualified Cardano.Ledger.Coin as L
import qualified Cardano.Ledger.Conway.Governance as L
import qualified Cardano.Ledger.Core as L
import Cardano.Ledger.Credential as Ledger (Credential)
import qualified Cardano.Ledger.Crypto as Ledger
import qualified Cardano.Ledger.Keys as Ledger
Expand All @@ -83,11 +86,13 @@ import qualified PlutusLedgerApi.V1 as Plutus
import Control.Monad (forM_)
import Data.Bifunctor (bimap, first)
import Data.ByteString.Short (ShortByteString)
import Data.Foldable (toList)
import Data.Function ((&))
import qualified Data.List as List
import Data.Map.Strict (Map)
import qualified Data.Map.Strict as Map
import Data.Maybe (catMaybes, fromMaybe, maybeToList)
import qualified Data.OSet.Strict as OSet
import Data.Ratio
import Data.Set (Set)
import qualified Data.Set as Set
Expand All @@ -96,14 +101,14 @@ import Lens.Micro ((.~), (^.))

{- HLINT ignore "Redundant return" -}

data AutoBalanceError era
data AutoBalanceError era
= AutoBalanceEstimationError (TxFeeEstimationError era)
| AutoBalanceCalculationError (TxBodyErrorAutoBalance era)
deriving Show

instance Error (AutoBalanceError era) where
instance Error (AutoBalanceError era) where
prettyError = \case
AutoBalanceEstimationError e -> prettyError e
AutoBalanceEstimationError e -> prettyError e
AutoBalanceCalculationError e -> prettyError e


Expand All @@ -117,27 +122,27 @@ estimateOrCalculateBalancedTxBody
-> Map (Ledger.Credential Ledger.DRepRole Ledger.StandardCrypto) L.Coin
-> AddressInEra era
-> Either (AutoBalanceError era) (BalancedTxBody era)
estimateOrCalculateBalancedTxBody era feeEstMode pparams txBodyContent poolids stakeDelegDeposits drepDelegDeposits changeAddr =
case feeEstMode of
CalculateWithSpendableUTxO utxo systemstart ledgerEpochInfo mOverride ->
first AutoBalanceCalculationError $
estimateOrCalculateBalancedTxBody era feeEstMode pparams txBodyContent poolids stakeDelegDeposits drepDelegDeposits changeAddr =
case feeEstMode of
CalculateWithSpendableUTxO utxo systemstart ledgerEpochInfo mOverride ->
first AutoBalanceCalculationError $
makeTransactionBodyAutoBalance era systemstart ledgerEpochInfo (LedgerProtocolParameters pparams)
poolids stakeDelegDeposits drepDelegDeposits utxo txBodyContent changeAddr mOverride

EstimateWithoutSpendableUTxO
totalPotentialCollateral totalUTxOValue exUnitsMap
(RequiredShelleyKeyWitnesses numKeyWits) (RequiredByronKeyWitnesses numByronWits)
(TotalReferenceScriptsSize totalRefScriptsSize) ->
EstimateWithoutSpendableUTxO
totalPotentialCollateral totalUTxOValue exUnitsMap
(RequiredShelleyKeyWitnesses numKeyWits) (RequiredByronKeyWitnesses numByronWits)
(TotalReferenceScriptsSize totalRefScriptsSize) ->
forShelleyBasedEraInEon
era
era
(Left $ AutoBalanceEstimationError TxFeeEstimationOnlyMaryOnwardsSupportedError)
(\w -> first AutoBalanceEstimationError $
estimateBalancedTxBody w txBodyContent pparams poolids stakeDelegDeposits drepDelegDeposits
exUnitsMap totalPotentialCollateral numKeyWits numByronWits
(\w -> first AutoBalanceEstimationError $
estimateBalancedTxBody w txBodyContent pparams poolids stakeDelegDeposits drepDelegDeposits
exUnitsMap totalPotentialCollateral numKeyWits numByronWits
totalRefScriptsSize changeAddr totalUTxOValue
)


data TxFeeEstimationError era
= TxFeeEstimationTransactionTranslationError (TransactionValidityError era)
| TxFeeEstimationScriptExecutionError (TxBodyErrorAutoBalance era)
Expand All @@ -149,19 +154,19 @@ data TxFeeEstimationError era



instance Error (TxFeeEstimationError era) where
instance Error (TxFeeEstimationError era) where
prettyError = \case
TxFeeEstimationTransactionTranslationError e -> prettyError e
TxFeeEstimationScriptExecutionError e -> prettyError e
TxFeeEstimationBalanceError e -> prettyError e
TxFeeEstimationxBodyError e -> prettyError e
TxFeeEstimationFinalConstructionError e -> prettyError e
TxFeeEstimationOnlyMaryOnwardsSupportedError->
"Only mary era onwards supported."
"Only mary era onwards supported."

-- | Use when you do not have access to the UTxOs you intend to spend
estimateBalancedTxBody
:: MaryEraOnwards era
:: forall era. MaryEraOnwards era
-> TxBodyContent BuildTx era
-> L.PParams (ShelleyLedgerEra era)
-> Set PoolId -- ^ The set of registered stake pools, that are being
Expand Down Expand Up @@ -192,11 +197,40 @@ estimateBalancedTxBody w txbodycontent pparams poolids

-- Step 2. We need to calculate the current balance of the tx. The user
-- must at least provide the total value of the UTxOs they intend to spend
-- for us to calulate the balance.
let change = toLedgerValue w $ calculateChangeValue sbe totalUTxOValue txbodycontent1
let maxLovelaceChange = L.Coin (2^(64 :: Integer)) - 1
let changeWithMaxLovelace = change & A.adaAssetL sbe .~ maxLovelaceChange
let changeTxOut = forShelleyBasedEraInEon sbe
-- for us to calulate the balance. NB: We must:
-- 1. Subtract certificate and proposal deposits
-- from the total available Ada value!
-- Page 24 Shelley ledger spec
let certificates =
case txCertificates txbodycontent1 of
TxCertificatesNone -> []
TxCertificates _ certs _ -> map toShelleyCertificate certs

proposalProcedures :: OSet.OSet (L.ProposalProcedure (ShelleyLedgerEra era))
proposalProcedures =
case unFeatured <$> txProposalProcedures txbodycontent1 of
Nothing -> OSet.empty
Just TxProposalProceduresNone -> OSet.empty
Just (TxProposalProcedures procedures _) -> procedures

totalDeposits :: L.Coin
totalDeposits =
-- Because we do not have access to the ledger state and to reduce the complexity of this function's
-- type signature, we assume the user is trying to register a stake pool that has not been
-- registered before and has not included duplicate stake pool registration certificates.
let assumeStakePoolHasNotBeenRegistered = const False
in sum [ maryEraOnwardsConstraints w $ L.getTotalDepositsTxCerts pparams assumeStakePoolHasNotBeenRegistered certificates
, mconcat $ map (^. L.pProcDepositL) $ toList proposalProcedures
carbolymer marked this conversation as resolved.
Show resolved Hide resolved
]

availableUTxOValue = mconcat [ totalUTxOValue
, negateValue (lovelaceToValue totalDeposits)
]

let change = toLedgerValue w $ calculateChangeValue sbe availableUTxOValue txbodycontent1
maxLovelaceChange = L.Coin (2^(64 :: Integer)) - 1
changeWithMaxLovelace = change & A.adaAssetL sbe .~ maxLovelaceChange
changeTxOut = forShelleyBasedEraInEon sbe
(lovelaceToTxOutValue sbe maxLovelaceChange)
(\w' -> maryEraOnwardsConstraints w' $ TxOutValueShelleyBased sbe changeWithMaxLovelace)

Expand Down Expand Up @@ -237,7 +271,7 @@ estimateBalancedTxBody w txbodycontent pparams poolids
txTotalCollateral = reqCol
}

let fakeUTxO = createFakeUTxO sbe txbodycontent1 $ selectLovelace totalUTxOValue
let fakeUTxO = createFakeUTxO sbe txbodycontent1 $ selectLovelace availableUTxOValue
balance = evaluateTransactionBalance sbe pparams poolids stakeDelegDeposits drepDelegDeposits fakeUTxO txbody2
-- check if the balance is positive or negative
-- in one case we can produce change, in the other the inputs are insufficient
Expand Down Expand Up @@ -862,15 +896,15 @@ data BalancedTxBody era
(TxOut CtxTx era) -- ^ Transaction balance (change output)
L.Coin -- ^ Estimated transaction fee

newtype RequiredShelleyKeyWitnesses
newtype RequiredShelleyKeyWitnesses
= RequiredShelleyKeyWitnesses { unRequiredShelleyKeyWitnesses :: Int }
deriving Show

newtype RequiredByronKeyWitnesses
newtype RequiredByronKeyWitnesses
= RequiredByronKeyWitnesses { unRequiredByronKeyWitnesses :: Int }
deriving Show

newtype TotalReferenceScriptsSize
newtype TotalReferenceScriptsSize
= TotalReferenceScriptsSize { unTotalReferenceScriptsSize :: Int }
deriving Show

Expand Down Expand Up @@ -1186,11 +1220,11 @@ calculateChangeValue sbe incoming txbodycontent =
-- | This is used in the balance calculation in the event where
-- the user does not supply the UTxO(s) they intend to spend
-- but they must supply their total balance of ADA.
-- evaluateTransactionBalance calls evalBalanceTxBody which requires a UTxO value.
-- This eventually calls getConsumedMaryValue which retrieves the balance
-- from the transaction itself. This necessitated a function to create a "fake" UTxO
-- to still use evalBalanceTxBody however this will fail for transactions
-- containing multi-assets, refunds and withdrawals.
-- evaluateTransactionBalance calls evalBalanceTxBody which requires a UTxO value.
-- This eventually calls getConsumedMaryValue which retrieves the balance
-- from the transaction itself. This necessitated a function to create a "fake" UTxO
-- to still use evalBalanceTxBody however this will fail for transactions
-- containing multi-assets, refunds and withdrawals.
-- TODO: Include multiassets
createFakeUTxO :: ShelleyBasedEra era -> TxBodyContent BuildTx era -> Coin -> UTxO era
createFakeUTxO sbe txbodycontent totalAdaInUTxO =
Expand All @@ -1207,7 +1241,7 @@ updateTxOut sbe updatedValue txout =

-- Essentially we check for the existence of collateral inputs. If they exist we
-- create a fictitious collateral return output. Why? Because we need to put dummy values
-- to get a fee estimate (i.e we overestimate the fee). The required collateral depends
-- to get a fee estimate (i.e we overestimate the fee). The required collateral depends
-- on the tx fee as per the Alonzo spec.
maybeDummyTotalCollAndCollReturnOutput
:: ShelleyBasedEra era
Expand Down
Loading