Skip to content

A repository with some information I may need to refer later about the Plutus lectures

Notifications You must be signed in to change notification settings

mateusap1/plutus-reference

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

35 Commits
 
 

Repository files navigation

plutus-reference

A repository with some information I may need to refer later about the plutus lectures

Note: This is still being written and, because of this, a lot of information is still missing and some of it may even be false, so make sure to do your own research when needed and do not rely on this repository for handling critical data.

adaSymbol

The 'CurrencySymbol' of the 'Ada' currency.

adaSymbol :: CurrencySymbol
adaSymbol = TH.currencySymbol emptyByteString

A CurrencySymbol is the hash of a script that will be executed to validate minting and burning. Because ADA is deflationary, it uses an "empty script", which means no one is able to mint nor burn any ADA. That's why adaSymbol is an emptyByteString.

adaToken

The 'TokenName' of the 'Ada' currency.

adaToken :: TokenName
adaToken = TH.tokenName emptyByteString

A TokenName is a string that represents a token (pretty self explanatory). So, for instance, if Bitcoin was a Plutus token, it's token name would probably be "btc". In this spirit, adaToken is simply Ada's token name, but because Ada is the Cardano's "default token", it doesn't receive an "ada" token name, but instead an emptyByteString.

Ada

ADA, the special currency on the Cardano blockchain. The unit of Ada is Lovelace, and 1M Lovelace is one Ada. See note [Currencies] in 'Ledger.Validation.Value.TH'.

newtype Ada = Lovelace { getLovelace :: Integer }
    deriving (Haskell.Enum)
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (ToJSON, FromJSON)
    deriving newtype (Eq, Ord, Haskell.Num, AdditiveSemigroup, AdditiveMonoid, AdditiveGroup, MultiplicativeSemigroup, MultiplicativeMonoid, Haskell.Integral, Haskell.Real, Serialise, PlutusTx.IsData)
    deriving Pretty via (Tagged "Lovelace:" Integer)

instance Haskell.Semigroup Ada where
    Lovelace a1 <> Lovelace a2 = Lovelace (a1 + a2)

instance Semigroup Ada where
    Lovelace a1 <> Lovelace a2 = Lovelace (a1 + a2)

instance Haskell.Monoid Ada where
    mempty = Lovelace 0

instance Monoid Ada where
    mempty = Lovelace 0

makeLift ''Ada

getAda

Get the amount of Ada (the unit of the currency Ada) in this 'Ada' value.

getAda :: Ada -> Micro
getAda (Lovelace i) = MkFixed i

Micro is a representation of µ, which is just another way of saying 10^-6. So, what getAda is doing is getting a Lovelace amount and wrapping it in a type that is a million times smaller than the value it holds.

toValue

Create a 'Value' containing only the given 'Ada'.

toValue :: Ada -> Value
toValue (Lovelace i) = TH.singleton adaSymbol adaToken i

Based on an Ada (an amount basically), create a Value, which contains the token symbol, name and amount (in Ada).

fromValue

Get the 'Ada' in the given 'Value'.

fromValue :: Value -> Ada
fromValue v = Lovelace (TH.valueOf v adaSymbol adaToken)

Ignores everything except the amount from a Value. So, for instance, one might want to verify if someone has sufficient funds. For that, he might not need to know the token's name or symbol, so he should use fromValue to extract the Ada amount only.

lovelaceOf

Create 'Ada' representing the given quantity of Lovelace (the unit of the currency Ada).

lovelaceOf :: Integer -> Ada
lovelaceOf = Lovelace

adaOf

Create 'Ada' representing the given quantity of Ada (1M Lovelace).

adaOf :: Micro -> Ada
adaOf (MkFixed x) = Lovelace x

lovelaceValueOf

A 'Value' with the given amount of Lovelace (the currency unit).

@lovelaceValueOf == toValue . lovelaceOf@

lovelaceValueOf :: Integer -> Value
lovelaceValueOf = TH.singleton adaSymbol adaToken

adaValueOf

A 'Value' with the given amount of Ada (the currency unit).

@adaValueOf == toValue . adaOf@

adaValueOf :: Micro -> Value
adaValueOf (MkFixed x) = TH.singleton adaSymbol adaToken x

divide

Divide one 'Ada' value by another.

divide :: Ada -> Ada -> Ada
divide (Lovelace a) (Lovelace b) = Lovelace (P.divide a b)

isZero

Check whether an 'Ada' value is zero.

isZero :: Ada -> Bool
isZero (Lovelace i) = i == 0

Address

Address with two kinds of credentials, normal and staking

data Address = Address{ addressCredential :: Credential, addressStakingCredential :: (Maybe StakingCredential)}
    deriving stock (Eq, Ord, Show, Generic)
    deriving anyclass (ToJSON, FromJSON, ToJSONKey, FromJSONKey, Serialise, Hashable, NFData)

instance Pretty Address where
    pretty (Address cred stakingCred) =
        let staking = maybe "no staking credential" pretty stakingCred in
        "addressed to" <+> pretty cred <+> parens staking


instance PlutusTx.Eq Address where
    Address cred stakingCred == Address cred' stakingCred' =
        cred PlutusTx.== cred'
        PlutusTx.&& stakingCred PlutusTx.== stakingCred'

Address is a data type that holds a Credential and a StakingCredential, The Credential is a data type that either holds a user's public key hash or a script validator hash. The StakingCredential, which can be Nothing, holds either a StakingHash or a StakingPtr and is used to assign rewards.

To be honest, I don't quite understand what a staking credential actually is, so fell free to submit a Pull Request giving a better explanation.

pubKeyAddress

The address that should be targeted by a transaction output locked by the given public key.

pubKeyAddress :: PubKey -> Address
pubKeyAddress pk = Address (PubKeyCredential (pubKeyHash pk)) Nothing

Takes the user's PubKey, hash it, makes a credential out of it and gives it as an Address constructor input.

pubKeyHashAddress

The address that should be targeted by a transaction output locked by the public key with the given hash.

pubKeyHashAddress :: PubKeyHash -> Address
pubKeyHashAddress pkh = Address (PubKeyCredential pkh) Nothing

The only difference from pubKeyAddress is that it already receives a hashed public key so it doesn't need to hash it again.

toPubKeyHash

The PubKeyHash of the address, if any

toPubKeyHash :: Address -> Maybe PubKeyHash
toPubKeyHash (Address (PubKeyCredential k) _) = Just k
toPubKeyHash _                                = Nothing

Takes an Address, verify if it's credential is a public key one and, if so, returns it. If, in the other hand, is a script validator, returns Nothing.

toValidatorHash

The validator hash of the address, if any

toValidatorHash :: Address -> Maybe ValidatorHash
toValidatorHash (Address (ScriptCredential k) _) = Just k
toValidatorHash _   

Same thing as toPubKeyHash, but instead of making sure that the address credential is a public key, it want's a script validator credential.

scriptAddress

The address that should be used by a transaction output locked by the given validator script.

scriptAddress :: Validator -> Address
scriptAddress validator = Address (ScriptCredential (validatorHash validator)) Nothing 

Same thing as pubKeyAddress, but with a script validator instead of a public key. Takes the user's Validator, hash it, makes a credential out of it and gives it as an Address constructor input.

scriptHashAddress

he address that should be used by a transaction output locked by the given validator script hash.

scriptHashAddress :: ValidatorHash -> Address
scriptHashAddress vh = Address (ScriptCredential vh) Nothing

Takes the user's ValidatorHash, makes a credential out of it and gives it as an Address constructor input.

stakingCredential

The staking credential of an address (if any)

stakingCredential :: Address -> Maybe StakingCredential
stakingCredential (Address _ s) = s

Returns a StakingCredential if the given address has one

Nothing here, feel free to contribute!

Nothing here, feel free to contribute!

TxInInfo

An input of a pending transaction.

data TxInInfo = TxInInfo
    { txInInfoOutRef   :: TxOutRef
    , txInInfoResolved :: TxOut
    } deriving (Generic)

ScriptPurpose

Purpose of the script that is currently running

data ScriptPurpose
    = Minting CurrencySymbol
    | Spending TxOutRef
    | Rewarding StakingCredential
    | Certifying DCert

The code definition says it all, it's a data type that defines the purpose of the current script. If a script purpose is Miniting, then it will need to store the CurrencySymbol, which is the hash of the contract that contains the minting policy.

If, alternativily, the script purpose is Spending, then it should store the TxOutRef, that is: the reference to the ouput of the transaction that will be "spending".

If, in the other hand, it's purpouse is Rewarding, then it should store the StakingCredential, which is a credential containing the necessary information to assign rewards. The last one, I have no idea what is used for.

TxInfo

A pending transaction. This is the view as seen by validator scripts, so some details are stripped out.

