Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Client support for server-sent events (SSE) #1317

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

vapourismo
Copy link

This PR is an attempt to implement SSE for the Servant client-side.


Small example

import qualified Data.Aeson                           as Aeson
import           Network.HTTP.Client.TLS
import           Servant.API
import           Servant.Client.Core.ServerSentEvents (JsonEventStreamT (..))
import           Servant.Client.Streaming
import           Servant.Types.SourceT                (foreachStep)

data NewsEntry

instance Show NewsEntry

instance Aeson.FromJSON NewsEntry

type NewsAPI = "news-stream" :> ServerSentEvents 'JsonEvent [NewsEntry]

newsAPI :: ClientM (JsonEventStreamT IO [NewsEntry])
newsAPI = client (Proxy @NewsAPI)

main :: IO ()
main = do
  mgr <- newTlsManager
  let env = mkClientEnv mgr $ BaseUrl Https "api.example.com" 443 "stable"
  withClientM newsAPI env $ either print $
    foreachStep fail print . unJsonEventStreamT

vapourismo added a commit to vapourismo/servant that referenced this pull request Jul 1, 2020
vapourismo added a commit to vapourismo/servant that referenced this pull request Jul 2, 2020
@vapourismo vapourismo force-pushed the feature/server-sent-events branch from 432cbe0 to 9dd371b Compare July 2, 2020 16:38
@vapourismo
Copy link
Author

Ping @haskell-servant/maintainers

@fisx
Copy link
Member

fisx commented Jul 16, 2020

Thanks for the PR / sorry for the low bandwidth!

I do not speak for all maintainers, but I think our priorities are #1312, then #1314, then release, then see what's up next, so it may take a while before this gets the attention it deserves.

Is there any reason why this should go to servant-client-core, as opposed to a newly created servant-client-sse? Then you could maintain this yourself until this PR gets handled.

Other maintainers: any thoughts? @vapourismo would you be open to joining us?

@vapourismo
Copy link
Author

Thanks for the update. I don’t need this merged right away - so no pressure.
I pinged the maintainers in case it slipped through the cracks.

vapourismo added a commit to vapourismo/servant that referenced this pull request Jul 20, 2020
@vapourismo vapourismo force-pushed the feature/server-sent-events branch from 9dd371b to 17ae207 Compare July 20, 2020 19:51
@hasufell
Copy link
Contributor

I don't see anything blocking this. Why has this not moved forward?

I rebased this branch against master here: https://github.com/hasufell/servant/tree/feature/server-sent-events

@tchoutri
Copy link
Contributor

Yes, ping @haskell-servant/maintainers on this. I personally don't see anything shocking.

@vapourismo vapourismo force-pushed the feature/server-sent-events branch from 17ae207 to 86d0e16 Compare February 22, 2022 18:43
@BlastWind
Copy link

BlastWind commented Jul 24, 2023

@vapourismo

Hey Ole, I am running into some trouble trying to use your great work here and would appreciate it if you can point me in the right direction.

My server: I have a server running with an SSE endpoint that spits back strings of DummyObj 3 times, spaced out by 1 second.
My client: I pasted your commit in my code which uses the latest Servant. My code is practically the same as your NewsEntry example, except with the DummyObj object.

My JsonEventStreamT IO DummyObj seems to be only spitting back S.Effect, and never S.Yield. I first slapped traceShow to mimeUnrender, which shows that the client in fact ingest a ByteString representing what I want from the server. I then wanted to find out when this ByteString is being converted into a DummyObj, but when I put a traceShow on DummyObj's parseJSON, it revealed that parseJSON was enver called, is this expected?

Here's my client main function:

main :: IO ()
main = do
  mgr <- newTlsManager
  let env = mkClientEnv mgr $ BaseUrl Http "localhost" 8000 ""
  withClientM cli env $ \case
    Left err -> (print err)
    Right stream -> do 
       putStrLn "operating on stream obj"
       let sourceT = unJsonEventStreamT stream
       S.unSourceT sourceT go
        where 
          go :: StepT IO (Event DummyObj) -> IO ()
          go S.Stop = return ()
          go (S.Error err) = do 
            print "error"
            print err
          go (S.Skip s) = do 
            print "skip"
            go s
          go (S.Effect ms) = do
            print "effect"
            result <- ms
            go result
          go (S.Yield a s) = do 
            putStrLn $ show "yield: " ++ show a 
            hFlush stdout
            go s

I had to modify your code a bit because it wasn't typechecking against the latest version of Servant. Perhaps something incorrect jump out at you in this little snippet already?
I am pattern matching on StepT directly so to get more log messages. Unfortunately, I only get multiple effect logs and then an error log: "Error in $: not enough input"

@vapourismo
Copy link
Author

That seems to suggest that the JSON could not be parsed. You'll only get the S.Yield events when the message data body could actually be parsed.

Could it be that the data messages sent by the server are incomplete or are transferred in a chunked way such that you need to stitch multiple messages together?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants