Skip to content

Commit

Permalink
Add new address discovery schemes for testing and benchmarking
Browse files Browse the repository at this point in the history
  • Loading branch information
rvl authored and KtorZ committed Apr 9, 2019
1 parent 00e5f66 commit 639e8c5
Show file tree
Hide file tree
Showing 10 changed files with 230 additions and 71 deletions.
5 changes: 5 additions & 0 deletions cardano-wallet.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -307,9 +307,13 @@ benchmark restore
base
, ansi-terminal
, async
, bytestring
, cardano-wallet
, containers
, criterion-measurement
, cryptonite
, deepseq
, digest
, docopt
, fmt
, generic-lens
Expand All @@ -328,6 +332,7 @@ benchmark restore
other-modules:
Cardano.CLI
Cardano.Launcher
Cardano.Wallet.Primitive.AddressDiscovery.Fixed
if os(windows)
build-depends: Win32
other-modules: Cardano.Launcher.Windows
Expand Down
63 changes: 37 additions & 26 deletions src/Cardano/Wallet.hs
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DuplicateRecordFields #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE LambdaCase #-}
Expand All @@ -21,6 +20,7 @@ module Cardano.Wallet
, NewWallet(..)
, ErrNoSuchWallet(..)
, ErrWalletAlreadyExists(..)
, InitState(..)

-- * Construction
, mkWalletLayer
Expand Down Expand Up @@ -48,11 +48,12 @@ import Cardano.Wallet.Primitive.AddressDerivation
, publicKey
)
import Cardano.Wallet.Primitive.AddressDiscovery
( AddressPoolGap, SeqState (..), mkAddressPool )
( SeqState (..), SeqStateConfig (..), mkAddressPool )
import Cardano.Wallet.Primitive.Model
( Wallet, applyBlocks, initWallet )
import Cardano.Wallet.Primitive.Types
( Block (..)
, IsOurs (..)
, SlotId (..)
, WalletDelegation (..)
, WalletId (..)
Expand All @@ -61,6 +62,8 @@ import Cardano.Wallet.Primitive.Types
, WalletPassphraseInfo (..)
, WalletState (..)
)
import Control.DeepSeq
( NFData )
import Control.Exception
( Exception )
import Control.Monad
Expand All @@ -77,8 +80,6 @@ import Data.Functor
( ($>) )
import Data.Time.Clock
( getCurrentTime )
import GHC.Generics
( Generic )

import qualified Cardano.Wallet.DB as DB

Expand All @@ -88,7 +89,7 @@ import qualified Cardano.Wallet.DB as DB

data WalletLayer s = WalletLayer
{ createWallet
:: NewWallet
:: NewWallet s
-> ExceptT (ErrWalletAlreadyExists "createWallet") IO WalletId
-- ^ Initialise and store a new wallet, returning its ID.
, readWallet
Expand All @@ -109,7 +110,7 @@ data WalletLayer s = WalletLayer
-- benchmarking.
}

data NewWallet = NewWallet
data NewWallet s = NewWallet
{ seed
:: !(Passphrase "seed")
, secondFactor
Expand All @@ -118,36 +119,46 @@ data NewWallet = NewWallet
:: !WalletName
, passphrase
:: !(Passphrase "encryption")
, gap
:: !AddressPoolGap
} deriving (Show, Generic)
, addressDiscoveryConfig
:: AddressDiscoveryConfig s
}

{-------------------------------------------------------------------------------
Construction
-------------------------------------------------------------------------------}

class (IsOurs s, NFData s, Show s) => InitState s where
initState :: NewWallet s -> (WalletId, s)

instance InitState SeqState where
initState w = (wid, seqState)
where
rootXPrv =
generateKeyFromSeed (seed w, secondFactor w) (passphrase w)
accXPrv =
deriveAccountPrivateKey mempty rootXPrv minBound
ad = addressDiscoveryConfig w
extPool =
mkAddressPool (publicKey accXPrv) (seqStateGap ad) ExternalChain []
intPool =
mkAddressPool (publicKey accXPrv) minBound InternalChain []
wid =
WalletId (digest $ publicKey rootXPrv)
seqState = SeqState
{ externalPool = extPool
, internalPool = intPool
}