data TxInfo = TxInfo
    { txInfoInputs      :: [TxInInfo] -- ^ Transaction inputs
    , txInfoOutputs     :: [TxOut] -- ^ Transaction outputs
    , txInfoFee         :: Value -- ^ The fee paid by this transaction.
    -- TODO: rename to txInfoMint, but this requires changes in cardano-ledger-specs
    , txInfoForge       :: Value -- ^ The 'Value' minted by this transaction.
    , txInfoDCert       :: [DCert] -- ^ Digests of certificates included in this transaction
    , txInfoWdrl        :: [(StakingCredential, Integer)] -- ^ Withdrawals
    , txInfoValidRange  :: POSIXTimeRange -- ^ The valid range for the transaction.
    , txInfoSignatories :: [PubKeyHash] -- ^ Signatures provided with the transaction, attested that they all signed the tx
    , txInfoData        :: [(DatumHash, Datum)]
    , txInfoId          :: TxId
    -- ^ Hash of the pending transaction (excluding witnesses)
    } deriving (Generic)

When validating a transaction inside a contract, it is important to know some information about this transaction, this is what TxInfo is used for. In order to make sure, for example, that a token is being minted by someone with permission, the script could verify if any PubKeyHash of txInfoSignatories is inside the list of allowed users.

txInfoInputs is a list of inputs (TxInInfo), which are basically previous UTxOs (or previous outputs to be more technically correct) that were spent to generate new ones. txInfoOutputs is a list of newly created outputs (TxOut).

ScriptContext

data ScriptContext = ScriptContext{scriptContextTxInfo :: TxInfo, scriptContextPurpose :: ScriptPurpose }

ScriptContext stores all the information about a possible transaction, as well as, what it is used for. Both of this actions are handled by TxInfo and ScriptPurpose, so it is simply a combination of both of these data types.

findOwnInput

Find the input currently being validated.

findOwnInput :: ScriptContext -> Maybe TxInInfo
findOwnInput ScriptContext{scriptContextTxInfo=TxInfo{txInfoInputs}, scriptContextPurpose=Spending txOutRef} =
    find (\TxInInfo{txInInfoOutRef} -> txInInfoOutRef == txOutRef) txInfoInputs
findOwnInput _ = Nothing

The first thing findOwnInput does is to limit the number of valid paramaters (only contexts with a spending purpose, for example, will pass). By doing that we get two important variables: txInfoInputs and txOutRef, the former is a list containing all inputs in the context of this script, the latter is a reference to the user's previous UTxO output (our current input). What the function does is to go over each input, extract it's TxInInfoOutRef and, if it is equal to our txOutRef, return it, otherwise return Nothing.

findDatum

Find the data corresponding to a data hash, if there is one

