Skip to content

Commit

Permalink
Merge pull request #87 from obsidiansystems/aa-frontend-cleanup
Browse files Browse the repository at this point in the history
frontend: use now-upstreamed functions in Frontend.App and remove dep…
  • Loading branch information
ali-abrar authored Aug 2, 2019
2 parents 630be64 + 0f40187 commit ebd55f3
Show file tree
Hide file tree
Showing 6 changed files with 23 additions and 145 deletions.
3 changes: 3 additions & 0 deletions ChangeLog.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@ This project's release branch is `master`. This log is written from the perspect
## 2019-08-02 - Unreleased

* Remove `Rhyolite.Backend.Snap`. That module has been made obsolete by Obelisk.
* Frontend.App: Use reflex's `matchResponsesWithRequests` instead of `identifyTags` (now deleted). Change the identifier used by `TaggedRequest` and `TaggedResponse` to an `Int`
* Remove `mapRequesterT`. This has been upstreamed to reflex as `withRequesterT`
* Remove Frontend.Request and Frontend.WebSocket

## 2019-08-01 - Unreleased

Expand Down
4 changes: 2 additions & 2 deletions common/Rhyolite/WebSocket.hs
Original file line number Diff line number Diff line change
Expand Up @@ -42,14 +42,14 @@ instance FromJSON (View app ()) => FromJSON (WebSocketResponse app)
instance ToJSON (View app ()) => ToJSON (WebSocketResponse app)

-- | A request tagged with an identifier
data TaggedRequest r = TaggedRequest Value (SomeRequest r)
data TaggedRequest r = TaggedRequest Int (SomeRequest r)
deriving (Typeable, Generic)

instance Request r => FromJSON (TaggedRequest r)
instance Request r => ToJSON (TaggedRequest r)

-- | A response tagged with an identifier matching the one in the 'TaggedRequest'. The identifier is the first argument.
data TaggedResponse = TaggedResponse Value Value
data TaggedResponse = TaggedResponse Int Value
deriving (Typeable, Generic)

