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

Sam/expose budget multiplier #119

Merged
merged 8 commits into from
Jun 15, 2022
Merged
Show file tree
Hide file tree
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
2 changes: 2 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,8 @@ Top-level configuration file fields:
collectStats: `true` or `false`
Save some stats during contract run (only transactions execution
budgets supported atm) (default: false)
budgetMultiplier: rational multiplier in form `1` or `1 % 2`
(default: 1)
```

To run the fake PAB, you need to prepare a few more things:
Expand Down
10 changes: 10 additions & 0 deletions src/BotPlutusInterface/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ import Data.String.ToString (toString)
import Data.Text (Text)
import Data.Text qualified as Text
import PlutusConfig.Base (
customRationalSpec,
enumToAtom,
filepathSpec,
maybeSpec,
Expand Down Expand Up @@ -95,6 +96,7 @@ instance ToValue PABConfig where
pcPort
pcEnableTxEndpoint
pcCollectStats
pcBudgetMultiplier
) =
Sections
()
Expand All @@ -116,6 +118,7 @@ instance ToValue PABConfig where
, Section () "port" $ toValue pcPort
, Section () "enableTxEndpoint" $ toValue pcEnableTxEndpoint
, Section () "collectStats" $ toValue pcCollectStats
, Section () "budgetMultiplier" $ toValue pcBudgetMultiplier
]
{- ORMOLU_ENABLE -}

Expand Down Expand Up @@ -206,6 +209,13 @@ pabConfigSpec = sectionsSpec "PABConfig" $ do
trueOrFalseSpec
"Save some stats during contract run (only transactions execution budgets supported atm)"

pcBudgetMultiplier <-
sectionWithDefault'
(pcBudgetMultiplier def)
"budgetMultiplier"
customRationalSpec
"Multiplier on the budgets automatically calculated"

pure PABConfig {..}

docPABConfig :: String
Expand Down
50 changes: 44 additions & 6 deletions src/BotPlutusInterface/ExBudget.hs
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,15 @@ import BotPlutusInterface.Types (
TxFile (..),
)
import Cardano.Api qualified as CAPI
import Cardano.Api.Shelley (ProtocolParameters (protocolParamMaxTxExUnits))
import Cardano.Prelude (maybeToEither)
import Control.Arrow (left)
import Data.Either (rights)
import Data.List (sort)
import Data.Map (Map)
import Data.Map qualified as Map
import Data.Text qualified as Text
import GHC.Natural (Natural)
import Ledger (ExBudget (ExBudget), ExCPU (ExCPU), ExMemory (ExMemory), MintingPolicyHash, TxOutRef)
import Ledger.Tx.CardanoAPI (fromCardanoPolicyId, fromCardanoTxIn)
import System.Directory.Internal.Prelude (getEnv)
Expand All @@ -43,13 +47,47 @@ estimateBudget pabConf txFile = do
(getExUnits debugNodeInf)
txBody

let txBudget = do
body <- txBody
budget <- budgetRes
(spendingBudgets, policyBudgets) <- mkBudgetMaps budget body
Right $ TxBudget spendingBudgets policyBudgets
return $
do
body <- txBody
budget <- budgetRes
maxUnits <- maybeToEither (BudgetEstimationError "Missing max units in parameters") $ protocolParamMaxTxExUnits pabConf.pcProtocolParams

scaledBudget <- getScaledBudget maxUnits pabConf.pcBudgetMultiplier budget

(spendingBudgets, policyBudgets) <- mkBudgetMaps scaledBudget body

Right $ TxBudget spendingBudgets policyBudgets

-- | Scale the budget clamping the total to the parameter limits
getScaledBudget :: CAPI.ExecutionUnits -> Rational -> ExUnitsMap -> Either BudgetEstimationError ExUnitsMap
getScaledBudget maxUnits scaler budget =
if fst scalers >= 1 && snd scalers >= 1
then Right $ fmap (fmap $ scaleBudget scalers) budget
else
Left $
BudgetEstimationError $
Text.pack $
"Exceeded global transaction budget\nCalculated: " ++ show budgetSum ++ "\nLimit: " ++ show maxUnits
where
budgetSum = foldr addBudgets (CAPI.ExecutionUnits 0 0) $ rights $ Map.elems budget
scalers =
( clampedScaler (CAPI.executionSteps budgetSum) (CAPI.executionSteps maxUnits) scaler
, clampedScaler (CAPI.executionMemory budgetSum) (CAPI.executionMemory maxUnits) scaler
)

clampedScaler :: Natural -> Natural -> Rational -> Rational
clampedScaler 0 _ scaler = scaler
clampedScaler val maxVal scaler = min scaler (toRational maxVal / toRational val)

-- | Scale the budget by the multipliers in config
scaleBudget :: (Rational, Rational) -> CAPI.ExecutionUnits -> CAPI.ExecutionUnits
scaleBudget (stepsScaler, memScaler) (CAPI.ExecutionUnits steps mem) = CAPI.ExecutionUnits (scale steps stepsScaler) (scale mem memScaler)
where
scale x scaler = round $ toRational x * scaler

return txBudget
addBudgets :: CAPI.ExecutionUnits -> CAPI.ExecutionUnits -> CAPI.ExecutionUnits
addBudgets (CAPI.ExecutionUnits steps mem) (CAPI.ExecutionUnits steps' mem') = CAPI.ExecutionUnits (steps + steps') (mem + mem')

-- | Deserialize transaction body from ".signed" file
deserialiseSigned :: FilePath -> IO (Either BudgetEstimationError (CAPI.Tx CAPI.AlonzoEra))
Expand Down
1 change: 1 addition & 0 deletions src/BotPlutusInterface/QueryNode.hs
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
-- | Several query functions to query local node
module BotPlutusInterface.QueryNode (
NodeInfo (..),
NodeQueryError (..),
queryProtocolParams,
querySystemStart,
queryEraHistory,
Expand Down
2 changes: 2 additions & 0 deletions src/BotPlutusInterface/Types.hs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ data PABConfig = PABConfig
, pcPort :: !Port
, pcEnableTxEndpoint :: !Bool
, pcCollectStats :: !Bool
, pcBudgetMultiplier :: !Rational
}
deriving stock (Show, Eq)

Expand Down Expand Up @@ -221,6 +222,7 @@ instance Default PABConfig where
, pcPort = 9080
, pcEnableTxEndpoint = False
, pcCollectStats = False
, pcBudgetMultiplier = 1
}

data RawTx = RawTx
Expand Down
1 change: 1 addition & 0 deletions test/Spec/BotPlutusInterface/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -113,4 +113,5 @@ pabConfigExample =
, pcPort = 1021
, pcEnableTxEndpoint = True
, pcCollectStats = False
, pcBudgetMultiplier = 1
}