findDatum :: DatumHash -> TxInfo -> Maybe Datum
findDatum dsh TxInfo{txInfoData} = snd <$> find f txInfoData
    where
        f (dsh', _) = dsh' == dsh

Given a DatumHash and a TxInfo, search for a Datum with the corresponding hash inside txInfoData (an element of TxInfo). This is important because TxOuts don't store the datum it self, but it's hash, so in order to extract the data it self from a TxOut we need to use this function.

findDatumHash

Find the hash of a datum, if it is part of the pending transaction's hashes

findDatumHash :: Datum -> TxInfo -> Maybe DatumHash
findDatumHash ds TxInfo{txInfoData} = fst <$> find f txInfoData
    where
        f (_, ds') = ds' == ds

Does the same thing as findDatum, but instead of searching for a Datum given a DatumHash, searchs for a DatumHash given a Datum.

findTxInByTxOutRef

findTxInByTxOutRef :: TxOutRef -> TxInfo -> Maybe TxInInfo
findTxInByTxOutRef outRef TxInfo{txInfoInputs} =
    find (\TxInInfo{txInInfoOutRef} -> txInInfoOutRef == outRef) txInfoInputs

Given a reference to an output (TxOutRef) and information about a transaction (TxInfo), findTxInByTxOutRef searches for an input whose reference to it's previous UTxO output is the same as the given output reference.

findContinuingOutputs

Finds all the outputs that pay to the same script address that we are currently spending from, if any.

findContinuingOutputs :: ScriptContext -> [Integer]
findContinuingOutputs ctx | Just TxInInfo{txInInfoResolved=TxOut{txOutAddress}} <- findOwnInput ctx = findIndices (f txOutAddress) (txInfoOutputs $ scriptContextTxInfo ctx)
    where
        f addr TxOut{txOutAddress=otherAddress} = addr == otherAddress
findContinuingOutputs _ = Builtins.error()

getContinuingOutputs

getContinuingOutputs :: ScriptContext -> [TxOut]
getContinuingOutputs ctx | Just TxInInfo{txInInfoResolved=TxOut{txOutAddress}} <- findOwnInput ctx = filter (f txOutAddress) (txInfoOutputs $ scriptContextTxInfo ctx)
    where
        f addr TxOut{txOutAddress=otherAddress} = addr == otherAddress
getContinuingOutputs _ = Builtins.error()

scriptCurrencySymbol

The 'CurrencySymbol' of a 'MintingPolicy'

scriptCurrencySymbol :: MintingPolicy -> CurrencySymbol
scriptCurrencySymbol scrpt = let (MintingPolicyHash hsh) = mintingPolicyHash scrpt in Value.currencySymbol hsh

A scriptCurrencySymbol is the hash of the script that validates the minting policy. So that's what this function does, it gets a MintingPolicy script, hashes it and returns it in a Value.currencySymbol form.

txSignedBy

Check if a transaction was signed by the given public key.

txSignedBy :: TxInfo -> PubKeyHash -> Bool
txSignedBy TxInfo{txInfoSignatories} k = case find ((==) k) txInfoSignatories of
    Just _  -> True
    Nothing -> False

Goes over each element of the txInfoSignatories list (a component of the TxInfo we received) and verifies if the signature is equal to the public key hash we want to compare (k).

pubKeyOutput

Get the public key hash that locks the transaction output, if any.

pubKeyOutput :: TxOut -> Maybe PubKeyHash
pubKeyOutput TxOut{txOutAddress} = toPubKeyHash txOutAddress

Every TxOut contains an address and every Address has a Credential that can be of type PubKeyCredential or ScriptCredential. If the received output has an address with a public key credential, the function returns it's hash, otherwise it returns Nothing.

ownHashes

Get the validator and datum hashes of the output that is curently being validated

ownHashes :: ScriptContext -> (ValidatorHash, DatumHash)
ownHashes (findOwnInput -> Just TxInInfo{txInInfoResolved=TxOut{txOutAddress=Address (ScriptCredential s) _, txOutDatumHash=Just dh}}) = (s,dh)
ownHashes _                                                                                                                            = Builtins.error ()

ownHash

Get the hash of the validator script that is currently being validated.

ownHash :: ScriptContext -> ValidatorHash
ownHash p = fst (ownHashes p)

fromSymbol

Convert a 'CurrencySymbol' to a 'ValidatorHash'

fromSymbol :: CurrencySymbol -> ValidatorHash
fromSymbol (CurrencySymbol s) = ValidatorHash s

scriptOutputsAt

Get the list of 'TxOut' outputs of the pending transaction at a given script address.

scriptOutputsAt :: ValidatorHash -> TxInfo -> [(DatumHash, Value)]
scriptOutputsAt h p =
    let flt TxOut{txOutDatumHash=Just ds, txOutAddress=Address (ScriptCredential s) _, txOutValue} | s == h = Just (ds, txOutValue)
        flt _ = Nothing
    in mapMaybe flt (txInfoOutputs p)

Return in a (DatumHash,Value) format every pending transactions output that has a matching ValidatorHash compared to the one received (h). In other words, returns a list with every transaction inside a script that has this user's public key as it's output.

valueLockedBy

Get the total value locked by the given validator in this transaction.

valueLockedBy :: TxInfo -> ValidatorHash -> Value
valueLockedBy ptx h =
    let outputs = map snd (scriptOutputsAt h ptx)
    in mconcat outputs

pubKeyOutputsAt

Get the values paid to a public key address by a pending transaction.

pubKeyOutputsAt :: PubKeyHash -> TxInfo -> [Value]
pubKeyOutputsAt pk p =
    let flt TxOut{txOutAddress = Address (PubKeyCredential pk') _, txOutValue} | pk == pk' = Just txOutValue
        flt _                             = Nothing
    in mapMaybe flt (txInfoOutputs p)

valuePaidTo

Get the total value paid to a public key address by a pending transaction.

valuePaidTo :: TxInfo -> PubKeyHash -> Value
valuePaidTo ptx pkh = mconcat (pubKeyOutputsAt pkh ptx)

adaLockedBy

Get the total amount of 'Ada' locked by the given validator in this transaction.

adaLockedBy :: TxInfo -> ValidatorHash -> Ada
adaLockedBy ptx h = Ada.fromValue (valueLockedBy ptx h)

signsTransaction

Check if the provided signature is the result of signing the pending transaction (without witnesses) with the given public key.

signsTransaction :: Signature -> PubKey -> TxInfo -> Bool
signsTransaction (Signature sig) (PubKey (LedgerBytes pk)) TxInfo{txInfoId=TxId h} =
    verifySignature pk h sig

signsTransaction makes sure that the given PubKey actually produced Signature by signing TxInfo.

More abstractly you could think of the signature as an actual written signature, the transaction information as an important document (maybe a contract) and the public key as the person who claims to have signed this document. So you are basically checking if this person actually signed the document.

More technically, every user has a private and public key. A cryptographic signature is a function that takes a private key and an arbitrary content and returns a value (the signature). Because of some mathematical correlation between the private and public key, we can use the public key together with the signature to know if they actually match (the owner of this public key actually signed it).

valueSpent

Get the total value of inputs spent by this transaction.

valueSpent :: TxInfo -> Value
valueSpent = foldMap (txOutValue . txInInfoResolved) . txInfoInputs

valueSpent goes over each input and, not caring about it's sender, add it together.

valueProduced

Get the total value of outputs produced by this transaction.

valueProduced :: TxInfo -> Value
valueProduced = foldMap txOutValue . txInfoOutputs

valueProduced goes over each output and, not caring about it's sender, add it together. In other words, does the same thing as valueSpent, but with outputs.

ownCurrencySymbol

The 'CurrencySymbol' of the current validator script.

ownCurrencySymbol :: ScriptContext -> CurrencySymbol
ownCurrencySymbol ScriptContext{scriptContextPurpose=Minting cs} = cs
ownCurrencySymbol _                                              = Builtins.error ()

Given a ScriptContext, if it's purpose is Minting, returns the currency symbol, otherwise throws an error, as other script purposes don't have any currency sybol.

spendsOutput

Check if the pending transaction spends a specific transaction output (identified by the hash of a transaction and an index into that transactions' outputs)

spendsOutput :: TxInfo -> TxId -> Integer -> Bool
spendsOutput p h i =
    let spendsOutRef inp =
            let outRef = txInInfoOutRef inp
            in h == txOutRefId outRef
                && i == txOutRefIdx outRef

    in any spendsOutRef (txInfoInputs p)

StakingCredential

Staking credential used to assign rewards

data StakingCredential
    = StakingHash Builtins.ByteString
    | StakingPtr Integer Integer Integer -- NB: The fields should really be Word64 / Natural / Natural, but 'Integer' is our only integral type so we need to use it instead.
    deriving stock (Eq, Ord, Show, Generic)
    deriving anyclass (ToJSON, FromJSON, Serialise, Hashable, NFData)

Credential

Credential required to unlock a transaction output

data Credential
  = PubKeyCredential PubKeyHash -- ^ The transaction that spends this output must be signed by the private key
  | ScriptCredential ValidatorHash -- ^ The transaction that spends this output must include the validator script and be accepted by the validator.
    deriving stock (Eq, Ord, Show, Generic)
    deriving anyclass (ToJSON, FromJSON, Serialise, Hashable, NFData)

Credential is used to, whenever you want to consume an UTxO, prove that you actually "own" this output.

PubKey

A cryptographic public key.

newtype PubKey = PubKey { getPubKey :: LedgerBytes }
    deriving stock (Eq, Ord, Generic)
    deriving anyclass (Newtype, ToJSON, FromJSON, NFData)
    deriving newtype (P.Eq, P.Ord, Serialise, PlutusTx.IsData)
    deriving IsString via LedgerBytes
    deriving (Show, Pretty) via LedgerBytes

PubKey is someone's identity. It's used to identify the user that signed a transaction.

PubKeyHash

The hash of a public key. This is frequently used to identify the public key, rather than the key itself.

newtype PubKeyHash = PubKeyHash { getPubKeyHash :: BS.ByteString }
    deriving stock (Eq, Ord, Generic)
    deriving anyclass (ToJSON, FromJSON, Newtype, ToJSONKey, FromJSONKey, NFData)
    deriving newtype (P.Eq, P.Ord, Serialise, PlutusTx.IsData, Hashable)
    deriving IsString via LedgerBytes
    deriving (Show, Pretty) via LedgerBytes

It's one layer above the actual PubKey, usually used to identify someone without actually providing the public key it self.

PrivateKey

A cryptographic private key.

newtype PrivateKey = PrivateKey { getPrivateKey :: LedgerBytes }
    deriving stock (Eq, Ord, Generic)
    deriving anyclass (ToJSON, FromJSON, Newtype, ToJSONKey, FromJSONKey)
    deriving newtype (P.Eq, P.Ord, Serialise, PlutusTx.IsData)
    deriving (Show, Pretty) via LedgerBytes

The use of a PrivateKey is the only way to sign transactions. Only the actual user is supposed to have access to this key and, by providing the signature, the transaction content and the public key, any user can make sure that the person that signed this transaction had access to the private key and, therefore, is trustworthy.

Signature

A message with a cryptographic signature.

newtype Signature = Signature { getSignature :: Builtins.ByteString }
    deriving stock (Eq, Ord, Generic)
    deriving newtype (P.Eq, P.Ord, Serialise, PlutusTx.IsData, NFData)
    deriving (Show, Pretty) via LedgerBytes

A signature is a value produced by a function that takes a private key and an arbitrary content (usually a transaction) and returns a byte string that can be verified by only using the public key.

You could think it as an actual signature where the private key is the persons knowledge on how to write the symbols and letters in the correct way, while the content (or transaction) is a document and the public key is a set of signatures from this person used to verify it's validity.

signedBy

Check whether the given 'Signature' was signed by the private key corresponding to the given public key.

signedBy :: Signature -> PubKey -> TxId -> Bool
signedBy (Signature s) (PubKey k) txId =
    let k' = ED25519.publicKey $ KB.getLedgerBytes k
        s' = ED25519.signature s
    in throwCryptoError $ ED25519.verify <$> k' <*> pure (getTxId txId) <*> s' -- TODO: is this what we want

signTx

Sign the hash of a transaction using a private key.

signTx :: TxId -> PrivateKey -> Signature
signTx (TxId txId) = sign txId

sign

Sign a message using a private key.

sign :: BA.ByteArrayAccess a => a -> PrivateKey -> Signature
sign  msg (PrivateKey privKey) =
    let k  = ED25519.secretKey $ KB.getLedgerBytes privKey
        pk = ED25519.toPublic <$> k
        salt :: BS.ByteString
        salt = "" -- TODO: do we need better salt?
        convert = Signature . BS.pack . BA.unpack
    in throwCryptoError $ fmap convert (ED25519.sign <$> k <*> pure salt <*> pk <*> pure msg)

fromHex

fromHex :: BS.ByteString -> Either String PrivateKey
fromHex = fmap PrivateKey . KB.fromHex

toPubKey

toPublicKey :: PrivateKey -> PubKey
toPublicKey = PubKey . KB.fromBytes . BS.pack . BA.unpack . ED25519.toPublic . f . KB.bytes . getPrivateKey where
    f = throwCryptoError . ED25519.secretKey

knownPrivateKeys

knownPrivateKeys :: [PrivateKey]
knownPrivateKeys = [privateKey1, privateKey2, privateKey3, privateKey4, privateKey5, privateKey6, privateKey7, privateKey8, privateKey9, privateKey10]

Arbitrary private keys usually used for testing purposes.

DCert

A representation of the ledger DCert. Some information is digested, and not included

data DCert
  = DCertDelegRegKey StakingCredential
  | DCertDelegDeRegKey StakingCredential
  | DCertDelegDelegate
      StakingCredential
      -- ^ delegator
      PubKeyHash
      -- ^ delegatee
  | -- | A digest of the PoolParams
    DCertPoolRegister
      PubKeyHash
      -- ^ poolId
      PubKeyHash
      -- ^ pool VFR
  | -- | The retiremant certificate and the Epoch N
    DCertPoolRetire PubKeyHash Integer -- NB: Should be Word64 but we only have Integer on-chain
  | -- | A really terse Digest
    DCertGenesis
  | -- | Another really terse Digest
    DCertMir
    deriving stock (Eq, Ord, Show, Generic)
    deriving anyclass (ToJSON, FromJSON, Serialise, Hashable, NFData)

I don't know what this is, feel free to make a pull request if you do.

alwaysSucceedingNAryFunction

Creates a script which has N arguments, and always succeeds.

alwaysSucceedingNAryFunction :: Natural -> SerializedScript
alwaysSucceedingNAryFunction n = toShort $ toStrict $ serialise $ Scripts.Script $ UPLC.Program () (PLC.defaultVersion ()) (body n)
    where
        -- No more arguments! The body can be anything that doesn't fail, so we return `\x . x`
        body i | i == 0 = UPLC.LamAbs() (UPLC.DeBruijn 0) $ UPLC.Var () (UPLC.DeBruijn 1)
        -- We're using de Bruijn indices, so we can use the same binder each time!
        body i = UPLC.LamAbs () (UPLC.DeBruijn 0) $ body (i-1)

alwaysFailingNAryFunction

Creates a script which has N arguments, and always fails.

alwaysFailingNAryFunction :: Natural -> SerializedScript
alwaysFailingNAryFunction n = toShort $ toStrict $ serialise $ Scripts.Script $ UPLC.Program () (PLC.defaultVersion ()) (body n)
    where
        -- No more arguments! The body should be error.
        body i | i == 0 = UPLC.Error ()
        -- We're using de Bruijn indices, so we can use the same binder each time!
        body i = UPLC.LamAbs () (UPLC.DeBruijn 0) $ body (i-1)

Interval

An interval of @a@s.

The interval may be either closed or open at either end, meaning that the endpoints may or may not be included in the interval.

The interval can also be unbounded on either side.

data Interval a = Interval { ivFrom :: LowerBound a, ivTo :: UpperBound a }
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (FromJSON, ToJSON, Serialise, Hashable, NFData)

Because Cardano is a decentralised system and sometimes it may take a while for a transaction to spread, it's very useful to pass deadlines or simmilar arguments in intervals of time instead of periods. Interval is a data type that is used to do exactly that.

Extended

A set extended with a positive and negative infinity.

data Extended a = NegInf | Finite a | PosInf
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (FromJSON, ToJSON, Serialise, Hashable, NFData)

Extends a value to include infinity

Closure

Whether a bound is inclusive or not.

type Closure = Bool

Given an interval 0 to 10, we can include 0 or exclude it and the same thing goes to 10. For instance if we define ]0, 10] (excluding 0), 0.0001 would be inside the interval, but not 0 it self.

UpperBound

The upper bound of an interval.

data UpperBound a = UpperBound (Extended a) Closure
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (FromJSON, ToJSON, Serialise, Hashable, NFData)

A data type for deining upper bounds, it takes an extended value and a closure. The value indicates the bound it self (Finite 7 or PosInf for instance) and the closure indicates if the value is included in the interval.

LowerBound

The lower bound of an interval.

data LowerBound a = LowerBound (Extended a) Closure
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (FromJSON, ToJSON, Serialise, Hashable, NFData)

A data type for deining lower bounds

strictUpperBound

strictUpperBound :: a -> UpperBound a
strictUpperBound a = UpperBound (Finite a) False

An UpperBound that doesn't include it's value, only what's smaller

strictLowerBound

strictLowerBound :: a -> LowerBound a
strictLowerBound a = LowerBound (Finite a) False

A LowerBound that doesn't include it's value, only what's greater

lowerBound

lowerBound :: a -> LowerBound a
lowerBound a = LowerBound (Finite a) True

A LowerBound that does include it's value together with greater values

upperBound

upperBound :: a -> UpperBound a
upperBound a = UpperBound (Finite a) True

An UpperBound that does include it's value together with smaller values

interval

@interval a b@ includes all values that are greater than or equal to @a@ and smaller than or equal to @b@. Therefore it includes @a@ and @b@.

interval :: a -> a -> Interval a
interval s s' = Interval (lowerBound s) (upperBound s')

Defines an interval from a given lower bound to a given upper bound, including the values themself

singleton

singleton :: a -> Interval a
singleton s = interval s s

Makes an interval representation of a point in time. Because a singleton only cares about one value, the upper bound is equal to the lower bound

from

@from a@ is an 'Interval' that includes all values that are greater than or equal to @a@.

from :: a -> Interval a
from s = Interval (lowerBound s) (UpperBound PosInf True)

Defines an interval that goes from the given value to infinity. This means that, given a value A, any other value greater or equal to A is part of this interval

to

@to a@ is an 'Interval' that includes all values that are smaller than @a@.

to :: a -> Interval a
to s = Interval (LowerBound NegInf True) (upperBound s)

Defines an interval that goes from the given value to negative infinity. This means that, given a value A, any other value less or equal to A is part of this interval

always

An 'Interval' that covers every slot.

always :: Interval a
always = Interval (LowerBound NegInf True) (UpperBound PosInf True)

never

An 'Interval' that is empty.

never :: Interval a
never = Interval (LowerBound PosInf True) (UpperBound NegInf True)

member

Check whether a value is in an interval.

member :: Ord a => a -> Interval a -> Bool
member a i = i `contains` singleton a

overlaps

Check whether two intervals overlap, that is, whether there is a value that is a member of both intervals.

overlaps :: Ord a => Interval a -> Interval a -> Bool
overlaps l r = not $ isEmpty (l `intersection` r)

intersection

'intersection a b' is the largest interval that is contained in 'a' and in 'b', if it exists.

intersection :: Ord a => Interval a -> Interval a -> Interval a
intersection (Interval l1 h1) (Interval l2 h2) = Interval (max l1 l2) (min h1 h2)

hull

'hull a b' is the smallest interval containing 'a' and 'b'.

hull :: Ord a => Interval a -> Interval a -> Interval a
hull (Interval l1 h1) (Interval l2 h2) = Interval (min l1 l2) (max h1 h2)

contains

@a contains b@ is true if the 'Interval' @b@ is entirely contained in @a@. That is, @a contains b@ if for every entry @s@, if @member s b@ then @member s a@.

contains :: Ord a => Interval a -> Interval a -> Bool
contains (Interval l1 h1) (Interval l2 h2) = l1 <= l2 && h2 <= h1

isEmpty

Check if an 'Interval' is empty.

isEmpty :: Ord a => Interval a -> Bool
isEmpty (Interval (LowerBound v1 in1) (UpperBound v2 in2)) = case v1 `compare` v2 of
    LT -> False
    GT -> True
    EQ -> not (in1 && in2)

before

Check if a value is earlier than the beginning of an 'Interval'.

before :: Ord a => a -> Interval a -> Bool
before h (Interval f _) = lowerBound h < f

after

Check if a value is later than the end of a 'Interval'.

after :: Ord a => a -> Interval a -> Bool
after h (Interval _ t) = upperBound h > t

Nothing here, feel free to contribute!

Script

A script on the chain. This is an opaque type as far as the chain is concerned.

newtype Script = Script { unScript :: UPLC.Program UPLC.DeBruijn PLC.DefaultUni PLC.DefaultFun () }
  deriving stock Generic

instance Eq Script where
    {-# INLINABLE (==) #-}
    a == b = BSL.toStrict (serialise a) == BSL.toStrict (serialise b)

instance Haskell.Eq Script where
    a == b = BSL.toStrict (serialise a) == BSL.toStrict (serialise b)

instance Ord Script where
    {-# INLINABLE compare #-}
    a `compare` b = BSL.toStrict (serialise a) `compare` BSL.toStrict (serialise b)

instance Haskell.Ord Script where
    a `compare` b = BSL.toStrict (serialise a) `compare` BSL.toStrict (serialise b)

instance Haskell.Show Script where
    showsPrec _ _ = Haskell.showString "<Script>"

instance NFData Script

scriptSize

The size of a 'Script'. No particular interpretation is given to this, other than that it is proportional to the serialized size of the script.

scriptSize :: Script -> Integer
scriptSize (Script s) = UPLC.programSize s

fromCompiledCode

Turn a 'CompiledCode' (usually produced by 'compile') into a 'Script' for use with this package.

fromCompiledCode :: CompiledCode a -> Script
fromCompiledCode = fromPlc . getPlc

fromPlc

fromPlc :: UPLC.Program UPLC.NamedDeBruijn PLC.DefaultUni PLC.DefaultFun () -> Script
fromPlc (UPLC.Program a v t) =
    let nameless = UPLC.termMapNames UPLC.unNameDeBruijn t
    in Script $ UPLC.Program a v nameless

applyScript

Given two 'Script's, compute the 'Script' that consists of applying the first to the second.

applyScript :: Script -> Script -> Script
applyScript (unScript -> s1) (unScript -> s2) = Script $ s1 `UPLC.applyProgram` s2

ScriptError

data ScriptError =
    EvaluationError [Haskell.String] -- ^ Expected behavior of the engine (e.g. user-provided error)
    | EvaluationException Haskell.String -- ^ Unexpected behavior of the engine (a bug)
    | MalformedScript Haskell.String -- ^ Script is wrong in some way
    deriving (Haskell.Show, Haskell.Eq, Generic, NFData)
    deriving anyclass (ToJSON, FromJSON)

A data type to handle erros that happened inside a script or in the process of creating a script

evaluateScript

Evaluate a script, returning the trace log.

evaluateScript :: forall m . (MonadError ScriptError m) => Script -> m [Haskell.String]
evaluateScript s = do
    -- TODO: evaluate the nameless debruijn program directly
    let namedProgram =
            let (UPLC.Program a v t) = unScript s
                named = UPLC.termMapNames (\(UPLC.DeBruijn ix) -> UPLC.NamedDeBruijn "" ix) t
            in UPLC.Program a v named
    p <- case PLC.runQuote $ runExceptT @PLC.FreeVariableError $ UPLC.unDeBruijnProgram namedProgram of
        Right p -> return p
        Left e  -> throwError $ MalformedScript $ Haskell.show e
    let (logOut, _tally, result) = evaluateCekTrace p
    case result of
        Right _ -> Haskell.pure ()
        Left errWithCause@(ErrorWithCause err _) -> throwError $ case err of
            InternalEvaluationError {} -> EvaluationException $ Haskell.show errWithCause
            UserEvaluationError {}     -> EvaluationError logOut -- TODO fix this error channel fuckery
    Haskell.pure logOut

mkValidatorScript

mkValidatorScript :: CompiledCode (Data -> Data -> Data -> ()) -> Validator
mkValidatorScript = Validator . fromCompiledCode

Given a compiled validator function (that takes the datum, redeemer and context and throws an error if something fails), returns an object of type Validator.

unValidatorScript

unValidatorScript :: Validator -> Script
unValidatorScript = getValidator

mkMintingPolicyScript

mkMintingPolicyScript :: CompiledCode (Data -> Data -> ()) -> MintingPolicy
mkMintingPolicyScript = MintingPolicy . fromCompiledCode

Given a compiled policy script function, returns an object of type MintingPolicy.

unMintingPolicyScript

unMintingPolicyScript :: MintingPolicy -> Script
unMintingPolicyScript = getMintingPolicy

Validator

'Validator' is a wrapper around 'Script's which are used as validators in transaction outputs.

newtype Validator = Validator { getValidator :: Script }
  deriving stock (Generic)
  deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Serialise)
  deriving anyclass (ToJSON, FromJSON, NFData)
  deriving Pretty via (PrettyShow Validator)

instance Haskell.Show Validator where
    show = const "Validator { <script> }"

instance BA.ByteArrayAccess Validator where
    length =
        BA.length . BSL.toStrict . serialise
    withByteArray =
        BA.withByteArray . BSL.toStrict . serialise

Validator is a type that represents our validator script (a function that runs and either returns void or throws an error).

Datum

'Datum' is a wrapper around 'Data' values which are used as data in transaction outputs.

newtype Datum = Datum { getDatum :: Data  }
  deriving stock (Generic, Haskell.Show)
  deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Serialise, IsData, NFData)
  deriving anyclass (ToJSON, FromJSON)
  deriving Pretty via Data

instance BA.ByteArrayAccess Datum where
    length =
        BA.length . BSL.toStrict . serialise
    withByteArray =
        BA.withByteArray . BSL.toStrict . serialise

Datum is a type that represents the dat that is stored inside script outputs. If we make a comparsion to traditional applications we could compare it (not very accuratily, but it may help) to a database, though they have a lot of differences specially when we talk about their architeture.

Redeemer

'Redeemer' is a wrapper around 'Data' values that are used as redeemers in transaction inputs.

newtype Redeemer = Redeemer { getRedeemer :: Data }
  deriving stock (Generic, Haskell.Show)
  deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Serialise, NFData)
  deriving anyclass (ToJSON, FromJSON)

instance Pretty Redeemer where
    pretty (Redeemer dat) = "Redeemer:" <+> pretty dat

instance BA.ByteArrayAccess Redeemer where
    length =
        BA.length . BSL.toStrict . serialise
    withByteArray =
        BA.withByteArray . BSL.toStrict . serialise

Differently from normal transactions, script transactions don't require the sender signature, but instead a type called Redeemer. This redeemer will be given by the person trying to consume the script UTxO and can be analyzed by the validator in order to decide if the user is or is not allowed to consume it.

MintingPolicy

-- | 'MintingPolicy' is a wrapper around 'Script's which are used as validators for minting constraints.

newtype MintingPolicy = MintingPolicy { getMintingPolicy :: Script }
  deriving stock (Generic)
  deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Serialise)
  deriving anyclass (ToJSON, FromJSON, NFData)
  deriving Pretty via (PrettyShow MintingPolicy)

