#ifndef __XRDSSIREQUEST_HH__ #define __XRDSSIREQUEST_HH__ /******************************************************************************/ /* */ /* X r d S s i R e q u e s t . h h */ /* */ /* (c) 2013 by the Board of Trustees of the Leland Stanford, Jr., University */ /* Produced by Andrew Hanushevsky for Stanford University under contract */ /* DE-AC02-76-SFO0515 with the Department of Energy */ /* */ /* This file is part of the XRootD software suite. */ /* */ /* XRootD is free software: you can redistribute it and/or modify it under */ /* the terms of the GNU Lesser General Public License as published by the */ /* Free Software Foundation, either version 3 of the License, or (at your */ /* option) any later version. */ /* */ /* XRootD is distributed in the hope that it will be useful, but WITHOUT */ /* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or */ /* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public */ /* License for more details. */ /* */ /* You should have received a copy of the GNU Lesser General Public License */ /* along with XRootD in a file called COPYING.LESSER (LGPL license) and file */ /* COPYING (GPL license). If not, see . */ /* */ /* The copyright holder's institutional names and contributor's names may not */ /* be used to endorse or promote products derived from this software without */ /* specific prior written permission of the institution or contributor. */ /******************************************************************************/ #include #include #include #include "XrdSsi/XrdSsiAtomics.hh" #include "XrdSsi/XrdSsiErrInfo.hh" #include "XrdSsi/XrdSsiRespInfo.hh" //----------------------------------------------------------------------------- //! The XrdSsiRequest class describes a client request and is used to effect a //! response to the request via a companion object described by XrdSsiResponder. //! Client-Side: Use this object to encapsulate your request and hand it off //! to XrdSsiService::Execute() either use GetResponseData() or //! the actual response structure to get the response data once the //! ProcessResponse() callback is invoked. //! //! Server-side: XrdSsiService::ProcessRequest() is called with this object. //! Use the XrdSsiResponder object to post a response. //! //! In either case, the client must invoke XrdSsiRequest::Finished() after the //! client-server exchange is complete in order to revert ownership of this //! object to the object's creator to allow it to be deleted or reused. //! //! This is an abstract class and several methods need to be implemented: //! //! Alert() Optional, allows receiving of server alerts. //! GetRequest() Mandatory to supply the buffer holding the request //! along with its length. //! RelRequestBuffer() Optional, allows memory optimization. //! ProcessResponse() Initial response: Mandatory //! ProcessResponseData() Data response: Mandatory only if response data is //! asynchronously received. //! //! All callbacks are invoked with no locks outstanding unless otherwise noted. //----------------------------------------------------------------------------- class XrdSsiResponder; class XrdSsiRequest { public: friend class XrdSsiResponder; friend class XrdSsiRRAgent; //----------------------------------------------------------------------------- //! Indicate that request processing has been finished. This method calls //! XrdSsiResponder::Finished() on the associated responder object. //! //! Note: This method locks the object's recursive mutex. //! //! @param cancel False -> the request/response sequence completed normally. //! True -> the request/response sequence aborted because of an //! error or the client cancelled the request. //! //! @return true Finish accepted. Request object may be reclaimed. //! @return false Finish cannot be accepted because this request object is //! not bound to a responder. This indicates a logic error. //----------------------------------------------------------------------------- bool Finished(bool cancel=false); //----------------------------------------------------------------------------- //! Obtain the detached request time to live value. If the value is non-zero, //! the request is detached. Otherwise, it is an attached request and requires a //! live TCP connection during it execution. //! //! @return The detached time to live value in seconds. //----------------------------------------------------------------------------- inline uint32_t GetDetachTTL() {return detTTL;} //----------------------------------------------------------------------------- //! Obtain the endpoint host name. //! //! @return A string containing the endpoint host name. If a null string is //! returned, the endpoint has not yet been determined. Generally, the //! endpoint is available on the first callback to this object. //----------------------------------------------------------------------------- std::string GetEndPoint(); //----------------------------------------------------------------------------- //! Obtain the metadata associated with a response. //! //! //! Note: This method locks the object's recursive mutex. //! //! @param dlen holds the length of the metadata after the call. //! //! @return =0 No metadata available, dlen has been set to zero. //! @return !0 Pointer to the buffer holding the metadata, dlen has the length //----------------------------------------------------------------------------- const char *GetMetadata(int &dlen); //----------------------------------------------------------------------------- //! Obtain the request data sent by a client. //! //! This method is duplicated in XrdSsiResponder to allow calling consistency. //! //! @param dlen holds the length of the request after the call. //! //! @return =0 No request data available, dlen has been set to zero. //! @return !0 Pointer to the buffer holding the request, dlen has the length //----------------------------------------------------------------------------- virtual char *GetRequest(int &dlen) = 0; //----------------------------------------------------------------------------- //! Get the request ID established at object creation time. //! //! @return Pointer to the request ID or nil if there is none. //----------------------------------------------------------------------------- inline const char *GetRequestID() {return reqID;} //----------------------------------------------------------------------------- //! Asynchronously obtain response data. This is a helper method that allows a //! client to deal with a passive stream response. This method also handles //! data response, albeit inefficiently by copying the data response. However, //! this allows for uniform response processing regardless of response type. //! //! @param buff pointer to the buffer to receive the data. The buffer must //! remain valid until ProcessResponseData() is called. //! @param blen the length of the buffer (i.e. maximum that can be returned). //----------------------------------------------------------------------------- void GetResponseData(char *buff, int blen); //----------------------------------------------------------------------------- //! Get timeout for initiating the request. //! //! @return The timeout value. //----------------------------------------------------------------------------- uint16_t GetTimeOut() {return tOut;} //----------------------------------------------------------------------------- //! Notify request that a response is ready to be processed. This method must //! be supplied by the request object's implementation. //! //! @param eInfo Error information. You can check if an error occurred using //! eInfo.hasError() or eInfo.isOK(). //! @param rInfo Raw response information. //! //! @return true Response processed. //! @return false Response could not be processed, the request is not active. //----------------------------------------------------------------------------- virtual bool ProcessResponse(const XrdSsiErrInfo &eInfo, const XrdSsiRespInfo &rInfo)=0; //----------------------------------------------------------------------------- //! Handle incoming async stream data or error. This method is called by a //! stream object after a successful GetResponseData() or an asynchronous //! stream SetBuff() call. //! //! @param eInfo Error information. You can check if an error occurred using //! eInfo.hasError() or eInfo.isOK(). //! @param buff Pointer to the buffer given to XrdSsiStream::SetBuff(). //! @param blen The number of bytes in buff or an error indication if blen < 0. //! @param last true This is the last stream segment, no more data remains. //! false More data may remain in the stream. //----------------------------------------------------------------------------- virtual void ProcessResponseData(const XrdSsiErrInfo &eInfo, char *buff, int blen, bool last) {} //----------------------------------------------------------------------------- //! Release the request buffer of the request bound to this object. This method //! duplicates the protected method RelRequestBuffer() and exists here for //! calling safety and consistency relative to the responder. //----------------------------------------------------------------------------- void ReleaseRequestBuffer(); //----------------------------------------------------------------------------- //! Constructor //! //! @param reqid Pointer to a request ID that can be used to group requests. //! See ProcessResponseData() and RestartDataReponse(). If reqid //! is nil then held responses are placed in the global queue. //! The pointer must be valid for the life of this object. //! //! @param tmo The request initiation timeout value 0 equals default). //----------------------------------------------------------------------------- XrdSsiRequest(const char *reqid=0, uint16_t tmo=0); protected: //----------------------------------------------------------------------------- //! @brief Send or receive a server generated alert. //! //! The Alert() method is used server-side to send one or more alerts before a //! response is posted (alerts afterwards are ignored). To avoid race conditions, //! server-side alerts should be sent via the Responder's Alert() method. //! Clients must implement this method in order to receive alerts. //! //! @param aMsg Reference to the message object containing the alert message. //! Non-positive alert lengths cause the alert call to be //! ignored. You should call the message RecycleMsg() method //! once you have consumed the message to release its resources. //----------------------------------------------------------------------------- virtual void Alert(XrdSsiRespInfoMsg &aMsg) {aMsg.RecycleMsg(false);} //----------------------------------------------------------------------------- //! Release the request buffer. Use this method to optimize storage use; this //! is especially relevant for long-running requests. If the request buffer //! has been consumed and is no longer needed, early return of the buffer will //! minimize memory usage. This method is also invoked via XrdSsiResponder. //! //! //! Note: This method is called with the object's recursive mutex locked when //! it is invoked via XrdSsiResponder's ReleaseRequestBuffer(). //----------------------------------------------------------------------------- virtual void RelRequestBuffer() {} //----------------------------------------------------------------------------- //! @brief Set the detached request time to live value. //! //! By default, requests are executed in the foreground (i.e. during its //! execution, if the TCP connection drops, the request is automatically //! cancelled. When a non-zero time to live is set, the request is executed in //! the background (i.e. detached) and no persistent TCP connection is required. //! You must use the XrdSsiService::Attach() method to foreground such a //! request within the number of seconds specified for dttl or the request is //! automatically cancelled. The value must be set before passing the request //! to XrdSsiService::ProcessRequest(). Once the request is started, a request //! handle is returned which can be passed to XrdSsiService::Attach(). //! //! @param dttl The detach time to live value. //----------------------------------------------------------------------------- inline void SetDetachTTL(uint32_t dttl) {detTTL = dttl;} //----------------------------------------------------------------------------- //! Set request retry notification. If a non-default value is desired, it must //! be set prior to calling XrdSsiService::ProcessRequest(). This is a one-time //! request and retry mode is turned off in the request object afterwards. //! //! @param onoff True to turn retry on and false to turn it off. //----------------------------------------------------------------------------- void SetRetry(bool onoff); //----------------------------------------------------------------------------- //! Set timeout for initiating the request. If a non-default value is desired, //! it must be set prior to calling XrdSsiService::ProcessRequest(). //! //! @param tmo The timeout value. //----------------------------------------------------------------------------- void SetTimeOut(uint16_t tmo) {tOut = tmo;} //----------------------------------------------------------------------------- //! Destructor. This object can only be deleted by the object creator. Once the //! object is passed to XrdSsiService::ProcessRequest() it may only be deleted //! after Finished() is called to allow the service to reclaim any resources //! allocated for the request object. //----------------------------------------------------------------------------- virtual ~XrdSsiRequest() {} private: virtual void BindDone() {} void CleanUp(); bool CopyData(char *buff, int blen); virtual void Dispose() {} const char *reqID; XrdSsiMutex *rrMutex; XrdSsiResponder *theRespond; // Set via XrdSsiResponder::BindRequest() XrdSsiRespInfo Resp; // Set via XrdSsiResponder::SetResponse() XrdSsiErrInfo errInfo; long long rsvd1; const char *epNode; uint32_t detTTL; uint16_t tOut; bool onClient; char flags; static const int isaRetry = 1; }; #endif