Unicode C++
Unicode C++
Firebase Receive Server-Sent Events (text/event-stream)
See more Firebase Examples
Demonstrates how to start receiving server-sent events and update your JSON database with each event.Chilkat Unicode C++ Downloads
#include <CkFileAccessW.h>
#include <CkRestW.h>
#include <CkAuthGoogleW.h>
#include <CkUrlW.h>
#include <CkJsonObjectW.h>
#include <CkStreamW.h>
#include <CkServerSentEventW.h>
#include <CkTaskW.h>
void ChilkatSample(void)
{
bool success = false;
// Demonstrates how to begin receiving server-sent events, and to update
// your JSON database for each event.
// This example requires the Chilkat API to have been previously unlocked.
// See Global Unlock Sample for sample code.
// This example assumes a JWT authentication token, if required, has been previously obtained.
// See Get Firebase Access Token from JSON Service Account Private Key for sample code.
// Load the previously obtained Firebase access token into a string.
CkFileAccessW fac;
const wchar_t *accessToken = fac.readEntireTextFile(L"qa_data/tokens/firebaseToken.txt",L"utf-8");
if (fac.get_LastMethodSuccess() == false) {
wprintf(L"%s\n",fac.lastErrorText());
return;
}
CkRestW rest;
// Make the initial connection (without sending a request yet).
// Once connected, any number of requests may be sent. It is not necessary to explicitly
// call Connect before each request.
success = rest.Connect(L"chilkat.firebaseio.com",443,true,true);
if (success == false) {
wprintf(L"%s\n",rest.lastErrorText());
return;
}
CkAuthGoogleW authGoogle;
authGoogle.put_AccessToken(accessToken);
rest.SetAuthGoogle(authGoogle);
rest.AddHeader(L"Accept",L"text/event-stream");
rest.AddHeader(L"Cache-Control",L"no-cache");
const wchar_t *responseBody = rest.fullRequestNoBody(L"GET",L"/.json");
// A 307 redirect response is expected.
if (rest.get_ResponseStatusCode() != 307) {
wprintf(L"Unexpected response code: %d\n",rest.get_ResponseStatusCode());
wprintf(L"%s\n",responseBody);
wprintf(L"Failed.\n");
return;
}
// Get the redirect URL
const wchar_t *urlStr = rest.lastRedirectUrl();
CkUrlW url;
url.ParseUrl(urlStr);
wprintf(L"redirect URL domain: %s\n",url.host());
wprintf(L"redirect URL path: %s\n",url.path());
wprintf(L"redirect URL query params: %s\n",url.query());
wprintf(L"redirect URL path with query params: %s\n",url.pathWithQueryParams());
// Our text/event-stream will be obtained from the redirect URL...
CkRestW rest2;
success = rest2.Connect(url.host(),443,true,true);
if (success != true) {
wprintf(L"%s\n",rest2.lastErrorText());
return;
}
rest2.AddHeader(L"Accept",L"text/event-stream");
rest2.AddHeader(L"Cache-Control",L"no-cache");
// Add the redirect query params to the request
rest2.AddQueryParams(url.query());
// In our case, we don't actually need the auth query param,
// so remove it.
rest2.RemoveQueryParam(L"auth");
// Send the request. (We are only sending the request here.
// We are not yet getting the response because the response
// will be a text/event-stream.)
success = rest2.SendReqNoBody(L"GET",url.path());
if (success != true) {
wprintf(L"%s\n",rest2.lastErrorText());
return;
}
// Read the response header.
// We want to first get the response header to see if it's a successful
// response status code. If not, then the response will not be a text/event-stream
// and we should read the response body normally.
int responseStatusCode = rest2.ReadResponseHeader();
if (responseStatusCode < 0) {
wprintf(L"%s\n",rest2.lastErrorText());
return;
}
// If successful, a 200 response code is expected.
// If the reponse code is not 200, then read the response body and fail..
if (responseStatusCode != 200) {
wprintf(L"Response Code: %d\n",responseStatusCode);
wprintf(L"Response Status Text: %s\n",rest2.responseStatusText());
wprintf(L"Response Header: %s\n",rest2.responseHeader());
responseBody = rest2.readRespBodyString();
if (rest2.get_LastMethodSuccess() == true) {
wprintf(L"Error Response Body: %s\n",responseBody);
}
wprintf(L"Failed.\n");
return;
}
// For this example, our JSON database will be empty at the beginning.
// The incoming events (put and patch) will be applied to this database.
CkJsonObjectW jsonDb;
// Make sure to set the JSON path delimiter to "/". The default is "." and this
// is not compatible with Firebase paths.
jsonDb.put_DelimiterChar(L"/");
// At this point, we've received the response header. Now it's time to begin
// receiving the event stream. We'll start a background thread to read the
// stream. (Our main application (foreground) thread can cancel it at any time.)
// While receiving in the background thread, our foreground thread can read the stream
// as it desires..
CkStreamW eventStream;
// This sse object will be used as a helper to parse the server-sent event stream.
CkServerSentEventW sse;
CkTaskW *task = rest2.ReadRespBodyStreamAsync(eventStream,true);
task->Run();
// For this example, we'll just read a few events, and then cancel the
// async task.
int count = 0;
while ((count < 3) && (task->get_Finished() == false)) {
// Get the next event, which is a series of text lines ending with
// a blank line.
// Note: This method blocks the calling thread until a message arrives.
// a program might instead periodically check the availability of
// data via the stream's DataAvailable property, and then do the read.
// An alternative to writing a while loop to read the event stream
// would be to setup some sort of timer event in your program (using whatever timer functionality
// is provided in a programming language/environment), to periodically check the eventStream's
// DataAvailable property and consume the incoming event.
const wchar_t *eventStr = eventStream.readUntilMatch(L"\r\n\r\n");
if (eventStream.get_LastMethodSuccess() != true) {
wprintf(L"%s\n",eventStream.lastErrorText());
// Force the loop to exit by setting the count to a high number.
count = 99999;
}
else {
wprintf(L"Event: [%s]\n",eventStr);
// We have an event. Let's update our local copy of the JSON database.
success = sse.LoadEvent(eventStr);
if (success != true) {
wprintf(L"Failed to load sse event: %s\n",eventStr);
}
else {
// Now we can easily access the event name and data, and apply it to our JSON database:
success = jsonDb.FirebaseApplyEvent(sse.eventName(),sse.data());
if (success != true) {
wprintf(L"Failed to apply event: %s: %s\n",sse.eventName(),sse.data());
}
else {
wprintf(L"Successfully applied event: %s: %s\n",sse.eventName(),sse.data());
}
}
}
count = count + 1;
}
// Make sure the background task is cancelled if still running.
task->Cancel();
delete task;
// Examine the JSON database after applying events..
jsonDb.put_EmitCompact(false);
wprintf(L"----\n");
wprintf(L"%s\n",jsonDb.emit());
}