instance Haskell.Show MintingPolicy where
    show = const "MintingPolicy { <script> }"

instance BA.ByteArrayAccess MintingPolicy where
    length =
        BA.length . BSL.toStrict . serialise
    withByteArray =
        BA.withByteArray . BSL.toStrict . serialise

A reference to the script that runs when someone is trying to mint a token. This script works just like the other ones, but instead of verifying if the user can consume the script UTxO, verifies if he can mint a token.

ValidatorHash

Script runtime representation of a @Digest SHA256@.

newtype ValidatorHash =
    ValidatorHash Builtins.ByteString
    deriving (IsString, Haskell.Show, Serialise, Pretty) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Hashable, IsData)
    deriving anyclass (FromJSON, ToJSON, ToJSONKey, FromJSONKey, NFData)

validatorHash

validatorHash :: Validator -> ValidatorHash
validatorHash vl = ValidatorHash $ BA.convert h' where
    h :: Digest SHA256 = hash $ BSL.toStrict e
    h' :: Digest SHA256 = hash h
    e = serialise vl

DatumHash

Script runtime representation of a @Digest SHA256@.

newtype DatumHash =
    DatumHash Builtins.ByteString
    deriving (IsString, Haskell.Show, Serialise, Pretty) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Hashable, IsData, NFData)
    deriving anyclass (FromJSON, ToJSON, ToJSONKey, FromJSONKey)

