feat: add block search capability

This commit is contained in:
Rene Vergara 2024-10-11 12:34:53 -05:00
parent a3139033b6
commit bb65f3693c
No known key found for this signature in database
GPG key ID: 65122AD495A7F5B2
10 changed files with 4270 additions and 599 deletions

View file

@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## [0.3.0.0]
### Added
- New block component
## [0.2.0.1] ## [0.2.0.1]
### Fixed ### Fixed

4609
dev/app.js

File diff suppressed because it is too large Load diff

View file

@ -17,7 +17,6 @@ to generate this file without the comments in this block.
, "affjax-web" , "affjax-web"
, "argonaut-codecs" , "argonaut-codecs"
, "arrays" , "arrays"
, "codec-argonaut"
, "console" , "console"
, "datetime" , "datetime"
, "effect" , "effect"
@ -35,6 +34,7 @@ to generate this file without the comments in this block.
, "routing-duplex" , "routing-duplex"
, "safe-coerce" , "safe-coerce"
, "simple-json" , "simple-json"
, "strings"
, "tailrec" , "tailrec"
, "transformers" , "transformers"
, "web-events" , "web-events"

View file

@ -133,3 +133,12 @@ sumBundles t s o =
Nothing -> 0 Nothing -> 0
Just sap -> sap.value Just sap -> sap.value
) )
type Block =
{ hash :: String
, height :: Int
, confirmations :: Int
, time :: Int
, tx :: Array String
}

127
src/Exblo/Block.purs Normal file
View file

@ -0,0 +1,127 @@
module Exblo.Block where
import Prelude
import Simple.JSON (readJSON)
import Affjax.StatusCode (StatusCode(..))
import Affjax.Web as AX
import Affjax.ResponseFormat as AXRF
import Data.Maybe (Maybe(..))
import Data.Either (Either(..))
import Effect.Aff.Class (class MonadAff)
import Exblo.API (Block, ZcashNet)
import Exblo.Navigate (class Navigate, navigate, Route(..))
import Exblo.Store as S
import Exblo.Utils (css, convertToDate, safeHref)
import Halogen as H
import Halogen.HTML as HH
import Halogen.HTML.Events as HE
import Halogen.Store.Connect (Connected, connect)
import Halogen.Store.Monad (class MonadStore)
import Halogen.Store.Select (selectEq)
import Network.RemoteData (RemoteData(..))
import Web.UIEvent.MouseEvent (MouseEvent)
type Input = String
type State =
{ block :: RemoteData String Block
, network :: ZcashNet
, blid :: Input
}
data Action
= Initialize
| Receive (Connected ZcashNet Input)
| Close MouseEvent
component :: forall q o m . MonadAff m => MonadStore S.Action S.Store m => Navigate m => H.Component q Input o m
component = connect (selectEq _.network) $
H.mkComponent
{ initialState
, render
, eval: H.mkEval $ H.defaultEval
{ handleAction = handleAction
, receive = Just <<< Receive
, initialize = Just Initialize
}
}
where
initialState :: Connected ZcashNet Input -> State
initialState { context: net, input: blid} =
{ block: NotAsked
, network: net
, blid: blid
}
render :: forall cs m. State -> H.ComponentHTML Action cs m
render state =
HH.div
[ css "bigcard"
]
[ HH.h1_ [ HH.text "exblo"]
, case state.block of
NotAsked -> HH.p_ [ HH.text "Explore the Zcash blockchain" ]
Loading -> HH.p_ [ HH.text "Processing Zebra response..." ]
Failure e ->
HH.div
[ css "card" ]
[ HH.p_ [ HH.text e ]
, HH.button
[ css "btn-primary raised"
, HE.onClick \ev -> Close ev
]
[ HH.text "Back"
, HH.i [ css "ri-arrow-go-back-fill ri-lg" ] []
]
]
Success b ->
HH.div
[ css "card" ]
[ HH.table_
[ HH.tr_ [ HH.th_ [ HH.text "hash" ], HH.td_ [ HH.text b.hash ] ]
, HH.tr_ [ HH.th_ [ HH.text "mined" ], HH.td_ [ HH.text $ convertToDate b.time ] ]
, HH.tr_ [ HH.th_ [ HH.text "height" ], HH.td_ [ HH.text $ show b.height ] ]
, HH.tr_ [ HH.th_ [ HH.text "confirmations" ], HH.td_ [ HH.text $ show b.confirmations ] ]
, HH.tr_ [ HH.th_ [ HH.text "txs" ]
, HH.td_ (map (\txid -> HH.p_ [ HH.a [ safeHref $ Tx txid ] [HH.text txid]] ) b.tx)]
]
, HH.button
[ css "btn-primary raised"
, HE.onClick \ev -> Close ev
]
[ HH.text "Back"
, HH.i [ css "ri-arrow-go-back-fill ri-lg" ] []
]
]
, HH.p_
[ HH.i [ css "ri-copyright-line" ] []
, HH.text "2024 Vergara Technologies LLC"
]
]
handleAction :: forall cs o m. MonadAff m => Navigate m => Action -> H.HalogenM State Action cs o m Unit
handleAction = case _ of
Initialize -> do
H.modify_ _ { block = Loading }
term <- H.gets _.blid
res <- H.liftAff $ AX.get AXRF.string ("https://api.exblo.app/getblockinfo/" <> term)
case res of
Left err -> do
H.modify_ _ { block = Failure $ AX.printError err }
Right response -> do
case response.status of
StatusCode 200 -> do
case readJSON response.body of
Right (b :: Block) ->
H.modify_ _ { block = Success b }
Left e -> H.modify_ _ { block = Failure $ show e }
_any -> H.modify_ _ { block = Failure "Invalid block identifier" }
Receive {context: network, input: bl } -> do
st <- H.get
if (st.blid /= bl) then do
H.modify_ _ { blid = bl, network = network }
handleAction Initialize
else
H.modify_ _ { network = network }
Close _e -> navigate Home

