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

Implement CIP129 class #778

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
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
1 change: 1 addition & 0 deletions cardano-api/cardano-api.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -188,6 +188,7 @@ library
other-modules:
Cardano.Api.Internal.Anchor
Cardano.Api.Internal.Certificate
Cardano.Api.Internal.CIP.CIP129
Cardano.Api.Internal.Compatible.Tx
Cardano.Api.Internal.Convenience.Construction
Cardano.Api.Internal.Convenience.Query
Expand Down
6 changes: 6 additions & 0 deletions cardano-api/src/Cardano/Api.hs
Original file line number Diff line number Diff line change
Expand Up @@ -709,6 +709,11 @@ module Cardano.Api
, Bech32DecodeError (..)
, UsingBech32 (..)

-- ** Bech32 CIP-129
, CIP129 (..)
, deserialiseFromBech32CIP129
, serialiseToBech32CIP129

-- ** Addresses

-- | Address serialisation is (sadly) special
Expand Down Expand Up @@ -1103,6 +1108,7 @@ where
import Cardano.Api.Internal.Address
import Cardano.Api.Internal.Anchor
import Cardano.Api.Internal.Block
import Cardano.Api.Internal.CIP.CIP129
import Cardano.Api.Internal.Certificate
import Cardano.Api.Internal.Convenience.Construction
import Cardano.Api.Internal.Convenience.Query
Expand Down
187 changes: 187 additions & 0 deletions cardano-api/src/Cardano/Api/Internal/CIP/CIP129.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-}
{-# OPTIONS_GHC -Wno-orphans #-}

module Cardano.Api.Internal.CIP.CIP129
( CIP129 (..)
, deserialiseFromBech32CIP129
, serialiseToBech32CIP129
)
where

import Cardano.Api.Internal.Governance.Actions.ProposalProcedure
import Cardano.Api.Internal.HasTypeProxy
import Cardano.Api.Internal.Orphans ()
import Cardano.Api.Internal.SerialiseBech32
import Cardano.Api.Internal.SerialiseRaw
import Cardano.Api.Internal.TxIn
import Cardano.Api.Internal.Utils

import Cardano.Binary qualified as CBOR
import Cardano.Ledger.Conway.Governance qualified as Gov
import Cardano.Ledger.Credential (Credential (..))
import Cardano.Ledger.Credential qualified as L
import Cardano.Ledger.Crypto (StandardCrypto)
import Cardano.Ledger.Keys qualified as L

import Codec.Binary.Bech32 qualified as Bech32
import Control.Monad (guard)
import Data.Bifunctor
import Data.ByteString (ByteString)
import Data.ByteString qualified as BS
import Data.ByteString.Base16 qualified as Base16
import Data.ByteString.Char8 qualified as C8
import Data.Text (Text)
import Data.Text.Encoding qualified as Text
import GHC.Exts (IsList (..))
import Text.Read

class SerialiseAsRawBytes a => CIP129 a where
cip129Bech32PrefixFor :: a -> Text
cip129HeaderHexByte :: a -> ByteString
cip129Bech32PrefixesPermitted :: AsType a -> [Text]

instance CIP129 (Credential L.ColdCommitteeRole StandardCrypto) where
cip129Bech32PrefixFor _ = "cc_cold"
cip129Bech32PrefixesPermitted AsColdCommitteeCredential = ["cc_cold"]
cip129HeaderHexByte c =
case c of
L.ScriptHashObj{} -> "\x13"
L.KeyHashObj{} -> "\x12"

instance HasTypeProxy (Credential L.ColdCommitteeRole StandardCrypto) where
data AsType (Credential L.ColdCommitteeRole StandardCrypto) = AsColdCommitteeCredential
proxyToAsType _ = AsColdCommitteeCredential

instance SerialiseAsRawBytes (Credential L.ColdCommitteeRole StandardCrypto) where
serialiseToRawBytes = CBOR.serialize'
deserialiseFromRawBytes AsColdCommitteeCredential =
first
( \e ->
SerialiseAsRawBytesError
("Unable to deserialise Credential ColdCommitteeRole StandardCrypto: " ++ show e)
)
. CBOR.decodeFull'

instance CIP129 (Credential L.HotCommitteeRole StandardCrypto) where
cip129Bech32PrefixFor _ = "cc_hot"
cip129Bech32PrefixesPermitted AsHotCommitteeCredential = ["cc_hot"]
cip129HeaderHexByte c =
case c of
L.ScriptHashObj{} -> "\x03"
L.KeyHashObj{} -> "\x02"

instance HasTypeProxy (Credential L.HotCommitteeRole StandardCrypto) where
data AsType (Credential L.HotCommitteeRole StandardCrypto) = AsHotCommitteeCredential
proxyToAsType _ = AsHotCommitteeCredential

instance SerialiseAsRawBytes (Credential L.HotCommitteeRole StandardCrypto) where
serialiseToRawBytes = CBOR.serialize'
deserialiseFromRawBytes AsHotCommitteeCredential =
first
( \e ->
SerialiseAsRawBytesError
("Unable to deserialise Credential HotCommitteeRole StandardCrypto: " ++ show e)
)
. CBOR.decodeFull'

instance CIP129 (Credential L.DRepRole StandardCrypto) where
cip129Bech32PrefixFor _ = "drep"
cip129Bech32PrefixesPermitted AsDrepCredential = ["drep"]
cip129HeaderHexByte c =
case c of
L.ScriptHashObj{} -> "\x23"
L.KeyHashObj{} -> "\x22"

instance HasTypeProxy (Credential L.DRepRole StandardCrypto) where
data AsType (Credential L.DRepRole StandardCrypto) = AsDrepCredential
proxyToAsType _ = AsDrepCredential

instance SerialiseAsRawBytes (Credential L.DRepRole StandardCrypto) where
serialiseToRawBytes = CBOR.serialize'
deserialiseFromRawBytes AsDrepCredential =
first
( \e ->
SerialiseAsRawBytesError ("Unable to deserialise Credential DRepRole StandardCrypto: " ++ show e)
)
. CBOR.decodeFull'

instance CIP129 (Gov.GovActionId StandardCrypto) where
cip129Bech32PrefixFor _ = "gov_action"
cip129Bech32PrefixesPermitted AsGovActionId = ["gov_action"]
cip129HeaderHexByte _ = "\x01"

instance HasTypeProxy (Gov.GovActionId StandardCrypto) where
data AsType (Gov.GovActionId StandardCrypto) = AsGovActionId
proxyToAsType _ = AsGovActionId

instance SerialiseAsRawBytes (Gov.GovActionId StandardCrypto) where
serialiseToRawBytes (Gov.GovActionId txid (Gov.GovActionIx ix)) =
let hex = Base16.encode $ C8.pack $ show ix
in mconcat [serialiseToRawBytes $ fromShelleyTxId txid, hex]
deserialiseFromRawBytes AsGovActionId bytes = do
let (txidBs, index) = BS.splitAt 32 bytes

txid <- deserialiseFromRawBytes AsTxId txidBs
let asciiIndex = C8.unpack $ Base16.decodeLenient index
case readMaybe asciiIndex of
Just ix -> return $ Gov.GovActionId (toShelleyTxId txid) (Gov.GovActionIx ix)
Nothing ->
Left $ SerialiseAsRawBytesError $ "Unable to deserialise GovActionId: invalid index: " <> asciiIndex

serialiseToBech32CIP129 :: forall a. CIP129 a => a -> Text
serialiseToBech32CIP129 a =
Bech32.encodeLenient
humanReadablePart
(Bech32.dataPartFromBytes (cip129HeaderHexByte a <> serialiseToRawBytes a))
where
prefix = cip129Bech32PrefixFor a
humanReadablePart =
case Bech32.humanReadablePartFromText prefix of
Right p -> p
Left err ->
error $
"serialiseToBech32: invalid prefix "
++ show prefix
++ ", "
++ show err

deserialiseFromBech32CIP129
:: forall a
. CIP129 a
=> AsType a -> Text -> Either Bech32DecodeError a
deserialiseFromBech32CIP129 asType bech32Str = do
(prefix, dataPart) <-
Bech32.decodeLenient bech32Str
?!. Bech32DecodingError

let actualPrefix = Bech32.humanReadablePartToText prefix
permittedPrefixes = cip129Bech32PrefixesPermitted asType
guard (actualPrefix `elem` permittedPrefixes)
?! Bech32UnexpectedPrefix actualPrefix (fromList permittedPrefixes)

payload <-
Bech32.dataPartToBytes dataPart
?! Bech32DataPartToBytesError (Bech32.dataPartToText dataPart)

let (header, credential) = BS.splitAt 1 payload

value <- case deserialiseFromRawBytes asType credential of
Right a -> Right a
Left _ -> Left $ Bech32DeserialiseFromBytesError payload

let expectedHeader = cip129HeaderHexByte value

guard (header == expectedHeader)
?! Bech32UnexpectedHeader (toBase16Text expectedHeader) (toBase16Text header)

let expectedPrefix = cip129Bech32PrefixFor value
guard (actualPrefix == expectedPrefix)
?! Bech32WrongPrefix actualPrefix expectedPrefix

return value
where
toBase16Text = Text.decodeUtf8 . Base16.encode
1 change: 1 addition & 0 deletions cardano-api/src/Cardano/Api/Internal/Keys/Shelley.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2045,6 +2045,7 @@ instance HasTextEnvelope (SigningKey DRepKey) where
---
--- Drep extended keys
---

data DRepExtendedKey

instance HasTypeProxy DRepExtendedKey where
Expand Down
10 changes: 10 additions & 0 deletions cardano-api/src/Cardano/Api/Internal/SerialiseBech32.hs
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,11 @@ data Bech32DecodeError
| -- | The human-readable prefix in the Bech32-encoded string does not
-- correspond to the prefix that should be used for the payload value.
Bech32WrongPrefix !Text !Text
| Bech32UnexpectedHeader
!Text
-- ^ Expected header
!Text
-- ^ Unexpected header
deriving (Eq, Show, Data)

instance Error Bech32DecodeError where
Expand Down Expand Up @@ -168,3 +173,8 @@ instance Error Bech32DecodeError where
[ "Mismatch in the Bech32 prefix: the actual prefix is " <> pshow actual
, ", but the prefix for this payload value should be " <> pshow expected
]
Bech32UnexpectedHeader expected actual ->
mconcat
[ "Unexpected CIP-129 Bech32 header: the actual header is " <> pshow actual
, ", but it was expected to be " <> pshow expected
]
Loading