32using namespace std::string_literals;
41 fHold->SetTextContent(
"console.log('execute holder script'); if (window) setTimeout (window.close, 1000); if (window) window.close();");
42 fHold->NotifyCondition();
99 for (
auto &conn : lst) {
100 conn->fActive =
false;
101 for (
auto &elem: conn->fEmbed)
102 elem.second->fMaster.reset();
105 fMgr->Unregister(*
this);
118 if (!
fConn.empty()) {
131std::shared_ptr<RWebWindowWSHandler>
150 return fMgr->GetUrl(*
this, remote);
158 return fMgr->GetServer();
168 return fMgr->ShowWindow(*
this, args);
185 connid =
fMgr->ShowWindow(*
this, args);
201 if (entry->fHeadlessMode)
202 return entry->fConnId;
205 for (
auto &conn :
fConn) {
206 if (conn->fHeadlessMode)
207 return conn->fConnId;
224 if (!entry->fHeadlessMode)
225 return entry->fConnId;
228 for (
auto &conn :
fConn) {
229 if (!conn->fHeadlessMode)
230 return conn->fConnId;
243 for (
auto &conn :
fConn) {
244 if (conn->fWSId == wsid)
252 std::shared_ptr<WebConn> key;
253 std::string keyvalue;
262 if (!keyvalue.empty())
274 fConn.emplace_back(key);
276 fConn.emplace_back(std::make_shared<WebConn>(++
fConnCnt, wsid));
289 std::shared_ptr<WebConn> res;
294 for (
size_t n = 0;
n <
fConn.size(); ++
n)
295 if (
fConn[
n]->fWSId == wsid) {
296 res = std::move(
fConn[
n]);
298 res->fActive =
false;
304 for (
auto &elem: res->fEmbed)
305 elem.second->fMaster.reset();
317 std::string query = arg->GetQuery();
319 if (query.compare(0, 4,
"key=") != 0)
322 std::string key = query.substr(4);
324 std::shared_ptr<THttpCallArg> prev;
326 bool found_key =
false;
332 if (entry->fKey == key) {
335 prev = std::move(entry->fHold);
340 for (
auto &conn :
fConn) {
341 if (conn->fKey == key) {
343 prev = std::move(conn->fHold);
351 prev->SetTextContent(
"console.log('execute holder script'); if (window) window.close();");
352 prev->NotifyCondition();
391 connid = entry.fConnId;
393 arg = std::move(entry.fData);
424 auto conn = std::make_shared<WebConn>(++
fConnCnt, headless_mode, key);
426 std::swap(conn->fDisplayHandle, handle);
443 if (entry->fKey == key)
447 for (
auto &conn :
fConn) {
448 if (conn->fKey == key)
464 return conn ? true :
false;
480 key = std::to_string(rnd.
Integer(0x100000));
481 }
while ((--ntry > 0) &&
HasKey(key));
483 if (ntry <= 0) key.clear();
498 float tmout =
fMgr->GetLaunchTmout();
505 auto pred = [&](std::shared_ptr<WebConn> &
e) {
506 std::chrono::duration<double> diff =
stamp -
e->fSendStamp;
508 if (diff.count() > tmout) {
510 selected.emplace_back(
e);
531 double batch_tmout = 20.;
533 std::vector<std::shared_ptr<WebConn>> clr;
538 auto pred = [&](std::shared_ptr<WebConn> &conn) {
539 std::chrono::duration<double> diff =
stamp - conn->fSendStamp;
541 if ((diff.count() > batch_tmout) && conn->fHeadlessMode) {
542 conn->fActive =
false;
543 clr.emplace_back(conn);
552 for (
auto &entry : clr)
608 fMgr->AssignWindowThreadId(*
this);
654 if (conn->fKeyUsed > 0) {
655 R__LOG_ERROR(
WebGUILog()) <<
"key " << key <<
" was used for establishing connection, call ShowWindow again";
683 if (conn->fKeyUsed < 0) {
715 char *str_end =
nullptr;
717 unsigned long ackn_oper = std::strtoul(buf, &str_end, 10);
718 if (!str_end || *str_end !=
':') {
723 unsigned long can_send = std::strtoul(str_end + 1, &str_end, 10);
724 if (!str_end || *str_end !=
':') {
729 unsigned long nchannel = std::strtoul(str_end + 1, &str_end, 10);
730 if (!str_end || *str_end !=
':') {
735 Long_t processed_len = (str_end + 1 - buf);
747 std::lock_guard<std::mutex> grd(conn->fMutex);
749 conn->fSendCredits += ackn_oper;
751 conn->fClientCredits = (
int)can_send;
752 conn->fRecvStamp =
stamp;
760 if ((nchannel != 0) || (cdata.find(
"READY=") == 0)) {
773 if ((cdata.find(
"READY=") == 0) && !conn->fReady) {
774 std::string key = cdata.substr(6);
781 if (!key.empty() && !conn->fKey.empty() && (conn->fKey != key)) {
782 R__LOG_ERROR(
WebGUILog()) <<
"Key mismatch after established connection " << key <<
" != " << conn->fKey;
795 }
else if (cdata.compare(0,8,
"CLOSECH=") == 0) {
796 int channel = std::stoi(cdata.substr(8));
797 auto iter = conn->fEmbed.find(channel);
798 if (iter != conn->fEmbed.end()) {
800 conn->fEmbed.erase(iter);
802 }
else if (cdata ==
"GENERATE_KEY") {
811 SubmitData(conn->fConnId,
true,
"NEW_KEY="s + newkey, -1);
817 }
else if (
fPanelName.length() && (conn->fReady < 10)) {
818 if (cdata ==
"PANEL_READY") {
826 }
else if (nchannel == 1) {
828 }
else if (nchannel > 1) {
830 auto embed_window = conn->fEmbed[nchannel];
832 embed_window->ProvideQueueEntry(conn->fConnId,
kind_Data, std::move(cdata));
852 std::lock_guard<std::mutex> grd(conn->fMutex);
853 conn->fDoingSend =
false;
872 if (conn->fSendCredits <= 0) {
877 if (conn->fDoingSend) {
883 buf.reserve(
data.length() + 100);
885 buf.append(std::to_string(conn->fRecvCount));
887 buf.append(std::to_string(conn->fSendCredits));
889 conn->fRecvCount = 0;
890 conn->fSendCredits--;
892 buf.append(std::to_string(chid));
897 }
else if (
data.length()==0) {
898 buf.append(
"$$nullbinary$$");
900 buf.append(
"$$binary$$");
912 std::string hdr,
data;
915 std::lock_guard<std::mutex> grd(conn->fMutex);
917 if (!conn->fActive || (conn->fSendCredits <= 0) || conn->fDoingSend)
return false;
919 if (!conn->fQueue.empty()) {
922 if (!hdr.empty() && !item.
fText)
925 }
else if ((conn->fClientCredits < 3) && (conn->fRecvCount > 1)) {
930 if (hdr.empty())
return false;
932 conn->fDoingSend =
true;
938 res =
fWSHandler->SendCharStarWS(conn->fWSId, hdr.c_str());
944 if (res >=0)
return true;
947 std::lock_guard<std::mutex> grd(conn->fMutex);
948 conn->fDoingSend =
false;
966 for (
auto &conn : arr)
972 }
while (!only_once);
1019 std::string res(
"../");
1020 res.append(
win.GetAddr());
1075 auto sz =
fConn.size();
1109 return ((num >= 0) && (num < (
int)
fConn.size()) &&
fConn[num]->fActive) ?
fConn[num]->fConnId : 0;
1121 for (
auto &conn :
fConn) {
1122 if (connid && (conn->fConnId != connid))
1124 if (conn->fActive || !only_active)
1130 if (!connid || (conn->fConnId == connid))
1169 for (
auto &conn :
fConn) {
1170 if ((conn->fActive || !only_active) && (!connid || (conn->fConnId == connid)))
1171 arr.push_back(conn);
1176 if (!connid || (conn->fConnId == connid))
1177 arr.push_back(conn);
1194 for (
auto &conn : arr) {
1196 std::lock_guard<std::mutex> grd(conn->fMutex);
1198 if (
direct && (!conn->fQueue.empty() || (conn->fSendCredits == 0) || conn->fDoingSend))
1201 if (conn->fQueue.size() >= maxqlen)
1218 std::lock_guard<std::mutex> grd(conn->fMutex);
1219 int len = conn->fQueue.size();
1220 if (
len > maxq) maxq =
len;
1239 auto cnt = arr.size();
1242 bool clear_queue =
false;
1251 for (
auto &conn : arr) {
1257 fname.append(
"msg");
1259 fname.append(txt ?
".txt" :
".bin");
1261 std::ofstream ofs(fname);
1262 ofs.write(
data.c_str(),
data.length());
1274 conn->fSendStamp =
stamp;
1276 std::lock_guard<std::mutex> grd(conn->fMutex);
1279 while (!conn->fQueue.empty())
1283 if (conn->fQueue.size() < maxqlen) {
1285 conn->fQueue.emplace(chid, txt, std::string(
data));
1287 conn->fQueue.emplace(chid, txt, std::move(
data));
1326 std::copy((
const char *)
data, (
const char *)
data +
len, buf.begin());
1327 SubmitData(connid,
false, std::move(buf), 1);
1345 }
else if (
fMgr->IsUseHttpThread()) {
1347 R__LOG_ERROR(
WebGUILog()) <<
"create web window from main thread when THttpServer created with special thread - not supported";
1379 std::thread thrd([
this] {
1468 return fMgr->WaitFor(*
this, check);
1496 return fMgr->WaitFor(*
this, check,
true, duration);
1529 if (arr.size() == 0)
1533 if (arr[0]->fEmbed.find(channel) != arr[0]->fEmbed.end())
1536 arr[0]->fEmbed[channel] = window;
1538 return arr[0]->fConnId;
1548 for (
auto &conn : arr) {
1549 auto iter = conn->fEmbed.find(channel);
1550 if (iter != conn->fEmbed.end())
1551 conn->fEmbed.erase(iter);
1580 std::swap(arr1,
fConn);
1601 window->fMaster = args.
fMaster;
1602 window->fMasterConnId = connid;
1615 return window->Show(args);
#define R__LOG_WARNING(...)
#define R__LOG_ERROR(...)
#define R__LOG_DEBUG(DEBUGLEVEL,...)
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void data
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char Pixmap_t Pixmap_t PictureAttributes_t attr const char char ret_data h unsigned char height h Atom_t Int_t ULong_t ULong_t unsigned char prop_list Atom_t Atom_t Atom_t Time_t stamp
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize id
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char Pixmap_t Pixmap_t PictureAttributes_t attr const char char ret_data h unsigned char height h Atom_t Int_t ULong_t ULong_t unsigned char prop_list Atom_t Atom_t Atom_t Time_t UChar_t len
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t win
char * Form(const char *fmt,...)
Formats a string in a circular formatting buffer.
Holds different arguments for starting browser with RWebDisplayHandle::Display() method.
void SetHeadless(bool on=true)
set headless mode
int fMasterChannel
! used master channel
EBrowserKind GetBrowserKind() const
returns configured browser kind, see EBrowserKind for supported values
std::shared_ptr< RWebWindow > fMaster
! master window
@ kEmbedded
window will be embedded into other, no extra browser need to be started
static int GetBoolEnv(const std::string &name, int dfl=-1)
Parse boolean gEnv variable which should be "yes" or "no".
Represents web window, which can be shown in web browser or any other supported environment.
bool CheckDataToSend(std::shared_ptr< WebConn > &conn)
Checks if one should send data for specified connection Returns true when send operation was performe...
std::vector< std::shared_ptr< WebConn > > ConnectionsList_t
int WaitFor(WebWindowWaitFunc_t check)
Waits until provided check function or lambdas returns non-zero value Check function has following si...
unsigned AddEmbedWindow(std::shared_ptr< RWebWindow > window, int channel)
Add embed window.
std::shared_ptr< RWebWindow > fMaster
! master window where this window is embedded
void CheckInactiveConnections()
Check if there are connection which are inactive for longer time For instance, batch browser will be ...
ConnectionsList_t GetConnections(unsigned connid=0, bool only_active=false) const
returns connection list (or all active connections)
std::string fUserArgs
! arbitrary JSON code, which is accessible via conn.getUserArgs() method
int fMasterChannel
! channel id in the master window
void StartThread()
Start special thread which will be used by the window to handle all callbacks One has to be sure,...
float GetOperationTmout() const
Returns timeout for synchronous WebWindow operations.
std::shared_ptr< WebConn > FindConnection(unsigned wsid)
Find connection with specified websocket id.
void SetConnToken(const std::string &token="")
Configures connection token (default none) When specified, in URL of webpage such token should be pro...
unsigned MakeHeadless(bool create_new=false)
Start headless browser for specified window Normally only single instance is used,...
std::string GetUrl(bool remote=true)
Return URL string to access web window.
void CloseConnections()
Closes all connection to clients Normally leads to closing of all correspondent browser windows Some ...
void SetDefaultPage(const std::string &page)
Set content of default window HTML page This page returns when URL address of the window will be requ...
std::thread fWindowThrd
! special thread for that window
int NumConnections(bool with_pending=false) const
Returns current number of active clients connections.
unsigned GetId() const
Returns ID for the window - unique inside window manager.
ConnectionsList_t fConn
! list of all accepted connections
void InvokeCallbacks(bool force=false)
Invoke callbacks with existing data Must be called from appropriate thread.
std::string fProtocolPrefix
! prefix for created files names
std::string GetClientVersion() const
Returns current client version.
void SetConnectCallBack(WebWindowConnectCallback_t func)
Set call-back function for new connection.
WebWindowConnectCallback_t fConnCallback
! callback for connect event
unsigned GetMaxQueueLength() const
Return maximal queue length of data which can be held by window.
void Sync()
Special method to process all internal activity when window runs in separate thread.
void UseServerThreads()
Let use THttpServer threads to process requests WARNING!!! only for expert use Should be only used wh...
void TerminateROOT()
Terminate ROOT session Tries to correctly close THttpServer, associated with RWebWindowsManager After...
unsigned fConnLimit
! number of allowed active connections
void Send(unsigned connid, const std::string &data)
Sends data to specified connection.
bool fCallbacksThrdIdSet
! flag indicating that thread id is assigned
unsigned Show(const RWebDisplayArgs &args="")
Show window in specified location.
THttpServer * GetServer()
Return THttpServer instance serving requests to the window.
unsigned AddDisplayHandle(bool headless_mode, const std::string &key, std::unique_ptr< RWebDisplayHandle > &handle)
Add display handle and associated key Key is random number generated when starting new window When cl...
unsigned fMasterConnId
! master connection id
void AssignThreadId()
Assign thread id which has to be used for callbacks WARNING!!! only for expert use Automatically done...
bool fSendMT
! true is special threads should be used for sending data
void SendBinary(unsigned connid, const void *data, std::size_t len)
Send binary data to specified connection.
static std::shared_ptr< RWebWindow > Create()
Create new RWebWindow Using default RWebWindowsManager.
std::thread::id fCallbacksThrdId
! thread id where callbacks should be invoked
std::chrono::time_point< std::chrono::system_clock > timestamp_t
std::string fClientVersion
! configured client version, used as prefix in scripts URL
bool ProcessBatchHolder(std::shared_ptr< THttpCallArg > &arg)
Process special http request, used to hold headless browser running Such requests should not be repli...
unsigned fConnCnt
! counter of new connections to assign ids
void SetDisconnectCallBack(WebWindowConnectCallback_t func)
Set call-back function for disconnecting.
std::string fPanelName
! panel name which should be shown in the window
void SetDataCallBack(WebWindowDataCallback_t func)
Set call-back function for data, received from the clients via websocket.
unsigned fProtocolConnId
! connection id, which is used for writing protocol
void SetUserArgs(const std::string &args)
Set arbitrary JSON data, which is accessible via conn.getUserArgs() method in JavaScript This JSON co...
static unsigned ShowWindow(std::shared_ptr< RWebWindow > window, const RWebDisplayArgs &args="")
Static method to show web window Has to be used instead of RWebWindow::Show() when window potentially...
void StopThread()
Stop special thread.
WebWindowDataCallback_t fDataCallback
! main callback when data over channel 1 is arrived
void SubmitData(unsigned connid, bool txt, std::string &&data, int chid=1)
Internal method to send data.
~RWebWindow()
RWebWindow destructor Closes all connections and remove window from manager.
void CloseConnection(unsigned connid)
Close specified connection.
unsigned GetConnectionId(int num=0) const
Returns connection id for specified connection sequence number Only active connections are returned -...
std::string GetConnToken() const
Returns configured connection token.
ConnectionsList_t fPendingConn
! list of pending connection with pre-assigned keys
void SetConnLimit(unsigned lmt=0)
Configure maximal number of allowed connections - 0 is unlimited Will not affect already existing con...
void SetPanelName(const std::string &name)
Configure window to show some of existing JSROOT panels It uses "file:rootui5sys/panel/panel....
RWebWindow()
RWebWindow constructor Should be defined here because of std::unique_ptr<RWebWindowWSHandler>
std::shared_ptr< WebConn > FindOrCreateConnection(unsigned wsid, bool make_new, const char *query)
Find connection with given websocket id.
bool fHasWindowThrd
! indicate if special window thread was started
int GetSendQueueLength(unsigned connid) const
Returns send queue length for specified connection.
std::shared_ptr< WebConn > RemoveConnection(unsigned wsid)
Remove connection with given websocket id.
std::shared_ptr< RWebWindowWSHandler > CreateWSHandler(std::shared_ptr< RWebWindowsManager > mgr, unsigned id, double tmout)
Assigns manager reference, window id and creates websocket handler, used for communication with the c...
std::shared_ptr< WebConn > _FindConnWithKey(const std::string &key) const
Find connection with specified key.
bool CanSend(unsigned connid, bool direct=true) const
Returns true if sending via specified connection can be performed.
std::string GetUserArgs() const
Returns configured user arguments for web window See SetUserArgs method for more details.
void RecordData(const std::string &fname="protocol.json", const std::string &fprefix="")
Configures recording of communication data in protocol file Provided filename will be used to store J...
void CheckThreadAssign()
Internal method to verify and thread id has to be assigned from manager again Special case when Proce...
bool HasKey(const std::string &key) const
Returns true if provided key value already exists (in processes map or in existing connections)
unsigned GetDisplayConnection() const
Returns first connection id where window is displayed It could be that connection(s) not yet fully es...
unsigned GetConnLimit() const
returns configured connections limit (0 - default)
std::string GetRelativeAddr(const std::shared_ptr< RWebWindow > &win) const
Returns relative URL address for the specified window Address can be required if one needs to access ...
void Run(double tm=0.)
Run window functionality for specified time If no action can be performed - just sleep specified time...
std::string GetAddr() const
Returns window address which is used in URL.
std::shared_ptr< RWebWindowWSHandler > fWSHandler
! specialize websocket handler for all incoming connections
bool fUseServerThreads
! indicates that server thread is using, no special window thread
std::string fProtocolFileName
! local file where communication protocol will be written
std::shared_ptr< RWebWindowsManager > fMgr
! display manager
void CheckPendingConnections()
Check if started process(es) establish connection.
std::string fConnToken
! value of "token" URL parameter which should be provided for connecting window
std::mutex fInputQueueMutex
! mutex to protect input queue
std::string _MakeSendHeader(std::shared_ptr< WebConn > &conn, bool txt, const std::string &data, int chid)
Internal method to prepare text part of send data Should be called under locked connection mutex.
bool IsNativeOnlyConn() const
returns true if only native (own-created) connections are allowed
bool ProcessWS(THttpCallArg &arg)
Processing of websockets call-backs, invoked from RWebWindowWSHandler Method invoked from http server...
bool HasConnection(unsigned connid=0, bool only_active=true) const
returns true if specified connection id exists
int fProtocolCnt
! counter for protocol recording
std::queue< QueueEntry > fInputQueue
! input queue for all callbacks
bool fProcessMT
! if window event processing performed in dedicated thread
std::string fProtocol
! protocol
void ProvideQueueEntry(unsigned connid, EQueueEntryKind kind, std::string &&arg)
Provide data to user callback User callback must be executed in the window thread.
void CompleteWSSend(unsigned wsid)
Complete websocket send operation Clear "doing send" flag and check if next operation has to be start...
unsigned fId
! unique identifier
float fOperationTmout
! timeout in seconds to perform synchronous operation, default 50s
unsigned FindHeadlessConnection()
Returns connection id of window running in headless mode This can be special connection which may run...
int WaitForTimed(WebWindowWaitFunc_t check)
Waits until provided check function or lambdas returns non-zero value Check function has following si...
WebWindowConnectCallback_t fDisconnCallback
! callback for disconnect event
void SetClientVersion(const std::string &vers)
Set client version, used as prefix in scripts URL When changed, web browser will reload all related J...
void RemoveEmbedWindow(unsigned connid, int channel)
Remove RWebWindow associated with the channel.
void SetCallBacks(WebWindowConnectCallback_t conn, WebWindowDataCallback_t data, WebWindowConnectCallback_t disconn=nullptr)
Set call-backs function for connect, data and disconnect events.
std::string GenerateKey() const
Generate new unique key for the window.
std::mutex fConnMutex
! mutex used to protect connection list
static bool IsMainThrd()
Returns true when called from main process Main process recognized at the moment when library is load...
static std::shared_ptr< RWebWindowsManager > & Instance()
Returns default window manager Used to display all standard ROOT elements like TCanvas or TFitPanel.
Contains arguments for single HTTP call.
UInt_t GetWSId() const
get web-socket id
const void * GetPostData() const
return pointer on posted with request data
const char * GetQuery() const
returns request query (string after ? in request URL)
Long_t GetPostDataLength() const
return length of posted with request data
Bool_t IsMethod(const char *name) const
returns kTRUE if post method is used
Online http server for arbitrary ROOT application.
Random number generator class based on M.
void SetSeed(ULong_t seed=0) override
Set the random generator sequence if seed is 0 (default value) a TUUID is generated and used to fill ...
virtual UInt_t Integer(UInt_t imax)
Returns a random integer uniformly distributed on the interval [ 0, imax-1 ].
This class represents a WWW compatible URL.
const char * GetValueFromOptions(const char *key) const
Return a value for a given key from the URL options.
void SetOptions(const char *opt)
Bool_t HasOption(const char *key) const
Returns true if the given key appears in the URL options list.
RLogChannel & WebGUILog()
Log channel for WebGUI diagnostics.
std::function< void(unsigned)> WebWindowConnectCallback_t
function signature for connect/disconnect call-backs argument is connection id
std::function< void(unsigned, const std::string &)> WebWindowDataCallback_t
function signature for call-backs from the window clients first argument is connection id,...
std::function< int(double)> WebWindowWaitFunc_t
function signature for waiting call-backs Such callback used when calling thread need to waits for so...
std::string fData
! text or binary data
~WebConn()
Destructor for WebConn Notify special HTTP request which blocks headless browser from exit.
std::shared_ptr< THttpCallArg > fHold
! request used to hold headless browser