View file

@ -7,16 +7,12 @@ import Affjax.Web as AX
import Affjax.ResponseFormat as AXRF import Affjax.ResponseFormat as AXRF
import Data.Maybe (Maybe(..)) import Data.Maybe (Maybe(..))
import Data.Either (Either(..)) import Data.Either (Either(..))
import Data.Int (toNumber) import Data.String.CodeUnits as SC
import Data.Nullable (Nullable, toMaybe)
import Data.Array (length, foldl)
import Data.Formatter.DateTime as FDT
import Data.DateTime.Instant (toDateTime, instant)
import Effect.Aff (Milliseconds(..))
import Effect.Aff as Aff import Effect.Aff as Aff
import Effect.Aff.Class (class MonadAff) import Effect.Aff.Class (class MonadAff)
import Effect.Class.Console (log) import Effect.Class.Console (log)
import Exblo.API (ZcashNet(..), Transaction(..), sumBundles) import Effect.Aff (Milliseconds(..))
import Exblo.API (ZcashNet(..))
import Exblo.Navigate (class Navigate, navigate, Route(..)) import Exblo.Navigate (class Navigate, navigate, Route(..))
import Exblo.Utils (css) import Exblo.Utils (css)
import Exblo.Store as S import Exblo.Store as S
@ -82,7 +78,7 @@ render state =
[ HH.input [ HH.input
[ css "input" [ css "input"
, HP.value state.term , HP.value state.term
, HP.placeholder "Search TX ID..." , HP.placeholder "TX ID/Block height..."
, HE.onValueInput \str -> SetTerm str , HE.onValueInput \str -> SetTerm str
] ]
, HH.button , HH.button
@ -121,7 +117,7 @@ render state =
, HH.text "Server: " , HH.text "Server: "
, HH.text state.version , HH.text state.version
, HH.br_ , HH.br_
, HH.text "UI: 0.2.0.1" , HH.text "UI: 0.3.0.0"
, HH.br_ , HH.br_
, HH.text "Zebra Node: " , HH.text "Zebra Node: "
, HH.text state.zebra , HH.text state.zebra
@ -161,16 +157,9 @@ handleAction = case _ of
Search event -> do Search event -> do
H.liftEffect $ Event.preventDefault event H.liftEffect $ Event.preventDefault event
term <- H.gets _.term term <- H.gets _.term
navigate $ Tx term if (SC.length term /= 64)
then navigate $ Blk term
convertToDate :: Int -> String else navigate $ Tx term
convertToDate secs =
case instant (Milliseconds (toNumber (1000 * secs))) of
Nothing -> "N/A"
Just i ->
case (FDT.formatDateTime "YYYY-DD-MM HH:mm:ss:SSS" <<< toDateTime) i of
Left _e -> "N/A"
Right d -> d
timer :: forall m a. MonadAff m => a -> m (HS.Emitter a) timer :: forall m a. MonadAff m => a -> m (HS.Emitter a)
timer val = do timer val = do

View file

@ -20,6 +20,7 @@ instance navigateHalogenM :: Navigate m => Navigate (HalogenM st act slots msg m
data Route data Route
= Home = Home
| Tx String | Tx String
| Blk String
derive instance genericRoute :: Generic Route _ derive instance genericRoute :: Generic Route _
derive instance eqRoute :: Eq Route derive instance eqRoute :: Eq Route
@ -29,6 +30,7 @@ routeCodec :: RouteDuplex' Route
routeCodec = root $ sum routeCodec = root $ sum
{ "Home": noArgs { "Home": noArgs
, "Tx": "tx" / segment , "Tx": "tx" / segment
, "Blk": "block" / segment
} }
safeHref :: forall r i. Route -> HH.IProp (href :: String | r) i safeHref :: forall r i. Route -> HH.IProp (href :: String | r) i

View file

@ -1,7 +1,6 @@
module Exblo.Router where module Exblo.Router where
import Prelude import Prelude
import Data.Codec.Argonaut as CA
import Data.Argonaut.Decode (decodeJson, parseJson) import Data.Argonaut.Decode (decodeJson, parseJson)
import Affjax.Web as AX import Affjax.Web as AX
import Affjax.ResponseFormat as AXRF import Affjax.ResponseFormat as AXRF
@ -9,11 +8,12 @@ import Data.Either (hush, Either(..))
import Data.Maybe (Maybe(..), fromMaybe) import Data.Maybe (Maybe(..), fromMaybe)
import Effect.Class.Console (log) import Effect.Class.Console (log)
import Effect.Aff.Class (class MonadAff) import Effect.Aff.Class (class MonadAff)
import Exblo.API (ExbloInfo(..)) import Exblo.API (ExbloInfo)
import Exblo.Home as Home import Exblo.Home as Home
import Exblo.Navigate (class Navigate, navigate, Route(..), routeCodec) import Exblo.Navigate (class Navigate, navigate, Route(..), routeCodec)
import Exblo.Store as S import Exblo.Store as S
import Exblo.Tx as Tx import Exblo.Tx as Tx
import Exblo.Block as Block
import Exblo.Utils (OpaqueSlot) import Exblo.Utils (OpaqueSlot)
import Halogen as H import Halogen as H
import Halogen.HTML as HH import Halogen.HTML as HH
@ -24,7 +24,6 @@ import Routing.Duplex (parse)
import Routing.Hash (getHash) import Routing.Hash (getHash)
import Type.Proxy (Proxy(..)) import Type.Proxy (Proxy(..))
data Query a = Navigate Route a data Query a = Navigate Route a
type State = type State =
@ -38,31 +37,33 @@ data Action
type ChildSlots = type ChildSlots =
( home :: OpaqueSlot Unit ( home :: OpaqueSlot Unit
, transaction :: OpaqueSlot Unit , transaction :: OpaqueSlot Unit
, block :: OpaqueSlot Unit
) )
deriveState :: Connected S.Store Unit -> State deriveState :: Connected S.Store Unit -> State
deriveState { context: store} = { route: Nothing } deriveState { context: store } = { route: Just Home }
component :: forall m. MonadAff m => Navigate m => MonadStore S.Action S.Store m => H.Component Query Unit Void m component :: forall m. MonadAff m => Navigate m => MonadStore S.Action S.Store m => H.Component Query Unit Void m
component = connect selectAll $ H.mkComponent component = connect selectAll $ H.mkComponent
{ initialState: deriveState { initialState: deriveState
, render: render , render: render
, eval: H.mkEval $ H.defaultEval , eval: H.mkEval $ H.defaultEval
{ handleAction = handleAction { handleAction = handleAction
, handleQuery = handleQuery , handleQuery = handleQuery
, receive = Just <<< Receive , receive = Just <<< Receive
, initialize = Just Initialize , initialize = Just Initialize
} }
} }
render :: forall m. MonadAff m => MonadStore S.Action S.Store m => Navigate m =>State -> H.ComponentHTML Action ChildSlots m render :: forall m. MonadAff m => MonadStore S.Action S.Store m => Navigate m => State -> H.ComponentHTML Action ChildSlots m
render { route } = render { route } =
case route of case route of
Just r -> Just r ->
case r of case r of
Home -> HH.slot_ (Proxy :: _ "home") unit Home.component unit Home -> HH.slot_ (Proxy :: _ "home") unit Home.component unit
Tx hex -> HH.slot_ (Proxy :: _ "transaction") unit Tx.component hex Tx hex -> HH.slot_ (Proxy :: _ "transaction") unit Tx.component hex
Nothing -> HH.div_ [ HH.text "Page not found." ] Blk id -> HH.slot_ (Proxy :: _ "block") unit Block.component id
Nothing -> HH.div_ [ HH.text "Page not found." ]
handleAction :: forall m. MonadAff m => Navigate m => MonadStore S.Action S.Store m => Action -> H.HalogenM State Action ChildSlots Void m Unit handleAction :: forall m. MonadAff m => Navigate m => MonadStore S.Action S.Store m => Action -> H.HalogenM State Action ChildSlots Void m Unit
handleAction = case _ of handleAction = case _ of
@ -70,13 +71,13 @@ handleAction = case _ of
s <- getStore s <- getStore
res <- H.liftAff $ AX.get AXRF.string (s.baseUrl <> "/getinfo") res <- H.liftAff $ AX.get AXRF.string (s.baseUrl <> "/getinfo")
case res of case res of
Left err -> do Left err -> do
log $ "/getinfo response failed " <> AX.printError err log $ "/getinfo response failed " <> AX.printError err
Right response -> do Right response -> do
case decodeJson =<< parseJson response.body of case decodeJson =<< parseJson response.body of
Right (x :: ExbloInfo) -> do Right (x :: ExbloInfo) -> do
updateStore $ S.SetInfo x updateStore $ S.SetInfo x
Left e1 -> log $ "/getinfo JSON decode failed " <> show e1 Left e1 -> log $ "/getinfo JSON decode failed " <> show e1
initialRoute <- hush <<< (parse routeCodec) <$> H.liftEffect getHash initialRoute <- hush <<< (parse routeCodec) <$> H.liftEffect getHash
navigate $ fromMaybe Home initialRoute navigate $ fromMaybe Home initialRoute
Receive { context: baseUrl } -> pure unit Receive { context: baseUrl } -> pure unit
@ -84,8 +85,8 @@ handleAction = case _ of
handleQuery :: forall a m. Query a -> H.HalogenM State Action ChildSlots Void m (Maybe a) handleQuery :: forall a m. Query a -> H.HalogenM State Action ChildSlots Void m (Maybe a)
handleQuery = case _ of handleQuery = case _ of
Navigate dest a -> do Navigate dest a -> do
{ route } <- H.get { route } <- H.get
when (route /= Just dest) do when (route /= Just dest) do
H.modify_ _ { route = Just dest } H.modify_ _ { route = Just dest }
pure (Just a) pure (Just a)

View file

@ -14,7 +14,7 @@ import Effect.Aff.Class (class MonadAff)
import Exblo.API (Transaction(..), ZcashNet, sumBundles) import Exblo.API (Transaction(..), ZcashNet, sumBundles)
import Exblo.Navigate (class Navigate, navigate, Route(..)) import Exblo.Navigate (class Navigate, navigate, Route(..))
import Exblo.Store as S import Exblo.Store as S
import Exblo.Utils (css) import Exblo.Utils (css, safeHref)
import Halogen as H import Halogen as H
import Halogen.HTML as HH import Halogen.HTML as HH
import Halogen.HTML.Events as HE import Halogen.HTML.Events as HE
@ -83,7 +83,7 @@ render state =
[ css "card" ] [ css "card" ]
[ HH.table_ [ HH.table_
[ HH.tr_ [ HH.th_ [ HH.text "tx id" ], HH.td_ [ HH.text t.txid ] ] [ HH.tr_ [ HH.th_ [ HH.text "tx id" ], HH.td_ [ HH.text t.txid ] ]
, HH.tr_ [ HH.th_ [ HH.text "block" ], HH.td_ [ HH.text (show t.height) ] ] , HH.tr_ [ HH.th_ [ HH.text "block" ], HH.td_ [ HH.a [ safeHref $ Blk $ show t.height] [ HH.text (show t.height) ] ] ]
, HH.tr_ [ HH.th_ [ HH.text "confirmations" ], HH.td_ [ HH.text (show t.confirmations) ] ] , HH.tr_ [ HH.th_ [ HH.text "confirmations" ], HH.td_ [ HH.text (show t.confirmations) ] ]
, HH.tr_ [ HH.th_ [ HH.text "expiry" ], HH.td_ [ HH.text (show t.expiry) ] ] , HH.tr_ [ HH.th_ [ HH.text "expiry" ], HH.td_ [ HH.text (show t.expiry) ] ]
, HH.tr_ [ HH.th_ [ HH.text "fee" ], HH.td_ [ HH.text (show $ sumBundles t.transparent t.sapling t.orchard) ] ] , HH.tr_ [ HH.th_ [ HH.text "fee" ], HH.td_ [ HH.text (show $ sumBundles t.transparent t.sapling t.orchard) ] ]
@ -152,8 +152,8 @@ render state =
[ css "btn-primary raised" [ css "btn-primary raised"
, HE.onClick \ev -> Close ev , HE.onClick \ev -> Close ev
] ]
[ HH.i [ css "ri-arrow-go-back-fill ri-lg" ] [] [ HH.text "Back"
, HH.text "Back" , HH.i [ css "ri-arrow-go-back-fill ri-lg" ] []
] ]
] ]
, HH.p_ , HH.p_
@ -178,8 +178,10 @@ handleAction = case _ of
case readJSON response.body of case readJSON response.body of
Right (t :: Transaction) -> Right (t :: Transaction) ->
H.modify_ _ { tx = Success t} H.modify_ _ { tx = Success t}
Left e -> H.modify_ _ { tx = Failure $ show e } Left e ->
_any -> H.modify_ _ { tx = Failure "Invalid transaction identifier" } H.modify_ _ { tx = Failure $ show e }
_any -> do
H.modify_ _ { tx = Failure "Invalid transaction identifier" }
Receive {context: network, input: hex } -> do Receive {context: network, input: hex } -> do
st <- H.get st <- H.get
if (st.hex /= hex) then do if (st.hex /= hex) then do

View file

@ -2,6 +2,14 @@ module Exblo.Utils where
import Prelude import Prelude
import Data.DateTime.Instant (toDateTime, instant)
import Data.Time.Duration (Seconds(..), convertDuration)
import Data.Either (Either(..))
import Data.Formatter.DateTime as FDT
import Data.Int (toNumber)
import Data.Maybe (Maybe(..))
import Exblo.Navigate (Route, routeCodec)
import Routing.Duplex (print)
import Halogen as H import Halogen as H
import Halogen.HTML as HH import Halogen.HTML as HH
import Halogen.HTML.Properties as HP import Halogen.HTML.Properties as HP
@ -10,3 +18,15 @@ type OpaqueSlot slot = forall query. H.Slot query Void slot
css :: forall r i. String -> HH.IProp (class :: String | r) i css :: forall r i. String -> HH.IProp (class :: String | r) i
css = HP.class_ <<< HH.ClassName css = HP.class_ <<< HH.ClassName
convertToDate :: Int -> String
convertToDate secs =
case instant (convertDuration $ Seconds (toNumber secs)) of
Nothing -> "N/A"
Just i ->
case (FDT.formatDateTime "YYYY-MM-DD HH:mm:ss" <<< toDateTime) i of
Left _e -> "N/A"
Right d -> d
safeHref :: forall r i. Route -> HH.IProp (href :: String | r) i
safeHref = HP.href <<< append "#" <<< print routeCodec