datumHash

datumHash :: Datum -> DatumHash
datumHash = DatumHash . Builtins.sha2_256 . BA.convert

RedeemerHash

Script runtime representation of a @Digest SHA256@.

newtype RedeemerHash =
    RedeemerHash Builtins.ByteString
    deriving (IsString, Haskell.Show, Serialise, Pretty) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Hashable, IsData)
    deriving anyclass (FromJSON, ToJSON, ToJSONKey, FromJSONKey)

redeemerHash

redeemerHash :: Redeemer -> RedeemerHash
redeemerHash = RedeemerHash . Builtins.sha2_256 . BA.convert

MintingPolicyHash

Script runtime representation of a @Digest SHA256@.

newtype MintingPolicyHash =
    MintingPolicyHash Builtins.ByteString
    deriving (IsString, Haskell.Show, Serialise, Pretty) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, Hashable, IsData)
    deriving anyclass (FromJSON, ToJSON, ToJSONKey, FromJSONKey)

mintingPolicyHash

mintingPolicyHash :: MintingPolicy -> MintingPolicyHash
mintingPolicyHash vl = MintingPolicyHash $ BA.convert h' where
    h :: Digest SHA256 = hash $ BSL.toStrict e
    h' :: Digest SHA256 = hash h
    e = serialise vl

Context

Information about the state of the blockchain and about the transaction that is currently being validated, represented as a value in 'Data'.

newtype Context = Context Data
    deriving stock (Generic, Haskell.Show)
    deriving anyclass (ToJSON, FromJSON)

applyValidator

Apply a 'Validator' to its 'Context', 'Datum', and 'Redeemer'.

applyValidator
    :: Context
    -> Validator
    -> Datum
    -> Redeemer
    -> Script
applyValidator (Context valData) (Validator validator) (Datum datum) (Redeemer redeemer) =
    ((validator `applyScript` (fromCompiledCode $ liftCode datum)) `applyScript` (fromCompiledCode $ liftCode redeemer)) `applyScript` (fromCompiledCode $ liftCode valData)

runScript

Evaluate a 'Validator' with its 'Context', 'Datum', and 'Redeemer', returning the log.

runScript
    :: (MonadError ScriptError m)
    => Context
    -> Validator
    -> Datum
    -> Redeemer
    -> m [Haskell.String]
runScript context validator datum redeemer = do
    evaluateScript (applyValidator context validator datum redeemer)

applyMintingPolicy

Apply 'MintingPolicy' to its 'Context' and 'Redeemer'.

applyMintingPolicyScript
    :: Context
    -> MintingPolicy
    -> Redeemer
    -> Script
applyMintingPolicyScript (Context valData) (MintingPolicy validator) (Redeemer red) =
    (validator `applyScript` (fromCompiledCode $ liftCode red)) `applyScript` (fromCompiledCode $ liftCode valData)

runMintingPolicyScript

Evaluate a 'MintingPolicy' with its 'Context' and 'Redeemer', returning the log.

runMintingPolicyScript
    :: (MonadError ScriptError m)
    => Context
    -> MintingPolicy
    -> Redeemer
    -> m [Haskell.String]
runMintingPolicyScript context mps red = do
    evaluateScript (applyMintingPolicyScript context mps red)

unitDatum

@()@ as a datum.

unitDatum :: Datum
unitDatum = Datum $ toData ()

unitRedeemer

@()@ as a redeemer.

unitRedeemer :: Redeemer
unitRedeemer = Redeemer $ toData ()

Slot

The slot number. This is a good proxy for time, since on the Cardano blockchain slots pass at a constant rate.

newtype Slot = Slot { getSlot :: Integer }
    deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
    deriving anyclass (FromJSON, FromJSONKey, ToJSON, ToJSONKey, NFData)
    deriving newtype (AdditiveSemigroup, AdditiveMonoid, AdditiveGroup, Eq, Ord, Enum, PlutusTx.IsData)
    deriving newtype (Haskell.Num, Haskell.Enum, Haskell.Real, Haskell.Integral, Serialise, Hashable)

makeLift ''Slot

instance Pretty Slot where
    pretty (Slot i) = "Slot" <+> pretty i

instance Pretty (Interval Slot) where
    pretty (Interval l h) = pretty l <+> comma <+> pretty h

SlotRange

An 'Interval' of 'Slot's.

type SlotRange = Interval Slot

width

Number of 'Slot's covered by the interval, if finite. @width (from x) == Nothing@.

width :: SlotRange -> Maybe Integer
width (Interval (LowerBound (Finite (Slot s1)) in1) (UpperBound (Finite (Slot s2)) in2)) =
    let lowestValue = if in1 then s1 else s1 + 1
        highestValue = if in2 then s2 else s2 - 1
    in if lowestValue <= highestValue
    -- +1 avoids fencepost error: width of [2,4] is 3.
    then Just $ (highestValue - lowestValue) + 1
    -- low > high, i.e. empty interval
    else Nothing
