graphql-engine/server/src-lib/Hasura/GraphQL/Transport/WebSocket/Protocol.hs
Lyndon Maydwell 24592a516b
Pass environment variables around as a data structure, via @sordina (#5374)
* 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>
2020-07-14 12:00:58 -07:00

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)