Skip to content

Commit

Permalink
Use ledger keys' roles casting with explicit whitelist, instead of co…
Browse files Browse the repository at this point in the history
…erceKeyRole

Remove keys casting
  • Loading branch information
carbolymer committed Oct 30, 2023
1 parent 9b855a1 commit 04564ef
Show file tree
Hide file tree
Showing 11 changed files with 42 additions and 113 deletions.
1 change: 0 additions & 1 deletion cardano-api/internal/Cardano/Api/Address.hs
Original file line number Diff line number Diff line change
Expand Up @@ -523,7 +523,6 @@ makeShelleyAddressInEra sbe nw pc scr =
--

data StakeAddress where

StakeAddress
:: Shelley.Network
-> Shelley.StakeCredential StandardCrypto
Expand Down
24 changes: 11 additions & 13 deletions cardano-api/internal/Cardano/Api/Certificate.hs
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}

-- | Certificates embedded in transactions
--
Expand Down Expand Up @@ -383,7 +382,7 @@ makeMIRCertificate (MirCertificateRequirements atMostEra mirPot mirTarget) =
data DRepRegistrationRequirements era where
DRepRegistrationRequirements
:: ConwayEraOnwards era
-> VotingCredential era
-> (Ledger.Credential Ledger.DRepRole (EraCrypto (ShelleyLedgerEra era)))
-> Lovelace
-> DRepRegistrationRequirements era