-- Infinity is involved!
width _ = Nothing

DeffiMiliSeconds

This is a length of time, as measured by a number of milliseconds.

newtype DiffMilliSeconds = DiffMilliSeconds Integer
  deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
  deriving anyclass (FromJSON, FromJSONKey, ToJSON, ToJSONKey, NFData)
  deriving newtype (Haskell.Num, AdditiveSemigroup, AdditiveMonoid, AdditiveGroup, Haskell.Enum, Eq, Ord, Haskell.Real, Haskell.Integral, Serialise, Hashable, PlutusTx.IsData)

makeLift ''DiffMilliSeconds

POSIXTime

POSIX time is measured as the number of milliseconds since 1970-01-01T00:00:00Z

newtype POSIXTime = POSIXTime { getPOSIXTime :: Integer }
  deriving stock (Haskell.Eq, Haskell.Ord, Haskell.Show, Generic)
  deriving anyclass (FromJSONKey, ToJSONKey, NFData)
  deriving newtype (AdditiveSemigroup, AdditiveMonoid, AdditiveGroup, Eq, Ord, Enum, PlutusTx.IsData)
  deriving newtype (Haskell.Num, Haskell.Enum, Haskell.Real, Haskell.Integral, Serialise, Hashable)

FromJSON

Custom FromJSON instance which allows to parse a JSON number to a 'POSIXTime' value. The parsed JSON value MUST be an 'Integer' or else the parsing fails.

instance FromJSON POSIXTime where
  parseJSON v@(Number n) =
      either (\_ -> prependFailure "parsing POSIXTime failed, " (typeMismatch "Integer" v))
             (return . POSIXTime)
             (floatingOrInteger n :: Either Haskell.Double Integer)
  parseJSON invalid =
      prependFailure "parsing POSIXTime failed, " (typeMismatch "Number" invalid)

ToJSON

Custom 'ToJSON' instance which allows to simply convert a 'POSIXTime' value to a JSON number.

instance ToJSON POSIXTime where
  toJSON (POSIXTime n) = Number $ scientific n 0
  
makeLift ''POSIXTime

POSIXTimeRange

An 'Interval' of 'POSIXTime's.

type POSIXTimeRange = Interval POSIXTime

fromMilliSecond

Simple conversion from 'DiffMilliSeconds' to 'POSIXTime'.

{-# INLINABLE fromMilliSeconds #-}
fromMilliSeconds :: DiffMilliSeconds -> POSIXTime
fromMilliSeconds (DiffMilliSeconds s) = POSIXTime s

Tx

A transaction, including witnesses for its inputs.

data Tx = Tx {
    txInputs      :: Set.Set TxIn,
    -- ^ The inputs to this transaction.
    txCollateral  :: Set.Set TxIn,
    -- ^ The collateral inputs to cover the fees in case validation of the transaction fails.
    txOutputs     :: [TxOut],
    -- ^ The outputs of this transaction, ordered so they can be referenced by index.
    txMint        :: !Value,
    -- ^ The 'Value' minted by this transaction.
    txFee         :: !Value,
    -- ^ The fee for this transaction.
    txValidRange  :: !SlotRange,
    -- ^ The 'SlotRange' during which this transaction may be validated.
    txMintScripts :: Set.Set MintingPolicy,
    -- ^ The scripts that must be run to check minting conditions.
    txSignatures  :: Map PubKey Signature,
    -- ^ Signatures of this transaction.
    txRedeemers   :: Redeemers,
    -- ^ Redeemers of the minting scripts.
    txData        :: Map DatumHash Datum
    -- ^ Datum objects recorded on this transaction.
    } deriving stock (Show, Eq, Generic)
      deriving anyclass (ToJSON, FromJSON, Serialise, NFData)

inputs

The inputs of a transaction.

inputs :: Lens' Tx (Set.Set TxIn)
inputs = lens g s where
    g = txInputs
    s tx i = tx { txInputs = i }

collateralInputs

The collateral inputs of a transaction for paying fees when validating the transaction fails.

collateralInputs :: Lens' Tx (Set.Set TxIn)
collateralInputs = lens g s where
    g = txCollateral
    s tx i = tx { txCollateral = i }

outputs

The outputs of a transaction.

outputs :: Lens' Tx [TxOut]
outputs = lens g s where
    g = txOutputs
    s tx o = tx { txOutputs = o }

validRange

The validity range of a transaction.

validRange :: Lens' Tx SlotRange
validRange = lens g s where
    g = txValidRange
    s tx o = tx { txValidRange = o }

signatures

signatures :: Lens' Tx (Map PubKey Signature)
signatures = lens g s where
    g = txSignatures
    s tx sig = tx { txSignatures = sig }

fee

fee :: Lens' Tx Value
fee = lens g s where
    g = txFee
    s tx v = tx { txFee = v }

mint

mint :: Lens' Tx Value
mint = lens g s where
    g = txMint
    s tx v = tx { txMint = v }

mintScripts

mintScripts :: Lens' Tx (Set.Set MintingPolicy)
mintScripts = lens g s where
    g = txMintScripts
    s tx fs = tx { txMintScripts = fs }

redeemers

redeemers :: Lens' Tx Redeemers
redeemers = lens g s where
    g = txRedeemers
    s tx reds = tx { txRedeemers = reds }

datumWitnesses

datumWitnesses :: Lens' Tx (Map DatumHash Datum)
datumWitnesses = lens g s where
    g = txData
    s tx dat = tx { txData = dat }

redeemers

redeemers :: Lens' Tx Redeemers
redeemers = lens g s where
    g = txRedeemers
    s tx reds = tx { txRedeemers = reds }

lookupSignature

lookupSignature :: PubKey -> Tx -> Maybe Signature
lookupSignature s Tx{txSignatures} = Map.lookup s txSignatures

Given a public key PubKey and a transaction Tx, verifies if this transaction's signatures field txSignatures (a map with PubKey representing the keys and Signature the values) contains a signature created by this public key. If it's able to find, return the signature wraped into a Just, otherwise return nothing.

lookupDatum

lookupDatum :: Tx -> DatumHash -> Maybe Datum
lookupDatum Tx{txData} h = Map.lookup h txData

Simmilar to lookupSignature, but instead of searching for a signature, searches for a Datum.

Given a datum hash DatumHash and a transaction Tx, verifies if this transaction's data field txData (a map with DatumHash representing the keys and Datum the values) contains a Datum corresponding to this hash. If it's able to find, return the data wraped into a Just, otherwise return nothing.

lookupRedeemer

lookupRedeemer :: Tx -> RedeemerPtr -> Maybe Redeemer
lookupRedeemer tx p = Map.lookup p (txRedeemers tx)

Simmilar to lookupSignature, but instead of searching for a signature, searches for a Redeemer.

Given a redeemer pointer RedeemerPtr and a transaction Tx, verifies if this transaction's redeemer field txRedeemers (a map with RedeemerPtr representing the keys and Redeemer the values) contains a Redeemer corresponding to this pointer. If it's able to find, return the redeemer wraped into a Just, otherwise return nothing.

validValuesTx

Check that all values in a transaction are non-negative.

validValuesTx :: Tx -> Bool
validValuesTx Tx{..}
  = all (nonNegative . txOutValue) txOutputs  && nonNegative txFee
    where
      nonNegative i = V.geq i mempty

TxStripped

A transaction without witnesses for its inputs.

data TxStripped = TxStripped {
    txStrippedInputs  :: Set.Set TxOutRef,
    -- ^ The inputs to this transaction, as transaction output references only.
    txStrippedOutputs :: [TxOut],
    -- ^ The outputs of this transation.
    txStrippedMint    :: !Value,
    -- ^ The 'Value' minted by this transaction.
    txStrippedFee     :: !Value
    -- ^ The fee for this transaction.
    } deriving (Show, Eq, Generic, Serialise)

strip

strip :: Tx -> TxStripped
strip Tx{..} = TxStripped i txOutputs txMint txFee where
    i = Set.map txInRef txInputs

Removes the witnesses from a transaction.

txId

Compute the id of a transaction.

txId :: Tx -> TxId
-- Double hash of a transaction, excluding its witnesses.
txId tx = TxId $ BA.convert h' where
    h :: Digest SHA256
    h = hash $ Write.toStrictByteString $ encode $ strip tx
    h' :: Digest SHA256
    h' = hash h

ScriptTag

A tag indicating the type of script that we are pointing to. NOTE: Cert/Reward are not supported right now.

data ScriptTag = Spend | Mint | Cert | Reward
    deriving stock (Show, Eq, Ord, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, NFData)

RedeemerPtr

A redeemer pointer is a pair of a script type tag t and an index i, picking out the ith script of type t in the transaction.

data RedeemerPtr = RedeemerPtr ScriptTag Integer
    deriving stock (Show, Eq, Ord, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, ToJSONKey, FromJSONKey, NFData)

A data type used to represent / index redeemers without actually holding their data.

Redeemers

type Redeemers = Map RedeemerPtr Redeemer

TxOutRef

A reference to a transaction output. This is a pair of a transaction reference, and an index indicating which of the outputs of that transaction we are referring to.

data TxOutRef = TxOutRef {
    txOutRefId  :: TxId,
    txOutRefIdx :: Integer -- ^ Index into the referenced transaction's outputs
    }
    deriving stock (Show, Eq, Ord, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, ToJSONKey, FromJSONKey, NFData)

A way of represent / index a transaction output without holding the output it self. Simmilar to RedeemerPtr.

txOutRefs

A list of a transaction's outputs paired with a 'TxOutRef's referring to them.

txOutRefs :: Tx -> [(TxOut, TxOutRef)]
txOutRefs t = mkOut <$> zip [0..] (txOutputs t) where
    mkOut (i, o) = (o, TxOutRef (txId t) i)

TxInType

The type of a transaction input.

data TxInType =
      -- TODO: these should all be hashes, with the validators and data segregated to the side
      ConsumeScriptAddress !Validator !Redeemer !Datum -- ^ A transaction input that consumes a script address with the given validator, redeemer, and datum.
    | ConsumePublicKeyAddress -- ^ A transaction input that consumes a public key address.
    deriving stock (Show, Eq, Ord, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, NFData)

Indicates if this input is consuming a normal transaction or an input one. This may be useful, because in order to consume a script a validator need's to run.

TxIn

A transaction input, consisting of a transaction output reference and an input type.

data TxIn = TxIn {
    txInRef  :: !TxOutRef,
    txInType :: Maybe TxInType
    }
    deriving stock (Show, Eq, Ord, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, NFData)

Transaction inputs TxIn are unspent outputs that are now going to be spent. To represent them, you only need a reference to the output that is being consumed txInRef and it's type txInType.

inRef

The 'TxOutRef' spent by a transaction input.

inRef :: Lens' TxIn TxOutRef
inRef = lens txInRef s where
    s txi r = txi { txInRef = r }

inType

The type of a transaction input.

inType :: Lens' TxIn (Maybe TxInType)
inType = lens txInType s where
    s txi t = txi { txInType = t }

inScripts

Validator, redeemer, and data scripts of a transaction input that spends a "pay to script" output.

inScripts :: TxIn -> Maybe (Validator, Redeemer, Datum)
inScripts TxIn{ txInType = t } = case t of
    Just (ConsumeScriptAddress v r d) -> Just (v, r, d)
    Just ConsumePublicKeyAddress      -> Nothing
    Nothing                           -> Nothing

inScripts represent the essential information of a script. Because not all inputs are of script type, this function first verifies the type and, if the input is in fact a ConsumeScriptAddress, returns it's essential components (validator, redeemer and datum).

pubKeyTxIn

A transaction input that spends a "pay to public key" output, given the witness.

pubKeyTxIn :: TxOutRef -> TxIn
pubKeyTxIn r = TxIn r (Just ConsumePublicKeyAddress)

scriptTxIn

A transaction input that spends a "pay to script" output, given witnesses.

scriptTxIn :: TxOutRef -> Validator -> Redeemer -> Datum -> TxIn
scriptTxIn ref v r d = TxIn ref . Just $ ConsumeScriptAddress v r d

pubKeyTxIns

Filter to get only the pubkey inputs.

pubKeyTxIns :: Fold (Set.Set TxIn) TxIn
pubKeyTxIns = folding (Set.filter (\TxIn{ txInType = t } -> t == Just ConsumePublicKeyAddress))

scriptTxIns

Filter to get only the script inputs.

scriptTxIns :: Fold (Set.Set TxIn) TxIn
scriptTxIns = folding . Set.filter $ \case
    TxIn{ txInType = Just ConsumeScriptAddress{} } -> True
    _                                              -> False

TxOut

A transaction output, consisting of a target address, a value, and optionally a datum hash.

data TxOut = TxOut {
    txOutAddress   :: Address,
    txOutValue     :: Value,
    txOutDatumHash :: Maybe DatumHash
    }
    deriving stock (Show, Eq, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON, NFData)

In order to represent a transaction output, you need to know who sent and how much he sent. Optionally you can also add a datum hash, which could be seen as a metadata. TxOut takes these two important arguments ( txOutAddress and txOutValue ), as well as an optional txOutDatumHash and store them.

txOutDatum

The datum attached to a 'TxOut', if there is one.

txOutDatum :: TxOut -> Maybe DatumHash
txOutDatum TxOut{txOutDatumHash} = txOutDatumHash

txOutPubKey

The public key attached to a 'TxOut', if there is one.

txOutPubKey :: TxOut -> Maybe PubKeyHash
txOutPubKey TxOut{txOutAddress} = toPubKeyHash txOutAddress

A transaction output address can be from a script or from a public key, txOutPubKey, returns the hash from the public key if the type is the right one.

outAddress

The address of a transaction output.

outAddress :: Lens' TxOut Address
outAddress = lens txOutAddress s where
    s tx a = tx { txOutAddress = a }

outValue

The value of a transaction output.

outValue :: Lens' TxOut Value
outValue = lens txOutValue s where
    s tx v = tx { txOutValue = v }

isPubKeyOut

Whether the output is a pay-to-pubkey output.

isPubKeyOut :: TxOut -> Bool
isPubKeyOut = isJust . txOutPubKey

isPayToScriptOut

Whether the output is a pay-to-script output.

isPayToScriptOut :: TxOut -> Bool
isPayToScriptOut = isJust . txOutDatum

TxOutTx

A 'TxOut' along with the 'Tx' it comes from, which may have additional information e.g. the full data script that goes with the 'TxOut'.

data TxOutTx = TxOutTx { txOutTxTx :: Tx, txOutTxOut :: TxOut }
    deriving stock (Show, Eq, Generic)
    deriving anyclass (Serialise, ToJSON, FromJSON)

txOutTxDatum

txOutTxDatum :: TxOutTx -> Maybe Datum
txOutTxDatum (TxOutTx tx out) = txOutDatum out >>= lookupDatum tx

Get's the TxOutTx output datum hash (if it exists) and use that as an input to lookupDatum tx (in case it return's a Just), which will then return a Just Datum if there is a data corresponding to the hash in the transaction context.

