From 2486e124c1985a5a70bd63590da61a73a5a2e715 Mon Sep 17 00:00:00 2001 From: Sergey Linev <S.Linev@gsi.de> Date: Tue, 18 Apr 2017 15:05:37 +0200 Subject: [PATCH] http: provide lonpoll handler similar to websocket It uses open requests to send data directly from server. For the moment buffers queue depth=1, but can be easily improved --- net/http/inc/THttpEngine.h | 4 ++- net/http/src/THttpEngine.cxx | 17 +++++----- net/http/src/THttpServer.cxx | 65 +++++++++++++++++++++++++++++++----- 3 files changed, 68 insertions(+), 18 deletions(-) diff --git a/net/http/inc/THttpEngine.h b/net/http/inc/THttpEngine.h index 3afaa668217..efeecb1f034 100644 --- a/net/http/inc/THttpEngine.h +++ b/net/http/inc/THttpEngine.h @@ -68,10 +68,12 @@ public: virtual void SendCharStar(const char *str); - virtual void ProcessData(THttpCallArg *arg); + virtual Bool_t PreviewData(THttpCallArg *) { return kTRUE; } // --------- method to work with Canvas (temporary solution) + virtual void ProcessData(THttpCallArg *arg); + virtual void AssignCanvas(TCanvas *canv); virtual void CanvasModified(); diff --git a/net/http/src/THttpEngine.cxx b/net/http/src/THttpEngine.cxx index 7dc6901cd98..d0a637afe9e 100644 --- a/net/http/src/THttpEngine.cxx +++ b/net/http/src/THttpEngine.cxx @@ -66,6 +66,15 @@ THttpWSEngine::~THttpWSEngine() AssignCanvas(0); } +//////////////////////////////////////////////////////////////////////////////// +/// Envelope for sending string via the websocket + +void THttpWSEngine::SendCharStar(const char *str) +{ + if (str) Send(str, strlen(str)); +} + + //////////////////////////////////////////////////////////////////////////////// /// react on canvas modifications @@ -190,14 +199,6 @@ void THttpWSEngine::ProcessData(THttpCallArg *arg) } } -//////////////////////////////////////////////////////////////////////////////// -/// Envelope for sending string via the websocket - -void THttpWSEngine::SendCharStar(const char *str) -{ - if (str) Send(str, strlen(str)); -} - //////////////////////////////////////////////////////////////////////////////// /// assign canvas to the web socket /// connects with CanvasModified signal diff --git a/net/http/src/THttpServer.cxx b/net/http/src/THttpServer.cxx index b355af83fa5..20d090425cc 100644 --- a/net/http/src/THttpServer.cxx +++ b/net/http/src/THttpServer.cxx @@ -70,11 +70,12 @@ public: class TLongPollEngine : public THttpWSEngine { protected: - THttpCallArg *fPoll; ///< polling connection, which can be used for the sending next operation + THttpCallArg *fPoll; ///< polling request, which can be used for the next sending + TString fBuf; ///< single entry to keep data which is not yet send to the client public: TLongPollEngine(const char *name, const char *title) - : THttpWSEngine(name, title), fPoll(0) + : THttpWSEngine(name, title), fPoll(0), fBuf() { } @@ -82,14 +83,53 @@ public: virtual UInt_t GetId() const { return TString::Hash((void *)this, sizeof(void *)); } - virtual void ClearHandle() { fPoll = 0; } + virtual void ClearHandle() + { + if (fPoll) { fPoll->Set404(); fPoll->NotifyCondition(); fPoll = 0; } + } - virtual void Send(const void *buf, int len) + virtual void Send(const void * /*buf*/, int /*len*/) { + Error("TLongPollEngine::Send", "Should never be called, only text is supported"); } virtual void SendCharStar(const char *buf) { + if (fPoll) { + fPoll->SetContentType("text/plain"); + fPoll->SetContent(buf); + fPoll->NotifyCondition(); + fPoll = 0; + } else + if (fBuf.Length() == 0) { + fBuf = buf; + } else { + Error("TLongPollEngine::SendCharStar", "Too many send operations, use TList object instead"); + } + } + + virtual Bool_t PreviewData(THttpCallArg *arg) + { + // function called in the user code before processing correspondent websocket data + + if (fPoll) { + // if there are pending request, reply it immediately + fPoll->SetContentType("text/plain"); + fPoll->SetContent(""); + fPoll->NotifyCondition(); + fPoll = 0; + } + + if (fBuf.Length() > 0) { + arg->SetContentType("text/plain"); + arg->SetContent(fBuf.Data()); + fBuf = ""; + } else { + arg->SetPostponed(); + fPoll = arg; + } + + return kTRUE; } }; @@ -518,7 +558,7 @@ void THttpServer::ProcessRequests() fSniffer->SetCurrentCallArg(0); } - arg->fCond.notify_one(); + arg->NotifyCondition(); } // regularly call Process() method of engine to let perform actions in ROOT context @@ -722,7 +762,7 @@ void THttpServer::ProcessRequest(THttpCallArg *arg) // try to emulate websocket connect // if accepted, reply with connection id, which must be used in the following communications arg->SetMethod("WS_CONNECT"); - if (canv->GetCanvasImp()->ProcessWSRequest(arg) && !arg->Is404()) { + if (canv->GetCanvasImp()->ProcessWSRequest(arg)) { arg->SetMethod("WS_READY"); TLongPollEngine* handle = new TLongPollEngine("longpoll", arg->fPathName.Data()); @@ -730,19 +770,26 @@ void THttpServer::ProcessRequest(THttpCallArg *arg) arg->SetWSId(handle->GetId()); arg->SetWSHandle(handle); - if (canv->GetCanvasImp()->ProcessWSRequest(arg) && !arg->Is404()) + if (canv->GetCanvasImp()->ProcessWSRequest(arg)) { arg->SetContent(TString::Format("%u",arg->GetWSId())); + arg->SetContentType("text/plain"); + } } + if (!arg->IsContentType("text/plain")) + arg->Set404(); } else { TUrl url; url.SetOptions(arg->fQuery); url.ParseOptions(); Int_t connid = url.GetIntValueFromOptions("connection"); arg->SetWSId((UInt_t)connid); - if (url.HasOption("close")) + if (url.HasOption("close")) { arg->SetMethod("WS_CLOSE"); - else + arg->SetContent("OK"); + arg->SetContentType("text/plain"); + } else { arg->SetMethod("WS_DATA"); + } if (!canv->GetCanvasImp()->ProcessWSRequest(arg)) arg->Set404(); } return; -- GitLab