Expand All @@ -392,7 +391,7 @@ makeDrepRegistrationCertificate :: ()
=> DRepRegistrationRequirements era
-> Maybe (Ledger.Anchor (EraCrypto (ShelleyLedgerEra era)))
-> Certificate era
makeDrepRegistrationCertificate (DRepRegistrationRequirements conwayOnwards (VotingCredential vcred) deposit) anchor =
makeDrepRegistrationCertificate (DRepRegistrationRequirements conwayOnwards vcred deposit) anchor =
ConwayCertificate conwayOnwards
. Ledger.ConwayTxCertGov
$ Ledger.ConwayRegDRep
Expand Down Expand Up @@ -437,14 +436,14 @@ makeCommitteeColdkeyResignationCertificate (CommitteeColdkeyResignationRequireme
data DRepUnregistrationRequirements era where
DRepUnregistrationRequirements
:: ConwayEraOnwards era
-> VotingCredential era
-> (Ledger.Credential Ledger.DRepRole (EraCrypto (ShelleyLedgerEra era)))
-> Lovelace
-> DRepUnregistrationRequirements era

makeDrepUnregistrationCertificate :: ()
=> DRepUnregistrationRequirements era
-> Certificate era
makeDrepUnregistrationCertificate (DRepUnregistrationRequirements conwayOnwards (VotingCredential vcred) deposit) =
makeDrepUnregistrationCertificate (DRepUnregistrationRequirements conwayOnwards vcred deposit) =
ConwayCertificate conwayOnwards
. Ledger.ConwayTxCertGov
. Ledger.ConwayUnRegDRep vcred
Expand Down Expand Up @@ -476,19 +475,18 @@ selectStakeCredential = fmap fromShelleyStakeCredential . \case
Ledger.RegTxCert sCred -> Just sCred
Ledger.UnRegTxCert sCred -> Just sCred
Ledger.DelegStakeTxCert sCred _ -> Just sCred
Ledger.RegPoolTxCert poolParams ->
Just . Ledger.coerceKeyRole . Ledger.KeyHashObj $ Ledger.ppId poolParams
Ledger.RetirePoolTxCert poolId _ ->
Just . Ledger.coerceKeyRole $ Ledger.KeyHashObj poolId
-- StakePool is always controlled by key, i.e. it is never a script. In other words,
-- @Credential StakePool@ cannot exist, because @ScriptHashObj@ constructor can't be used for that type.
Ledger.RegPoolTxCert _ -> Nothing -- contains StakePool key which cannot be a credential
Ledger.RetirePoolTxCert _ _ -> Nothing -- contains StakePool key which cannot be a credential

Ledger.MirTxCert _ -> Nothing
Ledger.GenesisDelegTxCert{} -> Nothing

ConwayCertificate cEra conwayCert -> conwayEraOnwardsConstraints cEra $
case conwayCert of
Ledger.RegPoolTxCert poolParams ->
Just . Ledger.coerceKeyRole . Ledger.KeyHashObj $ Ledger.ppId poolParams
Ledger.RetirePoolTxCert kh _ ->
Just . Ledger.coerceKeyRole $ Ledger.KeyHashObj kh
Ledger.RegPoolTxCert _ -> Nothing -- contains StakePool key which cannot be a credential
Ledger.RetirePoolTxCert _ _ -> Nothing -- contains StakePool key which cannot be a credential
Ledger.RegTxCert sCred -> Just sCred
Ledger.UnRegTxCert sCred -> Just sCred
Ledger.RegDepositTxCert sCred _ -> Just sCred
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ import qualified Cardano.Ledger.Api as L
import qualified Cardano.Ledger.Binary.Plain as Plain
import Cardano.Ledger.Core (EraCrypto)
import qualified Cardano.Ledger.Core as L
import Cardano.Ledger.Keys (HasKeyRole (..), KeyRole (DRepRole))
import Cardano.Ledger.Keys (KeyRole (DRepRole))

import Data.ByteString.Lazy (ByteString)
import qualified Data.Map as Map
Expand All @@ -55,74 +55,24 @@ instance IsShelleyBasedEra era => FromCBOR (GovernanceActionId era) where
!v <- shelleyBasedEraConstraints (shelleyBasedEra @era) $ Ledger.fromEraCBOR @(ShelleyLedgerEra era)
return $ GovernanceActionId v


-- TODO: Conway era -
-- These should be the different keys corresponding to the Constitutional Committee and DReps.
-- We can then derive the StakeCredentials from them.
data Voter era
= VoterCommittee (VotingCredential era) -- ^ Constitutional committee
| VoterDRep (VotingCredential era) -- ^ Delegated representative
| VoterSpo (Hash StakePoolKey) -- ^ Stake pool operator
newtype Voter era = Voter (Ledger.Voter (L.EraCrypto (ShelleyLedgerEra era)))
deriving (Show, Eq, Ord)

instance IsShelleyBasedEra era => ToCBOR (Voter era) where
toCBOR = \case
VoterCommittee v ->
CBOR.encodeListLen 2 <> CBOR.encodeWord 0 <> toCBOR v
VoterDRep v ->
CBOR.encodeListLen 2 <> CBOR.encodeWord 1 <> toCBOR v
VoterSpo v ->
CBOR.encodeListLen 2 <> CBOR.encodeWord 2 <> toCBOR v
toCBOR (Voter v) = shelleyBasedEraConstraints (shelleyBasedEra @era) $ Ledger.toEraCBOR @(ShelleyLedgerEra era) v

instance IsShelleyBasedEra era => FromCBOR (Voter era) where
fromCBOR = do
CBOR.decodeListLenOf 2
t <- CBOR.decodeWord
case t of
0 -> do
!x <- fromCBOR
return $ VoterCommittee x
1 -> do
!x <- fromCBOR
return $ VoterDRep x
2 -> do
!x <- fromCBOR
return $ VoterSpo x
_ ->
CBOR.cborError $ CBOR.DecoderErrorUnknownTag "Voter era" (fromIntegral t)
!v <- shelleyBasedEraConstraints (shelleyBasedEra @era) $ Ledger.fromEraCBOR @(ShelleyLedgerEra era)
pure $ Voter v


data Vote
= No
| Yes
| Abstain
deriving (Show, Eq)

toVoterRole :: ()
=> ConwayEraOnwards era
-> Voter era
-> Ledger.Voter (L.EraCrypto (ShelleyLedgerEra era))
toVoterRole eon =
conwayEraOnwardsConstraints eon $ \case
VoterCommittee (VotingCredential cred) ->
Ledger.CommitteeVoter $ coerceKeyRole cred -- TODO: Conway era - Alexey realllllyyy doesn't like this. We need to fix it.
VoterDRep (VotingCredential cred) ->
Ledger.DRepVoter cred
VoterSpo (StakePoolKeyHash kh) ->
Ledger.StakePoolVoter kh

fromVoterRole :: ()
=> ConwayEraOnwards era
-> Ledger.Voter (L.EraCrypto (ShelleyLedgerEra era))
-> Voter era
fromVoterRole eon =
conwayEraOnwardsConstraints eon $ \case
Ledger.CommitteeVoter cred ->
VoterCommittee (VotingCredential (coerceKeyRole cred)) -- TODO: Conway era - We shouldn't be using coerceKeyRole.
Ledger.DRepVoter cred ->
VoterDRep (VotingCredential cred)
Ledger.StakePoolVoter kh ->
VoterSpo (StakePoolKeyHash kh)

toVote :: Vote -> Ledger.Vote
toVote = \case
No -> Ledger.VoteNo
Expand All @@ -132,7 +82,7 @@ toVote = \case
toVotingCredential :: ()
=> ConwayEraOnwards era
-> StakeCredential
-> Either Plain.DecoderError (VotingCredential era)
-> Either Plain.DecoderError (Ledger.Credential DRepRole (EraCrypto (ShelleyLedgerEra era)))
toVotingCredential sbe (StakeCredentialByKey (StakeKeyHash kh)) = do
let cbor = Plain.serialize $ Ledger.KeyHashObj kh
eraDecodeVotingCredential sbe cbor
Expand All @@ -149,30 +99,8 @@ toVotingCredential _sbe (StakeCredentialByScript (ScriptHash _sh)) =
eraDecodeVotingCredential :: ()
=> ConwayEraOnwards era
-> ByteString
-> Either Plain.DecoderError (VotingCredential era)
eraDecodeVotingCredential eon bs =
conwayEraOnwardsConstraints eon $
case Plain.decodeFull bs of
Left e -> Left e
Right x -> Right $ VotingCredential x

newtype VotingCredential era = VotingCredential
{ unVotingCredential :: Ledger.Credential 'DRepRole (EraCrypto (ShelleyLedgerEra era))
}

deriving instance Show (VotingCredential crypto)
deriving instance Eq (VotingCredential crypto)
deriving instance Ord (VotingCredential crypto)

instance IsShelleyBasedEra era => ToCBOR (VotingCredential era) where
toCBOR = \case
VotingCredential v ->
shelleyBasedEraConstraints (shelleyBasedEra @era) $ CBOR.toCBOR v

instance IsShelleyBasedEra era => FromCBOR (VotingCredential era) where
fromCBOR = do
v <- shelleyBasedEraConstraints (shelleyBasedEra @era) CBOR.fromCBOR
return $ VotingCredential v
-> Either Plain.DecoderError (Ledger.Credential DRepRole (EraCrypto (ShelleyLedgerEra era)))
eraDecodeVotingCredential eon bs = conwayEraOnwardsConstraints eon $ Plain.decodeFull bs

createVotingProcedure :: ()
=> ConwayEraOnwards era
Expand Down
1 change: 0 additions & 1 deletion cardano-api/internal/Cardano/Api/Keys/Class.hs
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,6 @@ instance HasTypeProxy a => HasTypeProxy (SigningKey a) where

-- | Some key roles share the same representation and it is sometimes
-- legitimate to change the role of a key.
--
class CastVerificationKeyRole keyroleA keyroleB where

-- | Change the role of a 'VerificationKey', if the representation permits.
Expand Down
11 changes: 8 additions & 3 deletions cardano-api/internal/Cardano/Api/Keys/Shelley.hs
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,9 @@ module Cardano.Api.Keys.Shelley (
VerificationKey(..),
SigningKey(..),
Hash(..),

-- * Utilities
fromWitness
) where

import Cardano.Api.Error
Expand Down Expand Up @@ -1735,7 +1738,9 @@ instance CastVerificationKeyRole DRepExtendedKey DRepKey where
impossible =
error "castVerificationKey (DRep): byron and shelley key sizes do not match!"

--
-- Committee keys
--

-- | Coerce a key from a witness key into an arbitrary key. Dual to @Shelley.asWitness@.
fromWitness :: Shelley.HasKeyRole a
=> a Shelley.Witness c
-> a r c
fromWitness = Shelley.coerceKeyRole
1 change: 1 addition & 0 deletions cardano-api/internal/Cardano/Api/Orphans.hs
Original file line number Diff line number Diff line change
Expand Up @@ -333,3 +333,4 @@ instance Semigroup (Ledger.ConwayPParams StrictMaybe era) where
, Ledger.cppDRepDeposit = lastMappendWith Ledger.cppDRepDeposit p1 p2
, Ledger.cppDRepActivity = lastMappendWith Ledger.cppDRepActivity p1 p2
}