scriptTxOut'

Create a transaction output locked by a validator script hash with the given data script attached.

scriptTxOut' :: Value -> Address -> Datum -> TxOut
scriptTxOut' v a ds = TxOut a v (Just (datumHash ds))

scriptTxOut

Create a transaction output locked by a validator script and with the given data script attached.

scriptTxOut :: Value -> Validator -> Datum -> TxOut
scriptTxOut v vs = scriptTxOut' v (scriptAddress vs)

pubKeyTxOut

Create a transaction output locked by a public key.

pubKeyTxOut :: Value -> PubKey -> TxOut
pubKeyTxOut v pk = TxOut (pubKeyAddress pk) v Nothing

pubKeyHashTxOut

Create a transaction output locked by a public key.

pubKeyHashTxOut :: Value -> PubKeyHash -> TxOut
pubKeyHashTxOut v pkh = TxOut (pubKeyHashAddress pkh) v Nothing

unspentOutputsTx

The unspent outputs of a transaction.

unspentOutputsTx :: Tx -> Map TxOutRef TxOut
unspentOutputsTx t = Map.fromList $ fmap f $ zip [0..] $ txOutputs t where
    f (idx, o) = (TxOutRef (txId t) idx, o)

spentOutputs

The transaction output references consumed by a transaction.

spentOutputs :: Tx -> Set.Set TxOutRef
spentOutputs = Set.map txInRef . txInputs

updateUtxo

Update a map of unspent transaction outputs and signatures based on the inputs and outputs of a transaction.

updateUtxo :: Tx -> Map TxOutRef TxOut -> Map TxOutRef TxOut
updateUtxo tx unspent = (unspent `Map.withoutKeys` spentOutputs tx) `Map.union` unspentOutputsTx tx

updateUtxoCollateral

Update a map of unspent transaction outputs and signatures for a failed transaction using its collateral inputs.

updateUtxoCollateral :: Tx -> Map TxOutRef TxOut -> Map TxOutRef TxOut
updateUtxoCollateral tx unspent = unspent `Map.withoutKeys` (Set.map txInRef . txCollateral $ tx)

addSignature

Sign the transaction with a 'PrivateKey' and add the signature to the transaction's list of signatures.

addSignature :: PrivateKey -> Tx -> Tx
addSignature privK tx = tx & signatures . at pubK ?~ sig where
    sig = signTx (txId tx) privK
    pubK = toPublicKey privK

TxId

A transaction ID, using a SHA256 hash as the transaction id.

newtype TxId = TxId { getTxId :: BS.ByteString }
    deriving (Eq, Ord, Generic)
    deriving anyclass (ToJSON, ToJSONKey, FromJSON, FromJSONKey, NFData)
    deriving newtype (PlutusTx.Eq, PlutusTx.Ord, Serialise)
    deriving (Show, Pretty) via LedgerBytes

CurrencySymbol

