mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-17 20:41:49 +03:00
24592a516b
* Pass environment variables around as a data structure, via @sordina * Resolving build error * Adding Environment passing note to changelog * Removing references to ILTPollerLog as this seems to have been reintroduced from a bad merge * removing commented-out imports * Language pragmas already set by project * Linking async thread * Apply suggestions from code review Use `runQueryTx` instead of `runLazyTx` for queries. * remove the non-user facing entry in the changelog Co-authored-by: Phil Freeman <paf31@cantab.net> Co-authored-by: Phil Freeman <phil@hasura.io> Co-authored-by: Vamshi Surabhi <0x777@users.noreply.github.com>
161 lines
4.3 KiB
Haskell
161 lines
4.3 KiB
Haskell
-- | See: https://github.com/apollographql/subscriptions-transport-ws/blob/master/PROTOCOL.md
|
|
module Hasura.GraphQL.Transport.WebSocket.Protocol
|
|
( OperationId(..)
|
|
, ConnParams(..)
|
|
, StartMsg(..)
|
|
, StopMsg(..)
|
|
, ClientMsg(..)
|
|
, ServerMsg(..)
|
|
, ServerMsgType(..)
|
|
, encodeServerMsg
|
|
, serverMsgType
|
|
, DataMsg(..)
|
|
, ErrorMsg(..)
|
|
, ConnErrMsg(..)
|
|
, CompletionMsg(..)
|
|
) where
|
|
|
|
import qualified Data.Aeson as J
|
|
import qualified Data.Aeson.Casing as J
|
|
import qualified Data.Aeson.TH as J
|
|
import qualified Data.ByteString.Lazy as BL
|
|
import qualified Data.HashMap.Strict as Map
|
|
|
|
import Hasura.EncJSON
|
|
import Hasura.GraphQL.Transport.HTTP.Protocol
|
|
import Hasura.Prelude
|
|
|
|
-- | These come from the client and are websocket connection-local.
|
|
newtype OperationId
|
|
= OperationId { unOperationId :: Text }
|
|
deriving (Show, Eq, J.ToJSON, J.FromJSON, Hashable)
|
|
|
|
data StartMsg
|
|
= StartMsg
|
|
{ _smId :: !OperationId
|
|
, _smPayload :: !GQLReqUnparsed
|
|
} deriving (Show, Eq)
|
|
$(J.deriveJSON (J.aesonDrop 3 J.snakeCase) ''StartMsg)
|
|
|
|
data StopMsg
|
|
= StopMsg
|
|
{ _stId :: OperationId
|
|
} deriving (Show, Eq)
|
|
$(J.deriveJSON (J.aesonDrop 3 J.snakeCase) ''StopMsg)
|
|
|
|
data ClientMsg
|
|
= CMConnInit !(Maybe ConnParams)
|
|
| CMStart !StartMsg
|
|
| CMStop !StopMsg
|
|
| CMConnTerm
|
|
deriving (Show, Eq)
|
|
|
|
data ConnParams
|
|
= ConnParams
|
|
{ _cpHeaders :: Maybe (Map.HashMap Text Text)
|
|
} deriving (Show, Eq)
|
|
$(J.deriveJSON (J.aesonDrop 3 J.snakeCase) ''ConnParams)
|
|
|
|
instance J.FromJSON ClientMsg where
|
|
parseJSON = J.withObject "ClientMessage" $ \obj -> do
|
|
t <- obj J..: "type"
|
|
case t of
|
|
"connection_init" -> CMConnInit <$> obj J..:? "payload"
|
|
"start" -> CMStart <$> J.parseJSON (J.Object obj)
|
|
"stop" -> CMStop <$> J.parseJSON (J.Object obj)
|
|
"connection_terminate" -> return CMConnTerm
|
|
_ -> fail $ "unexpected type for ClientMessage: " <> t
|
|
|
|
-- server to client messages
|
|
data DataMsg
|
|
= DataMsg
|
|
{ _dmId :: !OperationId
|
|
, _dmPayload :: !GraphqlResponse
|
|
}
|
|
|
|
data ErrorMsg
|
|
= ErrorMsg
|
|
{ _emId :: !OperationId
|
|
, _emPayload :: !J.Value
|
|
} deriving (Show, Eq)
|
|
|
|
newtype CompletionMsg
|
|
= CompletionMsg { unCompletionMsg :: OperationId }
|
|
deriving (Show, Eq)
|
|
|
|
newtype ConnErrMsg
|
|
= ConnErrMsg { unConnErrMsg :: Text }
|
|
deriving (Show, Eq, J.ToJSON, J.FromJSON, IsString)
|
|
|
|
data ServerMsg
|
|
= SMConnAck
|
|
| SMConnKeepAlive
|
|
| SMConnErr !ConnErrMsg
|
|
| SMData !DataMsg
|
|
| SMErr !ErrorMsg
|
|
| SMComplete !CompletionMsg
|
|
|
|
data ServerMsgType
|
|
= SMT_GQL_CONNECTION_ACK
|
|
| SMT_GQL_CONNECTION_KEEP_ALIVE
|
|
| SMT_GQL_CONNECTION_ERROR
|
|
| SMT_GQL_DATA
|
|
| SMT_GQL_ERROR
|
|
| SMT_GQL_COMPLETE
|
|
deriving (Eq)
|
|
|
|
instance Show ServerMsgType where
|
|
show = \case
|
|
SMT_GQL_CONNECTION_ACK -> "connection_ack"
|
|
SMT_GQL_CONNECTION_KEEP_ALIVE -> "ka"
|
|
SMT_GQL_CONNECTION_ERROR -> "connection_error"
|
|
SMT_GQL_DATA -> "data"
|
|
SMT_GQL_ERROR -> "error"
|
|
SMT_GQL_COMPLETE -> "complete"
|
|
|
|
instance J.ToJSON ServerMsgType where
|
|
toJSON = J.toJSON . show
|
|
|
|
serverMsgType :: ServerMsg -> ServerMsgType
|
|
serverMsgType SMConnAck = SMT_GQL_CONNECTION_ACK
|
|
serverMsgType SMConnKeepAlive = SMT_GQL_CONNECTION_KEEP_ALIVE
|
|
serverMsgType (SMConnErr _) = SMT_GQL_CONNECTION_ERROR
|
|
serverMsgType (SMData _) = SMT_GQL_DATA
|
|
serverMsgType (SMErr _) = SMT_GQL_ERROR
|
|
serverMsgType (SMComplete _) = SMT_GQL_COMPLETE
|
|
|
|
encodeServerMsg :: ServerMsg -> BL.ByteString
|
|
encodeServerMsg msg =
|
|
encJToLBS $ encJFromAssocList $ case msg of
|
|
|
|
SMConnAck ->
|
|
[encTy SMT_GQL_CONNECTION_ACK]
|
|
|
|
SMConnKeepAlive ->
|
|
[encTy SMT_GQL_CONNECTION_KEEP_ALIVE]
|
|
|
|
SMConnErr connErr ->
|
|
[ encTy SMT_GQL_CONNECTION_ERROR
|
|
, ("payload", encJFromJValue connErr)
|
|
]
|
|
|
|
SMData (DataMsg opId payload) ->
|
|
[ encTy SMT_GQL_DATA
|
|
, ("id", encJFromJValue opId)
|
|
, ("payload", encodeGraphqlResponse payload)
|
|
]
|
|
|
|
SMErr (ErrorMsg opId payload) ->
|
|
[ encTy SMT_GQL_ERROR
|
|
, ("id", encJFromJValue opId)
|
|
, ("payload", encJFromJValue payload)
|
|
]
|
|
|
|
SMComplete compMsg ->
|
|
[ encTy SMT_GQL_COMPLETE
|
|
, ("id", encJFromJValue $ unCompletionMsg compMsg)
|
|
]
|
|
|
|
where
|
|
encTy ty = ("type", encJFromJValue ty)
|