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

Add DB-Layer Unit Tests #88

Merged
merged 6 commits into from
Mar 21, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -98,8 +98,9 @@ jobs:
name: "Haddock"
script:
- tar xzf $STACK_WORK_CACHE
- cp -Rv specifications/api api
- mkdir -p haddock && mv $(stack path --local-doc-root)/* haddock
- git add haddock && git commit -m $TRAVIS_COMMIT
- git add api haddock && git commit -m $TRAVIS_COMMIT
- git checkout gh-pages && git cherry-pick -X theirs -n - && git commit --allow-empty --no-edit
- git push -f -q https://WilliamKingNoel-Bot:[email protected]/input-output-hk/cardano-wallet gh-pages &>/dev/null

Expand Down
41 changes: 39 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,54 @@
<p align="center">
<big><strong>Cardano-Wallet</strong></big>
<big><strong>Cardano Wallet</strong></big>
</p>

<p align="center">
<img width="200" src=".github/images/cardano-logo.png"/>
</p>

<p align="center">
<!-- We will need to decide on that soon enough <img src="https://img.shields.io/badge/version-x.x.x-ff69b4.svg?style=for-the-badge" /> -->
<a href="https://github.com/input-output-hk/cardano-wallet/releases"><img src="https://img.shields.io/github/release/input-output-hk/cardano-wallet.svg?style=for-the-badge" /></a>
<a href="https://travis-ci.org/input-output-hk/cardano-wallet"><img src="https://img.shields.io/travis/input-output-hk/cardano-wallet.svg?style=for-the-badge" /></a>
<a href="https://coveralls.io/github/input-output-hk/cardano-wallet"><img src="https://img.shields.io/coveralls/github/input-output-hk/cardano-wallet.svg?style=for-the-badge" /></a>
</p>

<hr/>

Cardano Wallet helps you manage your Ada. You can use it to send and
receive payments on the [Cardano](https://www.cardano.org) blockchain.

This project provides an HTTP Application Programming Interface (API)
and command-line interface (CLI) for working with your wallet.

It can be used as a component of a frontend such as
[Daedalus](https://daedaluswallet.io), which provides a friendly user
interface for wallets. Most users who would like to use Cardano should
start with Daedalus.

## Development

This source code repository contains the next major version of Cardano
Wallet, which has been completely rewritten for the
[Shelley](https://cardanoroadmap.com/) phase.

The Byron version of Cardano Wallet is in the
[cardano-sl](https://github.com/input-output-hk/cardano-sl)
repository.

## How to build

Use [Haskell Stack](https://haskellstack.org/) to build this project:

stack build --test


## Documentation

* Users of the Cardano Wallet API can refer to the [API Documentation](https://input-output-hk.github.io/cardano-wallet/api/).
* Development-related information can be found in the [Wiki](https://github.com/input-output-hk/cardano-wallet/wiki).
* To help understand the source code, refer to the [Haddock Documentation](https://input-output-hk.github.io/cardano-wallet/haddock/).


<hr/>

<p align="center">
Expand Down
1 change: 1 addition & 0 deletions src/Cardano/Wallet.hs
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,7 @@ data Wallet s where
-> Wallet s

deriving instance Show (Wallet s)
deriving instance Eq s => Eq (Wallet s)

instance NFData (Wallet s) where
rnf (Wallet utxo pending sl s) =
Expand Down
3 changes: 1 addition & 2 deletions src/Cardano/Wallet/Binary/Packfile.hs
Original file line number Diff line number Diff line change
Expand Up @@ -45,8 +45,7 @@ decodePackfile pf = case runGetOrFail getHeader pf of
Left e -> Left e
Right () -> case runGetOrFail getBlobs rest of
Left (_, _, msg) -> Left (BlobDecodeError msg)
Right ("", _, res) -> Right res
Right (_, _, _) -> Left (BlobDecodeError "Unconsumed data")
Right (_, _, res) -> Right res

data Header = Header !BS.ByteString !Int

Expand Down
131 changes: 128 additions & 3 deletions test/unit/Cardano/DBLayer/MVarSpec.hs
Original file line number Diff line number Diff line change
@@ -1,14 +1,139 @@
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE StandaloneDeriving #-}

{-# OPTIONS_GHC -fno-warn-orphans #-}

module Cardano.DBLayer.MVarSpec
( spec
) where

import Prelude

import Cardano.DBLayer
( DBLayer (..), PrimaryKey (..) )
import Cardano.DBLayer.MVar
()
( newDBLayer )
import Cardano.Wallet
( Wallet, WalletId (..), initWallet )
import Cardano.Wallet.Primitive
( IsOurs (..) )
import Control.Concurrent.Async
( mapConcurrently_ )
import Control.DeepSeq
( NFData )
import Control.Monad.IO.Class
( liftIO )
import Data.List.NonEmpty
( NonEmpty ((:|)) )
import Test.Hspec
( Spec )
( Spec, before, describe, it, shouldBe )
import Test.QuickCheck
( Arbitrary (..), Property, choose, property, vectorOf )
import Test.QuickCheck.Monadic
( monadicIO )

import qualified Data.Set as Set
import qualified Data.Text as T

spec :: Spec
spec = return ()
spec = do
describe "DB works as expected" $ before newDBLayer $ do
it "readCheckpoints . putCheckpoints yields inserted checkpoints" $
\db -> (property $ dbReadCheckpointsProp db)
it "replacement of values returns last value that was put" $
\db -> (property $ dbReplaceValsProp db)
it "multiple sequential putCheckpoints work properly" $
\db -> (property $ dbMultiplePutsSeqProp db)
it "multiple parallel putCheckpoints work properly" $
\db -> (property $ dbMultiplePutsParProp db)

{-------------------------------------------------------------------------------
Properties
-------------------------------------------------------------------------------}


dbReadCheckpointsProp
:: DBLayer IO DummyState
-> (PrimaryKey WalletId, DummyState)
-> Property
dbReadCheckpointsProp db (key, val) = monadicIO $ liftIO $ do
putCheckpoints db key (toWalletState val)
resFromDb <- readCheckpoints db key

resFromDb `shouldBe` (Just $ toWalletState val)

dbReplaceValsProp
:: DBLayer IO DummyState
-> (PrimaryKey WalletId, DummyState, DummyState)
-> Property
dbReplaceValsProp db (key, val1, val2) = monadicIO $ liftIO $ do
putCheckpoints db key (toWalletState val1)
putCheckpoints db key (toWalletState val2)
resFromDb <- readCheckpoints db key

resFromDb `shouldBe` (Just $ toWalletState val2)

dbMultiplePutsSeqProp
:: DBLayer IO DummyState
-> KeyValPairs
-> Property
dbMultiplePutsSeqProp db (KeyValPairs keyValPairs) = monadicIO $ liftIO $ do
mapM_ (\(key, val) -> putCheckpoints db key (toWalletState val)) keyValPairs
resFromDb <- Set.fromList <$> readWallets db

resFromDb `shouldBe` (Set.fromList (map fst keyValPairs))

dbMultiplePutsParProp
:: DBLayer IO DummyState
-> KeyValPairs
-> Property
dbMultiplePutsParProp db (KeyValPairs keyValPairs) = monadicIO $ liftIO $ do
mapConcurrently_ (\(key, val) -> putCheckpoints db key (toWalletState val)) keyValPairs
resFromDb <- Set.fromList <$> readWallets db
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This makes me wonder if the DBLayer shouldn't just return a Set here in a first place 🤔 ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

for me using list suggests that the events/data are stored with "time order". For distributed systems with high throughput this is very high requirement, and usually it is better to start with was stored requirement (which implies Set). Probably, here at this point it does not matter. But to be honest, I do not know requirement here

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmmm.. That's a bit of a false assumption here. Especially because there's no particular ordering defined on the checkpoints and therefore, there's no guarantee that the DB layers will preserve the insertion order.
Yet, one thing a Set will convey is the absence of duplicate in the data, which is right.


resFromDb `shouldBe` (Set.fromList (map fst keyValPairs))

{-------------------------------------------------------------------------------
Tests machinery, Arbitrary instances
-------------------------------------------------------------------------------}


newtype KeyValPairs = KeyValPairs [(PrimaryKey WalletId, DummyState)]
deriving (Show, Eq)

instance Arbitrary KeyValPairs where
-- No shrinking
arbitrary = do
pairs <- choose (10, 50) >>= flip vectorOf arbitrary
KeyValPairs <$> pure pairs

newtype DummyState = DummyState Int
deriving (Show, Eq)

instance Arbitrary DummyState where
-- No shrinking
arbitrary = DummyState <$> arbitrary

deriving instance NFData DummyState

instance IsOurs DummyState where
isOurs _ num = (True, num)

instance Semigroup DummyState where
(DummyState num1) <> (DummyState num2)
= DummyState (num1 + num2)

deriving instance Show (PrimaryKey WalletId)

instance Arbitrary (PrimaryKey WalletId) where
-- No shrinking
arbitrary = do
nums <- vectorOf 10 $ choose (0 :: Int, 9)
let key = (T.pack . show) nums
fmap PrimaryKey $ WalletId <$> pure key

toWalletState
:: (IsOurs s, Semigroup s, NFData s, Show s) => s
-> NonEmpty (Wallet s)
toWalletState val = initWallet val :| []
6 changes: 5 additions & 1 deletion test/unit/Cardano/Wallet/Binary/PackfileSpec.hs
Original file line number Diff line number Diff line change
Expand Up @@ -47,10 +47,14 @@ spec = do
let decoded = decodePackfile "\254CARDANOYOLO\NUL\NUL\NUL\SOH"
decoded `shouldBe` Left WrongFileTypeError

it "should ensure pack file version" $ do
it "should ensure pack file version is lesser" $ do
let decoded = decodePackfile "\254CARDANOPACK\NUL\NUL\NUL\2"
decoded `shouldBe` Left VersionTooNewError

it "should ensure pack file version is greater" $ do
let decoded = decodePackfile "\254CARDANOPACK\NUL\NUL\NUL\0"
decoded `shouldBe` Left VersionTooOldError

it "should decode an empty pack file" $ do
decodePackfile packFileHeader `shouldBe` Right []

Expand Down
12 changes: 6 additions & 6 deletions test/unit/Cardano/WalletLayerSpec.hs
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,8 @@ import Test.QuickCheck
, InfiniteList (..)
, Property
, arbitraryBoundedEnum
, checkCoverage
, choose
, property
, vectorOf
)
import Test.QuickCheck.Monadic
Expand All @@ -72,13 +72,13 @@ spec :: Spec
spec = do
describe "WalletLayer works as expected" $ do
it "Wallet upon creation is written down in db"
(checkCoverage walletCreationProp)
(property walletCreationProp)
it "Wallet cannot be created more than once"
(checkCoverage walletDoubleCreationProp)
(property walletDoubleCreationProp)
it "Wallet after being created can be got using valid wallet Id"
(checkCoverage walletGetProp)
(property walletGetProp)
it "Wallet with wrong wallet Id cannot be got"
(checkCoverage walletGetWrongIdProp)
(property walletGetWrongIdProp)


{-------------------------------------------------------------------------------
Expand Down Expand Up @@ -121,7 +121,7 @@ walletGetWrongIdProp newWallet = monadicIO $ liftIO $ do


{-------------------------------------------------------------------------------
Tests machinary, Arbitrary instances
Tests machinery, Arbitrary instances
-------------------------------------------------------------------------------}

data WalletLayerFixture = WalletLayerFixture {
Expand Down