2023-05-22 02:11:40 +02:00
|
|
|
module App.AuthenticationForm where
|
|
|
|
|
2023-05-23 01:15:23 +02:00
|
|
|
import Prelude (Unit, Void, bind, discard, map, otherwise, pure, show, void, when, ($), (&&), (-), (<), (<$>), (<<<), (<>), (>=>), (>>=))
|
2023-05-22 02:11:40 +02:00
|
|
|
|
2023-06-08 21:51:12 +02:00
|
|
|
import Bulma as Bulma
|
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
import Control.Monad.Except (runExcept)
|
|
|
|
import Control.Monad.State (class MonadState)
|
|
|
|
import Data.Array as A
|
2023-05-23 01:15:23 +02:00
|
|
|
import Data.Tuple (Tuple(..))
|
2023-05-22 02:11:40 +02:00
|
|
|
import Data.Bifunctor (lmap)
|
|
|
|
import Data.Const (Const)
|
|
|
|
import Data.Either (Either(..))
|
|
|
|
import Data.Maybe (Maybe(..), isJust, isNothing, maybe)
|
|
|
|
import Data.String as String
|
|
|
|
import Effect (Effect)
|
|
|
|
import Effect.Aff.Class (class MonadAff)
|
|
|
|
import Foreign (Foreign)
|
|
|
|
import Foreign as F
|
|
|
|
import Halogen as H
|
|
|
|
import Halogen.HTML as HH
|
|
|
|
import Halogen.HTML.Events as HE
|
|
|
|
import Halogen.HTML.Properties as HP
|
|
|
|
import Halogen.Query.Event as HQE
|
|
|
|
import Halogen.Subscription as HS
|
|
|
|
import Web.Event.Event (Event)
|
|
|
|
import Web.Event.Event as Event
|
|
|
|
import Web.Socket.Event.CloseEvent as WSCE
|
|
|
|
import Web.Socket.Event.EventTypes as WSET
|
|
|
|
import Web.Socket.Event.MessageEvent as WSME
|
|
|
|
import Web.Socket.ReadyState (ReadyState(Connecting, Open, Closing, Closed))
|
|
|
|
import Web.Socket.WebSocket as WS
|
|
|
|
|
2023-05-23 01:15:23 +02:00
|
|
|
import Effect.Class (class MonadEffect)
|
|
|
|
|
|
|
|
import App.IPC as IPC
|
2023-06-03 01:53:58 +02:00
|
|
|
import App.Email as Email
|
2023-05-23 01:15:23 +02:00
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
import App.Messages.AuthenticationDaemon as AuthD
|
|
|
|
|
|
|
|
import Data.ArrayBuffer.Types (ArrayBuffer)
|
|
|
|
import Web.Socket.BinaryType (BinaryType(ArrayBuffer))
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- WebSocketEvent type
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
data WebSocketEvent :: Type -> Type
|
|
|
|
data WebSocketEvent msg
|
|
|
|
= WebSocketMessage { message :: msg, origin :: String, lastEventId :: String }
|
|
|
|
| WebSocketOpen
|
|
|
|
| WebSocketClose { code :: Int, reason :: String, wasClean :: Boolean }
|
|
|
|
| WebSocketError ErrorType
|
|
|
|
|
|
|
|
webSocketEmitter :: WS.WebSocket -> HS.Emitter (WebSocketEvent WebSocketMessageType)
|
|
|
|
webSocketEmitter socket = do
|
|
|
|
|
|
|
|
HS.makeEmitter \push -> do
|
|
|
|
|
|
|
|
openId <- HS.subscribe openEmitter push
|
|
|
|
errorId <- HS.subscribe errorEmitter push
|
|
|
|
closeId <- HS.subscribe closeEmitter push
|
|
|
|
messageId <- HS.subscribe messageEmitter push
|
|
|
|
|
|
|
|
pure do
|
|
|
|
HS.unsubscribe openId
|
|
|
|
HS.unsubscribe errorId
|
|
|
|
HS.unsubscribe closeId
|
|
|
|
HS.unsubscribe messageId
|
|
|
|
|
|
|
|
where
|
|
|
|
target = WS.toEventTarget socket
|
|
|
|
|
|
|
|
openEmitter :: HS.Emitter (WebSocketEvent WebSocketMessageType)
|
|
|
|
openEmitter =
|
|
|
|
HQE.eventListener WSET.onOpen target \_ ->
|
|
|
|
Just WebSocketOpen
|
|
|
|
|
|
|
|
errorEmitter :: HS.Emitter (WebSocketEvent WebSocketMessageType)
|
|
|
|
errorEmitter =
|
|
|
|
HQE.eventListener WSET.onError target \_ ->
|
|
|
|
Just (WebSocketError UnknownWebSocketError)
|
|
|
|
|
|
|
|
closeEmitter :: HS.Emitter (WebSocketEvent WebSocketMessageType)
|
|
|
|
closeEmitter =
|
|
|
|
HQE.eventListener WSET.onClose target \event ->
|
|
|
|
WSCE.fromEvent event >>= \closeEvent ->
|
|
|
|
Just $ WebSocketClose { code: WSCE.code closeEvent
|
|
|
|
, reason: WSCE.reason closeEvent
|
|
|
|
, wasClean: WSCE.wasClean closeEvent
|
|
|
|
}
|
|
|
|
|
|
|
|
messageEmitter :: HS.Emitter (WebSocketEvent WebSocketMessageType)
|
|
|
|
messageEmitter = HQE.eventListener WSET.onMessage target (WSME.fromEvent >=> decodeMessageEvent)
|
|
|
|
|
|
|
|
decodeMessageEvent :: WSME.MessageEvent -> Maybe (WebSocketEvent WebSocketMessageType)
|
|
|
|
decodeMessageEvent = \msgEvent -> do
|
|
|
|
let
|
|
|
|
foreign' :: Foreign
|
|
|
|
foreign' = WSME.data_ msgEvent
|
|
|
|
case foreignToArrayBuffer foreign' of
|
|
|
|
Left errs -> pure $ WebSocketError $ UnknownError errs
|
|
|
|
Right arrayBuffer -> pure $ WebSocketMessage { message: arrayBuffer, origin: WSME.origin msgEvent, lastEventId: WSME.lastEventId msgEvent }
|
|
|
|
|
|
|
|
---------------------------
|
|
|
|
-- Errors
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
data ErrorType
|
2023-05-30 16:42:52 +02:00
|
|
|
= UnknownError String
|
2023-05-22 02:11:40 +02:00
|
|
|
| UnknownWebSocketError
|
|
|
|
|
|
|
|
renderError :: ErrorType -> String
|
|
|
|
renderError = case _ of
|
|
|
|
UnknownError str ->
|
|
|
|
"Unknown error: " <> str
|
|
|
|
UnknownWebSocketError ->
|
|
|
|
"Unknown 'error' event has been fired by WebSocket event listener"
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- WebSocket message type
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
type WebSocketMessageType = ArrayBuffer
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- Root component module
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
2023-06-08 21:51:12 +02:00
|
|
|
data Output = AuthToken (Tuple Int String)
|
2023-06-01 03:20:53 +02:00
|
|
|
type Slot = H.Slot Query Output
|
2023-05-22 16:23:21 +02:00
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
type Query :: forall k. k -> Type
|
|
|
|
type Query = Const Void
|
|
|
|
type Input = String
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
data AuthenticationInput
|
|
|
|
= AUTH_INP_login String
|
|
|
|
| AUTH_INP_pass String
|
|
|
|
|
|
|
|
data RegisterInput
|
|
|
|
= REG_INP_login String
|
|
|
|
| REG_INP_email String
|
|
|
|
| REG_INP_pass String
|
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
data Action
|
|
|
|
= Initialize
|
|
|
|
| WebSocketParseError String
|
|
|
|
| ConnectWebSocket
|
2023-06-03 00:54:18 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
| HandleAuthenticationInput AuthenticationInput
|
|
|
|
| HandleRegisterInput RegisterInput
|
2023-06-03 03:50:54 +02:00
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
| AuthenticationAttempt Event
|
2023-06-03 00:54:18 +02:00
|
|
|
| RegisterAttempt Event
|
2023-06-02 00:53:01 +02:00
|
|
|
| Finalize
|
2023-05-22 02:11:40 +02:00
|
|
|
| HandleWebSocket (WebSocketEvent WebSocketMessageType)
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
type StateAuthenticationForm = { login :: String, pass :: String }
|
|
|
|
type StateRegistrationForm = { login :: String, email :: String, pass :: String }
|
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
type State =
|
2023-06-03 00:54:18 +02:00
|
|
|
{ messages :: Array String
|
|
|
|
, messageHistoryLength :: Int
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
, authenticationForm :: StateAuthenticationForm
|
|
|
|
, registrationForm :: StateRegistrationForm
|
2023-06-03 03:50:54 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
-- TODO: put network stuff in a record.
|
2023-06-03 00:54:18 +02:00
|
|
|
, wsUrl :: String
|
|
|
|
, wsConnection :: Maybe WS.WebSocket
|
|
|
|
, canReconnect :: Boolean
|
2023-05-22 02:11:40 +02:00
|
|
|
}
|
|
|
|
|
2023-05-22 16:23:21 +02:00
|
|
|
component :: forall m. MonadAff m => H.Component Query Input Output m
|
|
|
|
component =
|
2023-05-22 02:11:40 +02:00
|
|
|
H.mkComponent
|
|
|
|
{ initialState
|
|
|
|
, render
|
|
|
|
, eval: H.mkEval $ H.defaultEval
|
|
|
|
{ initialize = Just Initialize
|
|
|
|
, handleAction = handleAction
|
2023-06-02 00:53:01 +02:00
|
|
|
, finalize = Just Finalize
|
2023-05-22 02:11:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
initialState :: Input -> State
|
|
|
|
initialState input =
|
|
|
|
{ messages: []
|
|
|
|
, messageHistoryLength: 10
|
2023-06-03 00:54:18 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
, authenticationForm: { login: "", pass: "" }
|
|
|
|
, registrationForm: { login: "", email: "", pass: "" }
|
2023-06-03 00:54:18 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
-- TODO: put network stuff in a record.
|
2023-05-22 02:11:40 +02:00
|
|
|
, wsUrl: input
|
|
|
|
, wsConnection: Nothing
|
|
|
|
, canReconnect: false
|
|
|
|
}
|
|
|
|
|
|
|
|
render :: forall m. State -> H.ComponentHTML Action () m
|
2023-06-03 00:54:18 +02:00
|
|
|
render {
|
|
|
|
messages,
|
|
|
|
wsConnection,
|
|
|
|
canReconnect,
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
authenticationForm,
|
2023-06-08 18:13:59 +02:00
|
|
|
registrationForm }
|
2023-06-08 21:51:12 +02:00
|
|
|
= HH.div_
|
|
|
|
[ Bulma.columns_ [ Bulma.column_ auth_form, Bulma.column_ register_form ]
|
2023-06-03 00:54:18 +02:00
|
|
|
, render_messages
|
|
|
|
, renderReconnectButton (isNothing wsConnection && canReconnect)
|
|
|
|
]
|
|
|
|
where
|
2023-06-02 20:05:03 +02:00
|
|
|
|
2023-06-08 21:51:12 +02:00
|
|
|
auth_form
|
|
|
|
= [ Bulma.h3 "Authentication"
|
|
|
|
, render_auth_form
|
|
|
|
]
|
|
|
|
|
|
|
|
register_form
|
|
|
|
= [ Bulma.h3 "Register!"
|
|
|
|
, render_register_form
|
|
|
|
]
|
|
|
|
|
2023-06-03 00:54:18 +02:00
|
|
|
render_auth_form = HH.form
|
|
|
|
[ HE.onSubmit AuthenticationAttempt ]
|
2023-06-09 00:28:03 +02:00
|
|
|
[ Bulma.box_input "Login" "login" -- title, placeholder
|
|
|
|
(HandleAuthenticationInput <<< AUTH_INP_login) -- action
|
|
|
|
authenticationForm.login -- value
|
|
|
|
true -- validity (TODO)
|
|
|
|
(maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection) -- condition
|
|
|
|
, Bulma.box_password "Password" "password" -- title, placeholder
|
|
|
|
(HandleAuthenticationInput <<< AUTH_INP_pass) -- action
|
|
|
|
authenticationForm.pass -- value
|
|
|
|
true -- validity (TODO)
|
|
|
|
(maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection) -- condition
|
2023-06-09 01:40:31 +02:00
|
|
|
, HH.button
|
|
|
|
[ HP.style "padding: 0.5rem 1.25rem;"
|
|
|
|
, HP.type_ HP.ButtonSubmit
|
|
|
|
, maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection
|
|
|
|
]
|
|
|
|
[ HH.text "Send Message to Server" ]
|
2023-05-22 02:11:40 +02:00
|
|
|
]
|
2023-06-03 00:54:18 +02:00
|
|
|
|
|
|
|
render_register_form = HH.form
|
|
|
|
[ HE.onSubmit RegisterAttempt ]
|
2023-06-09 01:40:31 +02:00
|
|
|
[ Bulma.box_input "Login" "login" -- title, placeholder
|
|
|
|
(HandleRegisterInput <<< REG_INP_login) -- action
|
|
|
|
registrationForm.login -- value
|
|
|
|
true -- validity (TODO)
|
|
|
|
(maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection) -- condition
|
|
|
|
, Bulma.box_input "Email" "email@example.com" -- title, placeholder
|
|
|
|
(HandleRegisterInput <<< REG_INP_email) -- action
|
|
|
|
registrationForm.email -- value
|
|
|
|
true -- validity (TODO)
|
|
|
|
(maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection) -- condition
|
|
|
|
, Bulma.box_password "Password" "password" -- title, placeholder
|
|
|
|
(HandleRegisterInput <<< REG_INP_pass) -- action
|
|
|
|
registrationForm.pass -- value
|
|
|
|
true -- validity (TODO)
|
|
|
|
(maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection) -- condition
|
|
|
|
, HH.div_
|
|
|
|
[ HH.button
|
|
|
|
[ HP.style "padding: 0.5rem 1.25rem;"
|
|
|
|
, HP.type_ HP.ButtonSubmit
|
|
|
|
, maybe (HP.disabled true) (\_ -> HP.enabled true) wsConnection
|
2023-06-03 00:54:18 +02:00
|
|
|
]
|
2023-06-09 01:40:31 +02:00
|
|
|
[ HH.text "Send Message to Server" ]
|
2023-06-03 00:54:18 +02:00
|
|
|
]
|
2023-05-22 02:11:40 +02:00
|
|
|
]
|
|
|
|
|
2023-06-03 00:54:18 +02:00
|
|
|
render_messages = HH.ul_ $ map (\msg -> HH.li_ [ HH.text msg ]) messages
|
|
|
|
|
|
|
|
renderFootnote :: String -> H.ComponentHTML Action () m
|
|
|
|
renderFootnote txt =
|
|
|
|
HH.div [ HP.style "margin-bottom: 0.125rem; color: grey;" ] [ HH.small_ [ HH.text txt ] ]
|
|
|
|
|
|
|
|
renderReconnectButton :: Boolean -> H.ComponentHTML Action () m
|
|
|
|
renderReconnectButton cond =
|
|
|
|
if cond
|
|
|
|
then
|
|
|
|
HH.p_
|
|
|
|
[ HH.button
|
|
|
|
[ HP.type_ HP.ButtonButton
|
|
|
|
, HE.onClick \_ -> ConnectWebSocket
|
|
|
|
]
|
|
|
|
[ HH.text "Reconnect?" ]
|
|
|
|
]
|
|
|
|
else
|
|
|
|
HH.p_
|
|
|
|
[ renderFootnote "NOTE: A 'Reconnect?' button will appear if the connection drops"
|
|
|
|
]
|
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
handleAction :: forall m. MonadAff m => Action -> H.HalogenM State Action () Output m Unit
|
|
|
|
handleAction = case _ of
|
|
|
|
Initialize ->
|
|
|
|
handleAction ConnectWebSocket
|
|
|
|
|
2023-06-02 00:53:01 +02:00
|
|
|
Finalize -> do
|
|
|
|
{ wsConnection } <- H.get
|
|
|
|
systemMessage "Finalize"
|
|
|
|
case wsConnection of
|
|
|
|
Nothing -> systemMessage "No socket? How is that even possible?"
|
|
|
|
Just socket -> H.liftEffect $ WS.close socket
|
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
WebSocketParseError error ->
|
|
|
|
systemMessage $ renderError (UnknownError error)
|
|
|
|
|
|
|
|
ConnectWebSocket -> do
|
|
|
|
{ wsUrl } <- H.get
|
|
|
|
systemMessage ("Connecting to \"" <> wsUrl <> "\"...")
|
|
|
|
webSocket <- H.liftEffect $ WS.create wsUrl []
|
|
|
|
H.liftEffect $ WS.setBinaryType webSocket ArrayBuffer
|
|
|
|
H.modify_ _ { wsConnection = Just webSocket }
|
|
|
|
void $ H.subscribe (HandleWebSocket <$> webSocketEmitter webSocket)
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
HandleAuthenticationInput authinp -> do
|
|
|
|
case authinp of
|
|
|
|
AUTH_INP_login v -> H.modify_ _ { authenticationForm { login = v } }
|
|
|
|
AUTH_INP_pass v -> H.modify_ _ { authenticationForm { pass = v } }
|
|
|
|
|
|
|
|
HandleRegisterInput reginp -> do
|
|
|
|
case reginp of
|
|
|
|
REG_INP_login v -> H.modify_ _ { registrationForm { login = v } }
|
|
|
|
REG_INP_email v -> H.modify_ _ { registrationForm { email = v } }
|
|
|
|
REG_INP_pass v -> H.modify_ _ { registrationForm { pass = v } }
|
|
|
|
|
2023-06-03 00:54:18 +02:00
|
|
|
RegisterAttempt ev -> do
|
|
|
|
H.liftEffect $ Event.preventDefault ev
|
2023-06-03 01:53:58 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
{ wsConnection, registrationForm } <- H.get
|
|
|
|
let login = registrationForm.login
|
|
|
|
email = registrationForm.email
|
|
|
|
pass = registrationForm.pass
|
2023-06-03 01:53:58 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
case wsConnection, login, email, pass of
|
2023-06-03 01:53:58 +02:00
|
|
|
Nothing, _, _, _ ->
|
|
|
|
unableToSend "Not connected to server."
|
|
|
|
|
|
|
|
Just _, "", _, _ ->
|
|
|
|
unableToSend "Write your login!"
|
|
|
|
|
|
|
|
Just _, _, "", _ ->
|
|
|
|
unableToSend "Write your email!"
|
|
|
|
|
|
|
|
Just _, _, _, "" ->
|
|
|
|
unableToSend "Write your password!"
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
Just webSocket, _, _, _ -> do
|
2023-06-03 01:53:58 +02:00
|
|
|
H.liftEffect (WS.readyState webSocket) >>= case _ of
|
|
|
|
Connecting ->
|
|
|
|
unableToSend "Still connecting to server."
|
|
|
|
|
|
|
|
Closing ->
|
|
|
|
unableToSend "Connection to server is closing."
|
|
|
|
|
|
|
|
Closed -> do
|
|
|
|
unableToSend "Connection to server has been closed."
|
|
|
|
maybeCurrentConnection <- H.gets _.wsConnection
|
|
|
|
when (isJust maybeCurrentConnection) do
|
|
|
|
H.modify_ _ { wsConnection = Nothing, canReconnect = true }
|
|
|
|
|
|
|
|
Open -> do
|
|
|
|
H.liftEffect $ do
|
|
|
|
ab <- AuthD.serialize $ AuthD.MkRegister { login: login
|
|
|
|
, email: Just (Email.Email email)
|
|
|
|
, password: pass
|
|
|
|
, phone: Nothing}
|
|
|
|
sendArrayBuffer webSocket ab
|
|
|
|
appendMessageReset "[😇] Trying to register"
|
2023-06-03 00:54:18 +02:00
|
|
|
|
2023-05-22 02:11:40 +02:00
|
|
|
AuthenticationAttempt ev -> do
|
|
|
|
H.liftEffect $ Event.preventDefault ev
|
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
{ wsConnection, authenticationForm } <- H.get
|
2023-05-22 02:11:40 +02:00
|
|
|
|
2023-06-04 01:24:05 +02:00
|
|
|
case wsConnection, authenticationForm.login, authenticationForm.pass of
|
2023-05-22 02:11:40 +02:00
|
|
|
Nothing, _, _ ->
|
|
|
|
unableToSend "Not connected to server."
|
|
|
|
|
|
|
|
Just _ , "" , _ ->
|
|
|
|
unableToSend "Write your login!"
|
|
|
|
|
|
|
|
Just _ , _ , "" ->
|
|
|
|
unableToSend "Write your password!"
|
|
|
|
|
|
|
|
Just webSocket, login, pass -> do
|
|
|
|
H.liftEffect (WS.readyState webSocket) >>= case _ of
|
|
|
|
Connecting ->
|
|
|
|
unableToSend "Still connecting to server."
|
|
|
|
|
|
|
|
Closing ->
|
|
|
|
unableToSend "Connection to server is closing."
|
|
|
|
|
|
|
|
Closed -> do
|
|
|
|
unableToSend "Connection to server has been closed."
|
|
|
|
maybeCurrentConnection <- H.gets _.wsConnection
|
|
|
|
when (isJust maybeCurrentConnection) do
|
|
|
|
H.modify_ _ { wsConnection = Nothing, canReconnect = true }
|
|
|
|
|
|
|
|
Open -> do
|
|
|
|
H.liftEffect $ do
|
2023-06-10 18:30:31 +02:00
|
|
|
ab <- AuthD.serialize (AuthD.MkLogin { login: login, password: pass })
|
2023-05-22 02:11:40 +02:00
|
|
|
sendArrayBuffer webSocket ab
|
|
|
|
appendMessageReset $ "[😇] Trying to connect with login: " <> login
|
|
|
|
|
|
|
|
HandleWebSocket wsEvent ->
|
|
|
|
case wsEvent of
|
|
|
|
WebSocketMessage messageEvent -> do
|
|
|
|
receivedMessage <- H.liftEffect $ AuthD.deserialize messageEvent.message
|
|
|
|
case receivedMessage of
|
2023-05-23 01:15:23 +02:00
|
|
|
-- Cases where we didn't understand the message.
|
|
|
|
Left err -> do
|
|
|
|
case err of
|
2023-05-25 00:07:59 +02:00
|
|
|
(AuthD.JSONERROR jerr) -> do
|
2023-05-23 01:15:23 +02:00
|
|
|
print_json_string messageEvent.message
|
2023-05-25 00:07:59 +02:00
|
|
|
handleAction $ WebSocketParseError ("JSON parsing error: " <> jerr <> " JSON is: " <> jerr)
|
2023-05-23 01:15:23 +02:00
|
|
|
(AuthD.UnknownError unerr) -> handleAction $ WebSocketParseError ("Parsing error: AuthD.UnknownError" <> (show unerr))
|
|
|
|
(AuthD.UnknownNumber ) -> handleAction $ WebSocketParseError ("Parsing error: AuthD.UnknownNumber")
|
2023-06-02 00:10:08 +02:00
|
|
|
|
2023-05-23 01:15:23 +02:00
|
|
|
-- Cases where we understood the message.
|
2023-05-22 02:11:40 +02:00
|
|
|
Right response -> do
|
|
|
|
case response of
|
2023-05-23 01:15:23 +02:00
|
|
|
-- The authentication failed.
|
2023-06-02 20:05:03 +02:00
|
|
|
(AuthD.GotError errmsg) -> do
|
|
|
|
appendMessage $ "[😈] Failed: " <> maybe "server didn't tell why" (\v -> v) errmsg.reason
|
2023-05-23 01:15:23 +02:00
|
|
|
-- The authentication was a success!
|
2023-06-02 00:10:08 +02:00
|
|
|
(AuthD.GotToken msg) -> do
|
2023-05-22 02:11:40 +02:00
|
|
|
appendMessage $ "[😈] Success! user " <> (show msg.uid) <> " has token: " <> msg.token
|
2023-06-08 21:51:12 +02:00
|
|
|
H.raise $ AuthToken (Tuple msg.uid msg.token)
|
2023-06-04 02:01:50 +02:00
|
|
|
(AuthD.GotUserAdded msg) -> do
|
|
|
|
appendMessage $ "[😈] Success! Server added user: " <> show msg.user
|
2023-05-23 01:15:23 +02:00
|
|
|
-- WTH?!
|
2023-05-22 02:11:40 +02:00
|
|
|
_ -> do
|
2023-06-03 00:54:18 +02:00
|
|
|
appendMessage $ "[😈] Failed! Authentication server didn't send a valid message."
|
2023-05-22 02:11:40 +02:00
|
|
|
|
|
|
|
WebSocketOpen -> do
|
|
|
|
{ wsUrl } <- H.get
|
|
|
|
systemMessage ("Successfully connected to WebSocket at \"" <> wsUrl <> "\"!🎉")
|
|
|
|
|
|
|
|
WebSocketClose { code, reason, wasClean } -> do
|
|
|
|
systemMessage $ renderCloseMessage code wasClean reason
|
|
|
|
maybeCurrentConnection <- H.gets _.wsConnection
|
|
|
|
when (isJust maybeCurrentConnection) do
|
|
|
|
H.modify_ _ { wsConnection = Nothing, canReconnect = true }
|
|
|
|
|
|
|
|
WebSocketError errorType ->
|
|
|
|
systemMessage $ renderError errorType
|
|
|
|
|
|
|
|
where
|
|
|
|
renderCloseMessage
|
|
|
|
:: Int
|
|
|
|
-> Boolean
|
|
|
|
-> String
|
|
|
|
-> String
|
|
|
|
renderCloseMessage code wasClean = case _ of
|
|
|
|
"" -> baseCloseMessage
|
|
|
|
reason -> baseCloseMessage <> "Reason: " <> reason
|
|
|
|
where
|
|
|
|
baseCloseMessage :: String
|
|
|
|
baseCloseMessage =
|
|
|
|
String.joinWith " "
|
|
|
|
[ "Connection to WebSocket closed"
|
|
|
|
, "[ CODE:"
|
|
|
|
, show code
|
|
|
|
, "|"
|
|
|
|
, if wasClean then "CLEAN" else "DIRTY"
|
|
|
|
, "]"
|
|
|
|
]
|
|
|
|
|
|
|
|
sendArrayBuffer :: WS.WebSocket -> ArrayBuffer -> Effect Unit
|
|
|
|
sendArrayBuffer = WS.sendArrayBuffer
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- Helpers for updating the array of messages sent/received
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- Append a new message to the chat history, with a boolean that allows you to
|
|
|
|
-- clear the text input field or not. The number of displayed `messages` in the
|
|
|
|
-- chat history (including system) is controlled by the `messageHistoryLength`
|
|
|
|
-- field in the component `State`.
|
|
|
|
appendMessageGeneric :: forall m. MonadState State m => Boolean -> String -> m Unit
|
|
|
|
appendMessageGeneric clearField msg = do
|
|
|
|
histSize <- H.gets _.messageHistoryLength
|
|
|
|
if clearField
|
|
|
|
then H.modify_ \st ->
|
2023-06-04 01:24:05 +02:00
|
|
|
st { messages = appendSingle histSize msg st.messages, authenticationForm { login = "" }}
|
2023-05-22 02:11:40 +02:00
|
|
|
else H.modify_ \st ->
|
|
|
|
st { messages = appendSingle histSize msg st.messages }
|
|
|
|
where
|
|
|
|
-- Limits the nnumber of recent messages to `maxHist`
|
|
|
|
appendSingle :: Int -> String -> Array String -> Array String
|
|
|
|
appendSingle maxHist x xs
|
|
|
|
| A.length xs < maxHist = xs `A.snoc` x
|
|
|
|
| otherwise = (A.takeEnd (maxHist-1) xs) `A.snoc` x
|
|
|
|
|
|
|
|
-- Append a new message to the chat history, while not clearing
|
|
|
|
-- the user input field
|
|
|
|
appendMessage :: forall m. MonadState State m => String -> m Unit
|
|
|
|
appendMessage = appendMessageGeneric false
|
|
|
|
|
|
|
|
-- Append a new message to the chat history and also clear
|
|
|
|
-- the user input field
|
|
|
|
appendMessageReset :: forall m. MonadState State m => String -> m Unit
|
|
|
|
appendMessageReset = appendMessageGeneric true
|
|
|
|
|
|
|
|
-- Append a system message to the chat log.
|
|
|
|
systemMessage :: forall m. MonadState State m => String -> m Unit
|
|
|
|
systemMessage msg = appendMessage ("[🤖] System: " <> msg)
|
|
|
|
|
|
|
|
-- As above, but also clears the user input field. e.g. in
|
|
|
|
-- the case of a "/disconnect" command
|
|
|
|
systemMessageReset :: forall m. MonadState State m => String -> m Unit
|
|
|
|
systemMessageReset msg = appendMessageReset ("[🤖] System: " <> msg)
|
|
|
|
|
|
|
|
-- A system message to use when a message cannot be sent.
|
|
|
|
unableToSend :: forall m. MonadState State m => String -> m Unit
|
|
|
|
unableToSend reason = systemMessage ("Unable to send. " <> reason)
|
|
|
|
|
|
|
|
foreignToArrayBuffer :: Foreign -> Either String ArrayBuffer
|
|
|
|
foreignToArrayBuffer
|
|
|
|
= lmap renderForeignErrors
|
|
|
|
<<< runExcept
|
|
|
|
<<< F.unsafeReadTagged "ArrayBuffer"
|
|
|
|
where
|
|
|
|
renderForeignErrors :: F.MultipleErrors -> String
|
|
|
|
renderForeignErrors =
|
|
|
|
String.joinWith "; " <<< A.fromFoldable <<< map F.renderForeignError
|
2023-05-23 01:15:23 +02:00
|
|
|
|
|
|
|
print_json_string :: forall m. MonadEffect m => MonadState State m => ArrayBuffer -> m Unit
|
|
|
|
print_json_string arraybuffer = do
|
|
|
|
-- fromTypedIPC :: ArrayBuffer -> Effect (Either ParseError (Tuple UInt String))
|
|
|
|
value <- H.liftEffect $ IPC.fromTypedIPC arraybuffer
|
|
|
|
appendMessage $ case (value) of
|
|
|
|
Left _ -> "Cannot even fromTypedIPC the message."
|
2023-05-23 02:50:08 +02:00
|
|
|
Right (Tuple messageTypeNumber string) -> "Number is: " <> show messageTypeNumber <> ", received string: " <> string
|