instance FromJSON TaggedResponse
Expand Down
82 changes: 18 additions & 64 deletions frontend/Rhyolite/Frontend/App.hs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE RecursiveDo #-}
Expand All @@ -27,14 +26,13 @@ import Control.Monad.Reader
import Control.Monad.Ref
import Control.Monad.State.Strict
import Data.Aeson
import qualified Data.Aeson as Aeson
import Data.Aeson.Types
import Data.Bifunctor
import qualified Data.ByteString.Lazy as LBS
import Data.Coerce (coerce)
import Data.Constraint (Dict (..))
import Data.Default (Default)
import qualified Data.IntMap as IntMap
import Data.Dependent.Map (DSum (..))
import qualified Data.Map as Map
import Data.Semigroup ((<>))
import Data.Text (Text)
Expand Down Expand Up @@ -268,11 +266,11 @@ runPrerenderedRhyoliteWidget url child = do
return ( _appWebSocket_notification appWebSocket
, _appWebSocket_response appWebSocket
)
(request', response') <- identifyTags request $ ffor response $ \(TaggedResponse t v) -> (t, v)
let request'' = fmap (fmapMaybe (\(t, v) -> case fromJSON v of
(request', response') <- matchResponsesWithRequests apiRequestJson request $ ffor response $ \(TaggedResponse t v) -> (t, v)
let request'' = fmap (Map.elems . Map.mapMaybeWithKey (\t v -> case fromJSON v of
Success (v' :: (SomeRequest (AppRequest app))) -> Just $ TaggedRequest t v'
_ -> Nothing)) request'
((a, vs), request) <- flip runRequesterT response' $ runQueryT (unRhyoliteWidget child) view
((a, vs), request) <- flip runRequesterT (fmapMaybe (traverseRequesterData (fmap Identity)) response') $ runQueryT (unRhyoliteWidget child) view
nubbedVs :: Dynamic t (ViewSelector app SelectedCount) <- holdUniqDyn $ incrementalToDynamic (vs :: Incremental t (AdditivePatch (ViewSelector app SelectedCount)))
view <- fmap join $ prerender (pure mempty) $ fromNotifications nubbedVs $ fmap (\_ -> SelectedCount 1) <$> notification
return a
Expand All @@ -296,15 +294,25 @@ runRhyoliteWidget url child = do
rec appWebSocket <- openWebSocket' url request'' $ fmapMaybe (\c -> if c == mempty then Nothing else Just ()) <$> nubbedVs
let notification = _appWebSocket_notification appWebSocket
response = _appWebSocket_response appWebSocket
(request', response') <- identifyTags request $ ffor response $ \(TaggedResponse t v) -> (t, v)
let request'' = fmap (fmapMaybe (\(t, v) -> case fromJSON v of
(request', response') <- matchResponsesWithRequests apiRequestJson request $ ffor response $ \(TaggedResponse t v) -> (t, v)
let request'' = fmap (Map.elems . Map.mapMaybeWithKey (\t v -> case fromJSON v of
Success (v' :: (SomeRequest (AppRequest app))) -> Just $ TaggedRequest t v'
_ -> Nothing)) request'
((a, vs), request) <- flip runRequesterT response' $ runQueryT (unRhyoliteWidget child) view
((a, vs), request) <- flip runRequesterT (fmapMaybe (traverseRequesterData (fmap Identity)) response') $ runQueryT (unRhyoliteWidget child) view
nubbedVs :: Dynamic t (ViewSelector app SelectedCount) <- holdUniqDyn $ incrementalToDynamic (vs :: Incremental t (AdditivePatch (ViewSelector app SelectedCount)))
view <- fromNotifications nubbedVs $ fmap (\_ -> SelectedCount 1) <$> notification
return (appWebSocket, a)

apiRequestJson
:: Request v
=> v a
-> (Aeson.Value, Aeson.Value -> Maybe a)
apiRequestJson r =
( requestToJSON r
, \v -> case requestResponseFromJSON r of
Dict -> parseMaybe parseJSON v
)

fromNotifications :: forall m (t :: *) vs. (Query vs, MonadHold t m, PerformEvent t m, TriggerEvent t m, MonadIO (Performable m), Reflex t, MonadFix m, Monoid (QueryResult vs))
=> Dynamic t vs
-> Event t (QueryResult vs)
Expand All @@ -317,46 +325,6 @@ fromNotifications vs ePatch = do

data Decoder f = forall a. FromJSON a => Decoder (f a)

identifyTags
:: forall t v m.
( MonadFix m
, MonadHold t m
, Reflex t
, Request v
)
=> Event t (RequesterData v)
-> Event t (Data.Aeson.Value, Data.Aeson.Value)
-> m ( Event t [(Data.Aeson.Value, Data.Aeson.Value)]
, Event t (RequesterData Identity)
)
identifyTags send recv = do
rec nextId :: Behavior t Int <- hold 1 $ fmap (\(a, _, _) -> a) send'
waitingFor :: Incremental t (PatchMap Int (Decoder RequesterDataKey)) <- holdIncremental mempty $ leftmost
[ fmap (\(_, b, _) -> b) send'
, fmap snd recv'
]
let send' = flip pushAlways send $ \dm -> do
oldNextId <- sample nextId
let (result, newNextId) = flip runState oldNextId $ forM (requesterDataToList dm) $ \(k :=> v) -> do
n <- get
put $ succ n
return (n, k :=> v)
patchWaitingFor = PatchMap $ Map.fromList $ ffor result $ \(n, k :=> v) -> case requestResponseFromJSON v of
Dict -> (n, Just (Decoder k))
toSend = ffor result $ \(n, _ :=> v) -> (toJSON n, requestToJSON v)
return (newNextId, patchWaitingFor, toSend)
let recv' = flip push recv $ \(jsonN, jsonV) -> do
wf <- sample $ currentIncremental waitingFor
case parseMaybe parseJSON jsonN of
Nothing -> return Nothing
Just n ->
return $ case Map.lookup n wf of
Just (Decoder k) -> Just $
let Just v = parseMaybe parseJSON jsonV
in (singletonRequesterData k $ Identity v, PatchMap $ Map.singleton n Nothing)
Nothing -> Nothing
return (fmap (\(_, _, c) -> c) send', fst <$> recv')

data AppWebSocket t app = AppWebSocket
{ _appWebSocket_notification :: Event t (View app ())
, _appWebSocket_response :: Event t TaggedResponse
Expand Down Expand Up @@ -483,7 +451,7 @@ mapAuth
-> RhyoliteWidget app t m a
mapAuth token authorizeQuery authenticatedChild = RhyoliteWidget $ do
v <- askQueryResult
(a, vs) <- lift $ mapRequesterT authorizeReq id $ runQueryT (withQueryT authorizeQuery authenticatedChild) v
(a, vs) <- lift $ withRequesterT authorizeReq id $ runQueryT (withQueryT authorizeQuery authenticatedChild) v
-- tellQueryIncremental vs would seem simpler, but tellQueryDyn is more baked, subtracting off the removals properly.
tellQueryDyn $ incrementalToDynamic vs
return a
Expand All @@ -494,17 +462,3 @@ mapAuth token authorizeQuery authenticatedChild = RhyoliteWidget $ do
authorizeReq = \case
ApiRequest_Public a -> ApiRequest_Public a
ApiRequest_Private () a -> ApiRequest_Private token a

-- TODO: Upstream to reflex
mapRequesterT
:: (Reflex t, MonadFix m)
=> (forall x. req x -> req' x)
-> (forall x. rsp' x -> rsp x)
-> RequesterT t req rsp m a
-> RequesterT t req' rsp' m a
mapRequesterT freq frsp child = do
rec let rsp = fmap (runIdentity . traverseRequesterData (Identity . frsp)) rsp'
(a, req) <- lift $ runRequesterT child rsp
rsp' <- fmap (flip selectInt 0 . fanInt . fmapCheap unMultiEntry) $ requesting' $
fmapCheap (multiEntry . IntMap.singleton 0) $ fmap (runIdentity . traverseRequesterData (Identity . freq)) req
return a
43 changes: 0 additions & 43 deletions frontend/Rhyolite/Frontend/Request.hs

This file was deleted.

35 changes: 0 additions & 35 deletions frontend/Rhyolite/Frontend/WebSocket.hs

This file was deleted.

1 change: 0 additions & 1 deletion frontend/rhyolite-frontend.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,6 @@ library
Rhyolite.Frontend.App
Rhyolite.Frontend.Cookie
Rhyolite.Frontend.Form
Rhyolite.Frontend.WebSocket
Rhyolite.Frontend.Widget

ghc-options: -Wall -fno-warn-unused-do-bind -fwarn-tabs -funbox-strict-fields -O2 -fprof-auto-calls -rtsopts -fexpose-all-unfoldings

0 comments on commit ebd55f3

Please sign in to comment.