newtype CurrencySymbol = CurrencySymbol { unCurrencySymbol :: Builtins.ByteString }
    deriving (IsString, Haskell.Show, Serialise, Pretty) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, PlutusTx.IsData)
    deriving anyclass (Hashable, ToJSONKey, FromJSONKey,  NFData)

CurrencySymbol is a bytestring that represents the script monetary policy hash of a token.

mpsSymbol

The currency symbol of a monetay policy hash

mpsSymbol :: MintingPolicyHash -> CurrencySymbol
mpsSymbol (MintingPolicyHash h) = CurrencySymbol h

currencyMPSHash

The minting policy hash of a currency symbol

currencyMPSHash :: CurrencySymbol -> MintingPolicyHash
currencyMPSHash (CurrencySymbol h) = MintingPolicyHash h

currencySymbol

currencySymbol :: ByteString -> CurrencySymbol
currencySymbol = CurrencySymbol

TokenName

ByteString of a name of a token, shown as UTF-8 string when possible

newtype TokenName = TokenName { unTokenName :: Builtins.ByteString }
    deriving (Serialise) via LedgerBytes
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Eq, Ord, PlutusTx.IsData)
    deriving anyclass (Hashable, NFData)
    deriving Pretty via (PrettyShow TokenName)

fromText

fromText :: Text -> TokenName
fromText = TokenName . E.encodeUtf8

fromTokenName

fromTokenName :: (Builtins.ByteString -> r) -> (Text -> r) -> TokenName -> r
fromTokenName handleBytestring handleText (TokenName bs) = either (\_ -> handleBytestring bs) handleText $ E.decodeUtf8' bs

asBase16

asBase16 :: Builtins.ByteString -> Text
asBase16 bs = Text.concat ["0x", JSON.encodeByteString bs]

quoted

quoted :: Text -> Text
quoted s = Text.concat ["\"", s, "\""]

toString

toString :: TokenName -> Haskell.String
toString = Text.unpack . fromTokenName asBase16 id

tokenName

tokenName :: ByteString -> TokenName
tokenName = TokenName

AssetClass

An asset class, identified by currency symbol and token name.

newtype AssetClass = AssetClass { unAssetClass :: (CurrencySymbol, TokenName) }
    deriving stock (Generic)
    deriving newtype (Haskell.Eq, Haskell.Ord, Haskell.Show, Eq, Ord, PlutusTx.IsData, Serialise)
    deriving anyclass (Hashable, NFData, ToJSON, FromJSON)
    deriving Pretty via (PrettyShow (CurrencySymbol, TokenName))

assetClass

assetClass :: CurrencySymbol -> TokenName -> AssetClass
assetClass s t = AssetClass (s, t)

Value

A cryptocurrency value. This is a map from 'CurrencySymbol's to a quantity of that currency.

newtype Value = Value { getValue :: Map.Map CurrencySymbol (Map.Map TokenName Integer) }
    deriving stock (Generic)
    deriving anyclass (ToJSON, FromJSON, Hashable, NFData)
    deriving newtype (Serialise, PlutusTx.IsData)
    deriving Pretty via (PrettyShow Value)

The symbol of a token mapped to another map represented by the token names as the keys and integers as the values (the amount of the token in question).

normalizeValue

normalizeValue :: Value -> Value
normalizeValue = Value . Map.fromList . sort . filterRange (/=Map.empty)
               . mapRange normalizeTokenMap . Map.toList . getValue
  where normalizeTokenMap = Map.fromList . sort . filterRange (/=0) . Map.toList
        filterRange p kvs = [(k,v) | (k,v) <- kvs, p v]
        mapRange f xys = [(x,f y) | (x,y) <- xys]
        sort xs = Data.List.sortBy compare xs

valueOf

Get the quantity of the given currency in the 'Value'.

valueOf :: Value -> CurrencySymbol -> TokenName -> Integer
valueOf (Value mp) cur tn =
    case Map.lookup cur mp of
        Nothing -> 0 :: Integer
        Just i  -> case Map.lookup tn i of
            Nothing -> 0
            Just v  -> v

Based on a value, a currency symbol and a token name, tries to find the corresponding amount.

symbols

The list of 'CurrencySymbol's of a 'Value'.

symbols :: Value -> [CurrencySymbol]
symbols (Value mp) = Map.keys mp

singleton

Make a 'Value' containing only the given quantity of the given currency.

singleton :: CurrencySymbol -> TokenName -> Integer -> Value
singleton c tn i = Value (Map.singleton c (Map.singleton tn i))

assetClassValue

A 'Value' containing the given amount of the asset class.

assetClassValue :: AssetClass -> Integer -> Value
assetClassValue (AssetClass (c, t)) i = singleton c t i

Because an asset class holds a currency symbol and a token name, we can create a value with this symbol and this unique token (using the singleton function) together with the given integer (the amount).

assetClassValueOf

Get the quantity of the given 'AssetClass' class in the 'Value'.

assetClassValueOf :: Value -> AssetClass -> Integer
assetClassValueOf v (AssetClass (c, t)) = valueOf v c t

Because an asset class holds a currency symbol and a token name, we can find inside a value the corresponding amount of this token.

unionVal

Combine two 'Value' maps

unionVal :: Value -> Value -> Map.Map CurrencySymbol (Map.Map TokenName (These Integer Integer))
unionVal (Value l) (Value r) =
    let
        combined = Map.union l r
        unThese k = case k of
            This a    -> This <$> a
            That b    -> That <$> b
            These a b -> Map.union a b
    in unThese <$> combined

unionWith

unionWith :: (Integer -> Integer -> Integer) -> Value -> Value -> Value
unionWith f ls rs =
    let
        combined = unionVal ls rs
        unThese k' = case k' of
            This a    -> f a 0
            That b    -> f 0 b
            These a b -> f a b
    in Value (fmap (fmap unThese) combined)

Provided a function that takes two integers, as well as, two values, unionWith maps these values and applies the function giving both value amounts as paramaters, forming a new unique value.

flattenValue

Convert a value to a simple list, keeping only the non-zero amounts.

flattenValue :: Value -> [(CurrencySymbol, TokenName, Integer)]
flattenValue v = do
    (cs, m) <- Map.toList $ getValue v
    (tn, a) <- Map.toList m
    guard $ a /= 0
    return (cs, tn, a)

Converting the value into a list of tuples can be useful for visualization and comparison.

isZero

Check whether a 'Value' is zero.

isZero :: Value -> Bool
isZero (Value xs) = Map.all (Map.all (\i -> 0 == i)) xs

checkPred

checkPred :: (These Integer Integer -> Bool) -> Value -> Value -> Bool
checkPred f l r =
    let
      inner :: Map.Map TokenName (These Integer Integer) -> Bool
      inner = Map.all f
    in
      Map.all inner (unionVal l r)

checkBinRel

Check whether a binary relation holds for value pairs of two 'Value' maps, supplying 0 where a key is only present in one of them.

checkBinRel :: (Integer -> Integer -> Bool) -> Value -> Value -> Bool
checkBinRel f l r =
    let
        unThese k' = case k' of
            This a    -> f a 0
            That b    -> f 0 b
            These a b -> f a b
    in checkPred unThese l r

geq

Check whether one 'Value' is greater than or equal to another. See 'Value' for an explanation of how operations on 'Value's work.

geq :: Value -> Value -> Bool
-- If both are zero then checkBinRel will be vacuously true, but this is fine.
geq = checkBinRel (>=)

gt

Check whether one 'Value' is strictly greater than another. See 'Value' for an explanation of how operations on 'Value's work.

gt :: Value -> Value -> Bool
-- If both are zero then checkBinRel will be vacuously true. So we have a special case.
gt l r = not (isZero l && isZero r) && checkBinRel (>) l r

leq

Check whether one 'Value' is less than or equal to another. See 'Value' for an explanation of how operations on 'Value's work.

leq :: Value -> Value -> Bool
-- If both are zero then checkBinRel will be vacuously true, but this is fine.
leq = checkBinRel (<=)

lt

Check whether one 'Value' is strictly less than another. See 'Value' for an explanation of how operations on 'Value's work.

lt :: Value -> Value -> Bool
-- If both are zero then checkBinRel will be vacuously true. So we have a special case.
lt l r = not (isZero l && isZero r) && checkBinRel (<) l r

eq

Check whether one 'Value' is equal to another. See 'Value' for an explanation of how operations on 'Value's work.

eq :: Value -> Value -> Bool
-- If both are zero then checkBinRel will be vacuously true, but this is fine.
eq = checkBinRel (==)

split

Split a value into its positive and negative parts. The first element of the tuple contains the negative parts of the value, the second element contains the positive parts.

split :: Value -> (Value, Value)
split (Value mp) = (negate (Value neg), Value pos) where
  (neg, pos) = Map.mapThese splitIntl mp

  splitIntl :: Map.Map TokenName Integer -> These (Map.Map TokenName Integer) (Map.Map TokenName Integer)
  splitIntl mp' = These l r where
    (l, r) = Map.mapThese (\i -> if i <= 0 then This i else That i) mp'

About

A repository with some information I may need to refer later about the Plutus lectures

Resources

Stars

Watchers

Forks