-
Notifications
You must be signed in to change notification settings - Fork 220
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
Review folder structure, file documentation and pragmas #37
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,36 +1,32 @@ | ||
{-# LANGUAGE DataKinds #-} | ||
{-# LANGUAGE RankNTypes #-} | ||
{-# OPTIONS_GHC -fno-warn-name-shadowing #-} | ||
|
||
-- | These are (partial) CBOR decoders for blocks and block headers. | ||
-- Note that we ignore most of the block's and header's content and only | ||
-- retrieve the pieces of information relevant to us, wallet (we do assume a | ||
-- trusted node and therefore, we needn't to care about verifying signatures and | ||
-- blocks themselves). | ||
-- | | ||
-- Copyright: © 2018-2019 IOHK | ||
-- License: MIT | ||
-- | ||
-- These are (partial) CBOR decoders for blocks and block headers. Note that we | ||
KtorZ marked this conversation as resolved.
Show resolved
Hide resolved
|
||
-- ignore most of the block's and header's content and only retrieve the pieces | ||
-- of information relevant to us, wallet (we do assume a trusted node and | ||
-- therefore, we needn't to care about verifying signatures and blocks | ||
-- themselves). | ||
-- | ||
-- The format described in the decoders below are the one used in the Byron era | ||
-- of Cardano and will endure in the first stages of Shelley. They are also used | ||
-- by components like the Rust <https://github.com/input-output-hk/cardano-http-bridge cardano-http-bridge>. | ||
|
||
module Cardano.Wallet.Binary | ||
( decodeBlock | ||
, decodeBlockHeader | ||
|
||
-- * Helpers | ||
, inspectNextToken | ||
, decodeList | ||
, decodeListIndef | ||
) where | ||
|
||
import qualified Codec.CBOR.Decoding as CBOR | ||
import qualified Codec.CBOR.Encoding as CBOR | ||
import qualified Codec.CBOR.Read as CBOR | ||
import qualified Codec.CBOR.Write as CBOR | ||
import Control.Monad | ||
( void ) | ||
import qualified Data.ByteString.Lazy as BL | ||
import Data.Set | ||
( Set ) | ||
import qualified Data.Set as Set | ||
import Data.Text | ||
( Text ) | ||
import Data.Word | ||
( Word16, Word64 ) | ||
import Prelude | ||
|
||
import Cardano.Wallet.Binary.Helpers | ||
( decodeList, decodeListIndef ) | ||
import Cardano.Wallet.Primitive | ||
( Address (..) | ||
, Block (..) | ||
|
@@ -41,8 +37,22 @@ import Cardano.Wallet.Primitive | |
, TxIn (..) | ||
, TxOut (..) | ||
) | ||
import Control.Monad | ||
( void ) | ||
import qualified Data.ByteString.Lazy as BL | ||
import Data.Set | ||
( Set ) | ||
import qualified Data.Set as Set | ||
import Data.Word | ||
( Word16, Word64 ) | ||
import Debug.Trace | ||
( traceShow ) | ||
|
||
import qualified Codec.CBOR.Decoding as CBOR | ||
import qualified Codec.CBOR.Encoding as CBOR | ||
import qualified Codec.CBOR.Read as CBOR | ||
import qualified Codec.CBOR.Write as CBOR | ||
|
||
{-# ANN module ("HLint: ignore Use <$>" :: Text) #-} | ||
|
||
decodeAddress :: CBOR.Decoder s Address | ||
decodeAddress = do | ||
|
@@ -69,14 +79,15 @@ decodeAttributes = do | |
_ <- CBOR.decodeMapLenCanonical -- Empty map of attributes | ||
return ((), CBOR.encodeMapLen 0) | ||
|
||
{-# ANN decodeBlock ("HLint: ignore Use <$>" :: String) #-} | ||
decodeBlock :: CBOR.Decoder s Block | ||
decodeBlock = do | ||
CBOR.decodeListLenCanonicalOf 2 | ||
t <- CBOR.decodeWordCanonical | ||
case t of | ||
0 -> do -- Genesis Block | ||
_ <- CBOR.decodeListLenCanonicalOf 3 | ||
header <- decodeGenesisBlockHeader | ||
h <- decodeGenesisBlockHeader | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🤔 sure… There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, there was a hlint annotation at the root of the file asking to "ignore name shadowing", I removed it and adjusted a few names to get rid of the warnings. |
||
-- NOTE | ||
-- We don't decode the body of genesis block because we don't need | ||
-- it. Genesis blocks occur at boundaries and contain various pieces | ||
|
@@ -87,14 +98,14 @@ decodeBlock = do | |
-- In theory, we should also: | ||
-- | ||
-- _ <- decodeGenesisBlockBody | ||
return $ Block header mempty | ||
return $ Block h mempty | ||
|
||
1 -> do -- Main Block | ||
_ <- CBOR.decodeListLenCanonicalOf 3 | ||
header <- decodeMainBlockHeader | ||
transactions <- decodeMainBlockBody | ||
h <- decodeMainBlockHeader | ||
txs <- decodeMainBlockBody | ||
-- _ <- decodeMainExtraData | ||
return $ Block header transactions | ||
return $ Block h txs | ||
|
||
_ -> do | ||
fail $ "decodeBlock: unknown block constructor: " <> show t | ||
|
@@ -144,22 +155,22 @@ decodeGenesisBlockHeader = do | |
_ <- decodeProtocolMagic | ||
previous <- decodePreviousBlockHeader | ||
_ <- decodeGenesisProof | ||
epochIndex <- decodeGenesisConsensusData | ||
epoch <- decodeGenesisConsensusData | ||
_ <- decodeGenesisExtraData | ||
-- NOTE | ||
-- Careful here, we do return a slot number of 0, which means that if we | ||
-- naively parse all blocks from an epoch, two of them will have a slot | ||
-- number of `0`. In practices, when parsing a full epoch, we can discard | ||
-- the genesis block entirely and we won't bother about modelling this | ||
-- extra complexity at the type-level. That's a bit dodgy though. | ||
return $ BlockHeader epochIndex 0 previous | ||
return $ BlockHeader epoch 0 previous | ||
|
||
decodeGenesisConsensusData :: CBOR.Decoder s Word64 | ||
decodeGenesisConsensusData = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
epochIndex <- CBOR.decodeWord64 | ||
epoch <- CBOR.decodeWord64 | ||
_ <- decodeDifficulty | ||
return epochIndex | ||
return epoch | ||
|
||
decodeGenesisExtraData :: CBOR.Decoder s () | ||
decodeGenesisExtraData = do | ||
|
@@ -205,9 +216,9 @@ decodeMainBlockHeader = do | |
_ <- decodeProtocolMagic | ||
previous <- decodePreviousBlockHeader | ||
_ <- decodeMainProof | ||
(epochIndex, slotNumber) <- decodeMainConsensusData | ||
(epoch, slot) <- decodeMainConsensusData | ||
_ <- decodeMainExtraData | ||
return $ BlockHeader epochIndex slotNumber previous | ||
return $ BlockHeader epoch slot previous | ||
|
||
decodeMainConsensusData :: CBOR.Decoder s (Word64, Word16) | ||
decodeMainConsensusData = do | ||
|
@@ -262,7 +273,7 @@ decodeProtocolMagic = do | |
return () | ||
|
||
decodeProxySignature | ||
:: (forall s. CBOR.Decoder s ()) | ||
:: (forall x. CBOR.Decoder x ()) | ||
-> CBOR.Decoder s () | ||
decodeProxySignature decodeIndex = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
|
@@ -303,9 +314,9 @@ decodeSharesProof = do | |
decodeSlotId :: CBOR.Decoder s (Word64, Word16) | ||
decodeSlotId = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
epochIndex <- CBOR.decodeWord64 | ||
slotNumber <- CBOR.decodeWord16 | ||
return (epochIndex, slotNumber) | ||
epoch <- CBOR.decodeWord64 | ||
slot <- CBOR.decodeWord16 | ||
return (epoch, slot) | ||
|
||
decodeSoftwareVersion :: CBOR.Decoder s () | ||
decodeSoftwareVersion = do | ||
|
@@ -318,15 +329,16 @@ decodeTx :: CBOR.Decoder s Tx | |
decodeTx = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
_ <- CBOR.decodeListLenCanonicalOf 3 | ||
inputs <- decodeListIndef decodeTxIn | ||
outputs <- decodeListIndef decodeTxOut | ||
ins <- decodeListIndef decodeTxIn | ||
outs <- decodeListIndef decodeTxOut | ||
_ <- decodeAttributes | ||
_ <- decodeList decodeTxWitness | ||
return $ Tx inputs outputs | ||
return $ Tx ins outs | ||
|
||
decodeTxPayload :: CBOR.Decoder s (Set Tx) | ||
decodeTxPayload = Set.fromList <$> decodeListIndef decodeTx | ||
|
||
{-# ANN decodeTxIn ("HLint: ignore Use <$>" :: String) #-} | ||
decodeTxIn :: CBOR.Decoder s TxIn | ||
decodeTxIn = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
|
@@ -343,16 +355,17 @@ decodeTxIn = do | |
decodeTxIn' :: CBOR.Decoder s TxIn | ||
decodeTxIn' = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
txId <- CBOR.decodeBytes | ||
tx <- Hash <$> CBOR.decodeBytes | ||
index <- CBOR.decodeWord32 | ||
return $ TxIn (Hash txId) index | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🤔 I guess the reasoning here is that the I think I'd prefer tx <- Hash <$> CBOR.decodeBytes
...
return $ TxIn tx index to make it obvious that we're not hashing a tx, but that There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. |
||
return $ TxIn tx index | ||
|
||
{-# ANN decodeTxOut ("HLint: ignore Use <$>" :: String) #-} | ||
decodeTxOut :: CBOR.Decoder s TxOut | ||
decodeTxOut = do | ||
_ <- CBOR.decodeListLenCanonicalOf 2 | ||
addr <- decodeAddress | ||
coin <- CBOR.decodeWord64 | ||
return $ TxOut addr (Coin coin) | ||
c <- CBOR.decodeWord64 | ||
return $ TxOut addr (Coin c) | ||
|
||
decodeTxProof :: CBOR.Decoder s () | ||
decodeTxProof = do | ||
|
@@ -376,3 +389,49 @@ decodeUpdateProof :: CBOR.Decoder s () | |
decodeUpdateProof = do | ||
_ <- CBOR.decodeBytes -- Update Hash | ||
return () | ||
|
||
|
||
-- * Helpers | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 |
||
|
||
-- | Inspect the next token that has to be decoded and print it to the console | ||
-- as a trace. Useful for debugging Decoders. | ||
-- Example: | ||
-- | ||
-- @ | ||
-- myDecoder :: CBOR.Decoder s MyType | ||
-- myDecoder = do | ||
-- a <- CBOR.decodeWord64 | ||
-- inspectNextToken | ||
-- [...] | ||
-- @ | ||
inspectNextToken :: CBOR.Decoder s () | ||
inspectNextToken = | ||
CBOR.peekTokenType >>= flip traceShow (return ()) | ||
|
||
-- | Decode an list of known length. Very similar to @decodeListIndef@. | ||
-- | ||
-- @ | ||
-- myDecoder :: CBOR.Decoder s [MyType] | ||
-- myDecoder = decodeList decodeOne | ||
-- where | ||
-- decodeOne :: CBOR.Decoder s MyType | ||
-- @ | ||
decodeList :: forall s a . CBOR.Decoder s a -> CBOR.Decoder s [a] | ||
decodeList decodeOne = do | ||
l <- CBOR.decodeListLenCanonical | ||
CBOR.decodeSequenceLenN (flip (:)) [] reverse l decodeOne | ||
|
||
-- | Decode an arbitrary long list. CBOR introduce a "break" character to | ||
-- mark the end of the list, so we simply decode each item until we encounter | ||
-- a break character. | ||
-- | ||
-- @ | ||
-- myDecoder :: CBOR.Decoder s [MyType] | ||
-- myDecoder = decodeListIndef decodeOne | ||
-- where | ||
-- decodeOne :: CBOR.Decoder s MyType | ||
-- @ | ||
decodeListIndef :: forall s a. CBOR.Decoder s a -> CBOR.Decoder s [a] | ||
decodeListIndef decodeOne = do | ||
_ <- CBOR.decodeListLenIndef | ||
CBOR.decodeSequenceLenIndef (flip (:)) [] reverse decodeOne |
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,7 +6,7 @@ import Cardano.Wallet.Binary | |
( decodeBlock ) | ||
import Cardano.Wallet.Binary.Packfile | ||
( PackfileError (..), decodePackfile ) | ||
import Cardano.Wallet.BinaryHelpers | ||
import Cardano.Wallet.BinarySpec | ||
( unsafeDeserialiseFromBytes ) | ||
import Cardano.Wallet.Primitive | ||
( Block (..), BlockHeader (..) ) | ||
|
@@ -34,7 +34,7 @@ testTwoBlobs = packFileHeader | |
-- Get this file from cardano-http-bridge with: | ||
-- wget -O test/data/epoch-mainnet-104 http://localhost:8080/mainnet/epoch/104 | ||
testPackfile :: FilePath | ||
testPackfile = "test/data/epoch-mainnet-104" | ||
testPackfile = "test/data/Cardano/Wallet/Binary/PackfileSpec-epoch-mainnet-104" | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 I'm starting to wonder whether there are editor plugins for browsing the shared structure There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🤷♂️ |
||
|
||
spec :: Spec | ||
spec = do | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I assume this is good, and for when an individual file is included in another project.
Other IOHK projects doesn't seem to have this. Any particular reason for why we/now?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is actually a standard practice in Haskell modules that are exported. This get parsed by
haddock
to generate a header for each module.