-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathHandlers.hs
130 lines (124 loc) · 5.13 KB
/
Handlers.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
module Api.Handlers (
startClusterHandler,
stopClusterHandler,
) where
import Cardano.Api (serialiseToCBOR)
import Cardano.Launcher.Node (nodeSocketFile)
import Cardano.Wallet.Shelley.Launch.Cluster (RunningNode (RunningNode))
import Control.Concurrent (threadDelay)
import Control.Concurrent.Async (race)
import Control.Concurrent.MVar (isEmptyMVar, putMVar, tryTakeMVar)
import Control.Monad (unless)
import Control.Monad.Except (runExceptT, throwError)
import Control.Monad.Extra (unlessM)
import Control.Monad.IO.Class (liftIO)
import Control.Monad.Reader (ReaderT, ask, asks)
import Data.ByteString.Base16 qualified as Base16
import Data.Default (def)
import Data.Either (fromRight)
import Data.Foldable (for_)
import Data.List.Extra (firstJust)
import Data.Text.Encoding qualified as Text
import Data.Traversable (for)
import System.Directory (doesFileExist)
import System.FilePath (replaceFileName)
import Test.Plutip.Config (chainIndexPort, relayNodeLogs)
import Test.Plutip.Internal.BotPlutusInterface.Setup (keysDir)
import Test.Plutip.Internal.BotPlutusInterface.Wallet (BpiWallet (signKey), addSomeWallet)
import Test.Plutip.Internal.LocalCluster (startCluster, stopCluster)
import Test.Plutip.Internal.Types (ClusterEnv (runningNode))
import Test.Plutip.LocalCluster (cardanoMainnetAddress)
import Test.Plutip.Tools.CardanoApi (AwaitWalletFundedError (AwaitingCapiError, AwaitingTimeoutError), awaitWalletFunded)
import Types (
AppM,
ClusterStartupFailureReason (
ClusterIsRunningAlready,
NegativeLovelaces,
NodeConfigNotFound
),
ClusterStartupParameters (
ClusterStartupParameters,
keysDirectory,
nodeConfigPath,
nodeSocketPath,
privateKeys
),
Env (status),
Lovelace (unLovelace),
PrivateKey,
ServerOptions (ServerOptions, nodeLogs),
StartClusterRequest (StartClusterRequest, keysToGenerate),
StartClusterResponse (
ClusterStartupFailure,
ClusterStartupSuccess
),
StopClusterRequest (StopClusterRequest),
StopClusterResponse (StopClusterFailure, StopClusterSuccess),
)
import UnliftIO.Exception (throwString)
startClusterHandler :: ServerOptions -> StartClusterRequest -> AppM StartClusterResponse
startClusterHandler
ServerOptions {nodeLogs}
StartClusterRequest {keysToGenerate} = interpret $ do
-- Check that lovelace amounts are positive
for_ keysToGenerate $ \lovelaceAmounts -> do
for_ lovelaceAmounts $ \lovelaces -> do
unless (unLovelace lovelaces > 0) $ do
throwError NegativeLovelaces
statusMVar <- asks status
isClusterDown <- liftIO $ isEmptyMVar statusMVar
unless isClusterDown $ throwError ClusterIsRunningAlready
let cfg = def {relayNodeLogs = nodeLogs, chainIndexPort = Nothing}
(statusTVar, (clusterEnv, wallets)) <- liftIO $ startCluster cfg setup
liftIO $ putMVar statusMVar statusTVar
res <- liftIO $ race (threadDelay 2_000_000) $ waitForFundingTxs clusterEnv wallets
-- throw Exception for cardano-cli errors.
-- Ignore wait timeout error - return from this handler doesn't guarantee funded wallets immedietely.
maybe (return ()) throwString $ fromRight Nothing res
let nodeConfigPath = getNodeConfigFile clusterEnv
-- safeguard against directory tree structure changes
unlessM (liftIO $ doesFileExist nodeConfigPath) $ throwError NodeConfigNotFound
pure $
ClusterStartupSuccess $
ClusterStartupParameters
{ privateKeys = getWalletPrivateKey <$> wallets
, nodeSocketPath = getNodeSocketFile clusterEnv
, nodeConfigPath = nodeConfigPath
, keysDirectory = keysDir clusterEnv
}
where
setup :: ReaderT ClusterEnv IO (ClusterEnv, [BpiWallet])
setup = do
wallets <- do
for keysToGenerate $ \lovelaceAmounts -> do
addSomeWallet (fromInteger . unLovelace <$> lovelaceAmounts)
env <- ask
return (env, wallets)
-- wait for confirmation of funding txs, throw the first error if there's any
waitForFundingTxs clusterEnv wallets = do
res <- for wallets $ \w ->
awaitWalletFunded clusterEnv (cardanoMainnetAddress w)
return $
firstJust
( \case
Left (AwaitingCapiError e) -> Just $ show e
Left AwaitingTimeoutError -> Nothing
Right () -> Nothing
)
res
getNodeSocketFile (runningNode -> RunningNode conn _ _ _) = nodeSocketFile conn
getNodeConfigFile =
-- assumption is that node.config lies in the same directory as node.socket
flip replaceFileName "node.config" . getNodeSocketFile
getWalletPrivateKey :: BpiWallet -> PrivateKey
getWalletPrivateKey = Text.decodeUtf8 . Base16.encode . serialiseToCBOR . signKey
interpret = fmap (either ClusterStartupFailure id) . runExceptT
stopClusterHandler :: StopClusterRequest -> AppM StopClusterResponse
stopClusterHandler StopClusterRequest = do
statusMVar <- asks status
maybeClusterStatus <- liftIO $ tryTakeMVar statusMVar
case maybeClusterStatus of
Nothing -> pure $ StopClusterFailure "Cluster is not running"
Just statusTVar -> do
liftIO $ stopCluster statusTVar
pure $ StopClusterSuccess