4 changes: 2 additions & 2 deletions cardano-api/internal/Cardano/Api/ReexposeLedger.hs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ module Cardano.Api.ReexposeLedger
, ShelleyEraTxCert(..)
, GenesisDelegCert(..)
, PoolParams (..)
, HasKeyRole(..)
, HasKeyRole
, MIRPot(..)
, MIRTarget(..)
, MIRCert(..)
Expand Down Expand Up @@ -129,7 +129,7 @@ import Cardano.Ledger.Core (EraCrypto, PParams (..), PoolCert (..), fr
import Cardano.Ledger.Credential (Credential (..))
import Cardano.Ledger.Crypto (Crypto, StandardCrypto)
import Cardano.Ledger.DRep (DRep (..), drepAnchorL, drepDepositL, drepExpiryL)
import Cardano.Ledger.Keys (HasKeyRole (..), KeyHash (..), KeyRole (..))
import Cardano.Ledger.Keys (HasKeyRole, KeyHash (..), KeyRole (..))
import Cardano.Ledger.PoolParams (PoolMetadata (..), PoolParams (..), StakePoolRelay (..))
import Cardano.Ledger.Shelley.TxCert (EraTxCert (..), GenesisDelegCert (..), MIRCert (..),
MIRPot (..), MIRTarget (..), ShelleyDelegCert (..), ShelleyEraTxCert (..),
Expand Down
8 changes: 4 additions & 4 deletions cardano-api/internal/Cardano/Api/Script.hs
Original file line number Diff line number Diff line change
Expand Up @@ -1197,7 +1197,7 @@ toShelleyMultiSig = go
where
go :: SimpleScript -> Either MultiSigError (Shelley.MultiSig era)
go (RequireSignature (PaymentKeyHash kh)) =
return $ Shelley.RequireSignature (Shelley.coerceKeyRole kh)
return $ Shelley.RequireSignature (Shelley.asWitness kh)
go (RequireAllOf s) = mapM go s <&> Shelley.RequireAllOf
go (RequireAnyOf s) = mapM go s <&> Shelley.RequireAnyOf
go (RequireMOf m s) = mapM go s <&> Shelley.RequireMOf m
Expand All @@ -1211,7 +1211,7 @@ fromShelleyMultiSig = go
where
go (Shelley.RequireSignature kh)
= RequireSignature
(PaymentKeyHash (Shelley.coerceKeyRole kh))
(PaymentKeyHash (fromWitness kh))
go (Shelley.RequireAllOf s) = RequireAllOf (map go s)
go (Shelley.RequireAnyOf s) = RequireAnyOf (map go s)
go (Shelley.RequireMOf m s) = RequireMOf m (map go s)
Expand All @@ -1226,7 +1226,7 @@ toAllegraTimelock = go
where
go :: SimpleScript -> Timelock.Timelock era
go (RequireSignature (PaymentKeyHash kh))
= Timelock.RequireSignature (Shelley.coerceKeyRole kh)
= Timelock.RequireSignature (Shelley.asWitness kh)
go (RequireAllOf s) = Timelock.RequireAllOf (Seq.fromList (map go s))
go (RequireAnyOf s) = Timelock.RequireAnyOf (Seq.fromList (map go s))
go (RequireMOf m s) = Timelock.RequireMOf m (Seq.fromList (map go s))
Expand All @@ -1241,7 +1241,7 @@ fromAllegraTimelock :: (Era era, EraCrypto era ~ StandardCrypto)
fromAllegraTimelock = go
where
go (Timelock.RequireSignature kh) = RequireSignature
(PaymentKeyHash (Shelley.coerceKeyRole kh))
(PaymentKeyHash (fromWitness kh))
go (Timelock.RequireTimeExpire t) = RequireTimeBefore t
go (Timelock.RequireTimeStart t) = RequireTimeAfter t
go (Timelock.RequireAllOf s) = RequireAllOf (map go (toList s))
Expand Down
8 changes: 4 additions & 4 deletions cardano-api/internal/Cardano/Api/Tx.hs
Original file line number Diff line number Diff line change
Expand Up @@ -793,16 +793,16 @@ getShelleyKeyWitnessVerificationKey
:: ShelleySigningKey
-> Shelley.VKey Shelley.Witness StandardCrypto
getShelleyKeyWitnessVerificationKey (ShelleyNormalSigningKey sk) =
(Shelley.coerceKeyRole :: Shelley.VKey Shelley.Payment StandardCrypto
-> Shelley.VKey Shelley.Witness StandardCrypto)
(Shelley.asWitness :: Shelley.VKey Shelley.Payment StandardCrypto
-> Shelley.VKey Shelley.Witness StandardCrypto)
. (\(PaymentVerificationKey vk) -> vk)
. getVerificationKey
. PaymentSigningKey
$ sk

getShelleyKeyWitnessVerificationKey (ShelleyExtendedSigningKey sk) =
(Shelley.coerceKeyRole :: Shelley.VKey Shelley.Payment StandardCrypto
-> Shelley.VKey Shelley.Witness StandardCrypto)
(Shelley.asWitness :: Shelley.VKey Shelley.Payment StandardCrypto
-> Shelley.VKey Shelley.Witness StandardCrypto)
. (\(PaymentVerificationKey vk) -> vk)
. (castVerificationKey :: VerificationKey PaymentExtendedKey
-> VerificationKey PaymentKey)
Expand Down
6 changes: 3 additions & 3 deletions cardano-api/internal/Cardano/Api/TxBody.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2459,7 +2459,7 @@ fromLedgerTxExtraKeyWitnesses sbe body =
then TxExtraKeyWitnessesNone
else
TxExtraKeyWitnesses w
[ PaymentKeyHash (Shelley.coerceKeyRole keyhash)
[ PaymentKeyHash (fromWitness keyhash)
| keyhash <- Set.toList $ body ^. L.reqSignerHashesTxBodyL
]
)
Expand Down Expand Up @@ -2668,12 +2668,12 @@ convMintValue txMintValue =
case toMaryValue v of
MaryValue _ ma -> ma

convExtraKeyWitnesses :: TxExtraKeyWitnesses era -> Set (Shelley.KeyHash r' StandardCrypto)
convExtraKeyWitnesses :: TxExtraKeyWitnesses era -> Set (Shelley.KeyHash Shelley.Witness StandardCrypto)
convExtraKeyWitnesses txExtraKeyWits =
case txExtraKeyWits of
TxExtraKeyWitnessesNone -> Set.empty
TxExtraKeyWitnesses _ khs -> Set.fromList
[ Shelley.coerceKeyRole kh
[ Shelley.asWitness kh
| PaymentKeyHash kh <- khs ]

convScripts
Expand Down
1 change: 0 additions & 1 deletion cardano-api/src/Cardano/Api/Shelley.hs
Original file line number Diff line number Diff line change
Expand Up @@ -261,7 +261,6 @@ module Cardano.Api.Shelley
GovernancePollAnswer(..),
GovernancePollError(..),
Vote(..),
VotingCredential(..),
Voter(..),
createProposalProcedure,
createVotingProcedure,
Expand Down

0 comments on commit 04564ef

Please sign in to comment.