-- | Create a new instance of the wallet layer.
mkWalletLayer
:: (Exception e0)
=> DBLayer IO SeqState
:: (InitState s, Exception e0)
=> DBLayer IO s
-> NetworkLayer IO e0 e1
-> WalletLayer SeqState
-> WalletLayer s
mkWalletLayer db network = WalletLayer
{ createWallet = \w -> do
let rootXPrv =
generateKeyFromSeed (seed w, secondFactor w) (passphrase w)
let accXPrv =
deriveAccountPrivateKey mempty rootXPrv minBound
let extPool =
mkAddressPool (publicKey accXPrv) (gap w) ExternalChain []
let intPool =
mkAddressPool (publicKey accXPrv) minBound InternalChain []
let wid =
WalletId (digest $ publicKey rootXPrv)
let checkpoint = initWallet $ SeqState
{ externalPool = extPool
, internalPool = intPool
}
let (wid, s) = initState w
let checkpoint = initWallet s
now <- liftIO getCurrentTime
let metadata = WalletMetadata
{ name = Cardano.Wallet.name w
Expand Down
8 changes: 5 additions & 3 deletions src/Cardano/Wallet/Api/Server.hs
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ import Cardano.Wallet.Api.Types
, getApiMnemonicT
)
import Cardano.Wallet.Primitive.AddressDiscovery
( SeqState (..), defaultAddressPoolGap )
( SeqState (..), SeqStateConfig (..), defaultAddressPoolGap )
import Cardano.Wallet.Primitive.Model
( availableBalance, getState, totalBalance )
import Cardano.Wallet.Primitive.Types
Expand Down Expand Up @@ -127,8 +127,10 @@ postWallet w req = do
getApiT (req ^. #name)
, passphrase =
getApiT (req ^. #passphrase)
, gap =
maybe defaultAddressPoolGap getApiT (req ^. #addressPoolGap)
, addressDiscoveryConfig = SeqStateConfig
{ seqStateGap =
maybe defaultAddressPoolGap getApiT (req ^. #addressPoolGap)
}
}
getWallet w (ApiT wid)

Expand Down
8 changes: 8 additions & 0 deletions src/Cardano/Wallet/Primitive/AddressDiscovery.hs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}

-- |
-- Copyright: © 2018-2019 IOHK
Expand Down Expand Up @@ -35,6 +36,7 @@ module Cardano.Wallet.Primitive.AddressDiscovery

-- ** State
, SeqState (..)
, SeqStateConfig (..)
) where

import Prelude
Expand Down Expand Up @@ -255,6 +257,10 @@ data SeqState = SeqState

instance NFData SeqState

data SeqStateConfig = SeqStateConfig
{ seqStateGap :: !AddressPoolGap
} deriving stock (Generic, Show)

-- NOTE
-- We have to scan both the internal and external chain. Note that, the
-- account discovery algorithm is only specified for the external chain so
Expand All @@ -265,6 +271,8 @@ instance NFData SeqState
-- that they are just created in sequence by the wallet software. Hence an
-- address pool with a gap of 1 should be sufficient for the internal chain.
instance IsOurs SeqState where
type AddressDiscoveryConfig SeqState = SeqStateConfig

isOurs addr (SeqState !s1 !s2) =
let
(res1, !s1') = lookupAddress addr s1
Expand Down
1 change: 1 addition & 0 deletions src/Cardano/Wallet/Primitive/Types.hs
Original file line number Diff line number Diff line change
Expand Up @@ -402,6 +402,7 @@ newtype SignedTx = SignedTx { signedTx :: ByteString }
-- same time, and this little abstraction can buy us this without introducing
-- too much overhead.
class IsOurs s where
type AddressDiscoveryConfig s
isOurs :: Address -> s -> (Bool, s)

newtype Address = Address
Expand Down
90 changes: 90 additions & 0 deletions test/bench/Cardano/Wallet/Primitive/AddressDiscovery/Fixed.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}

-- |
-- Copyright: © 2018-2019 IOHK
-- License: MIT
--
-- Custom address discovery schemes used for testing and benchmarking.
--

module Cardano.Wallet.Primitive.AddressDiscovery.Fixed
( -- * Fixed address list
FixedAddressState (..)

-- ** State
, AnyAddressState (..)
) where

import Prelude

import Cardano.Wallet
( InitState (..), NewWallet (..) )
import Cardano.Wallet.Primitive.Types
( Address (..), IsOurs (..), WalletId (..) )
import Control.DeepSeq
( NFData )
import Crypto.Hash
( hash )
import Data.Digest.CRC32
( crc32 )
import Data.Set
( Set )
import Data.Word
( Word32 )
import GHC.Generics
( Generic )

import qualified Data.ByteString.Char8 as B8
import qualified Data.Set as Set

----------------------------------------------------------------------------

-- | Fixed Address Derivation
--
-- The given set of addresses are recognized as "ours".
data FixedAddressState = FixedAddressState
{ knownAddresses :: !(Set Address)
}
deriving stock (Generic, Show)

instance NFData FixedAddressState

instance IsOurs FixedAddressState where
type AddressDiscoveryConfig FixedAddressState = FixedAddressState
isOurs addr s = (Set.member addr (knownAddresses s), s)

instance InitState FixedAddressState where
initState w = (walletId cfg, cfg)
where cfg = addressDiscoveryConfig w

----------------------------------------------------------------------------

-- | Any Address Derivation
--
-- An arbitrary fraction of addreses are recognized as "ours". This is done by
-- looking at a checksum of the address.
data AnyAddressState = AnyAddressState
{ oursProportion :: !Double
}
deriving stock (Generic, Show)

instance NFData AnyAddressState

instance IsOurs AnyAddressState where
type AddressDiscoveryConfig AnyAddressState = AnyAddressState

isOurs (Address addr) s@(AnyAddressState p) = (crc32 addr < p', s)
where
p' = floor (fromIntegral (maxBound :: Word32) * p)

instance InitState AnyAddressState where
initState w = (walletId cfg, cfg)
where cfg = addressDiscoveryConfig w

walletId :: Show a => a -> WalletId
walletId = WalletId . hash . B8.pack . show
Loading

0 comments on commit 639e8c5

Please sign in to comment.