/* * Copyright (C) 1996-2023 The Squid Software Foundation and contributors * * Squid software is distributed under GPLv2+ license and includes * contributions from numerous individuals and organizations. * Please see the COPYING and CONTRIBUTORS files for details. */ #ifndef SQUID_SRC_CLIENTSTREAM_H #define SQUID_SRC_CLIENTSTREAM_H #include "base/RefCount.h" #include "clientStreamForward.h" #include "dlink.h" #include "StoreIOBuffer.h" /** \defgroup ClientStreamAPI Client Streams API \ingroup Components \section ClientStreamIntroduction Introduction \par * A ClientStream implements a unidirectional, non-blocking, * pull pipeline. They allow code to be inserted into the * reply logic on an as-needed basis. For instance, * transfer-encoding logic is only needed when sending a * HTTP/1.1 reply. * \par * Each node consists of four methods - read, callback, detach, and status, * along with the stream housekeeping variables (a dlink node and pointer * to the head of the list), context data for the node, and read request * parameters - readbuf, readlen and readoff (in the body). * \par * clientStream is the basic unit for scheduling, and the clientStreamRead() * and clientStreamCallback() calls allow for deferred scheduled activity if * desired. * \section OperationTheory Theory on stream operation \par \li Something creates a pipeline. At a minimum it needs a head with a * status method and a read method, and a tail with a callback method * and a valid initial read request. \li Other nodes may be added into the pipeline. \li The tail-1th node's read method is called. * \par * For each node going up the pipeline, the node either: \li satisfies the read request, or \li inserts a new node above it and calls clientStreamRead(), or \li calls clientStreamRead() * \par * There is no requirement for the Read parameters from different * nodes to have any correspondence, as long as the callbacks provided are * correct. * \section WhatsInANode Whats in a node * * TODO: ClientStreams: These details should really be codified as a class which all ClientStream nodes inherit from. * \par Each node must have: \li read method - to allow loose coupling in the pipeline. (The reader may therefore change if the pipeline is altered, even mid-flow). \li callback method - likewise. \li status method - likewise. \li detach method - used to ensure all resources are cleaned up properly. \li dlink head pointer - to allow list inserts and deletes from within a node. \li context data - to allow the called back nodes to maintain their private information. \li read request parameters - For two reasons: \li To allow a node to determine the requested data offset, length and target buffer dynamically. Again, this is to promote loose coupling. \li Because of the callback nature of squid, every node would have to keep these parameters in their context anyway, so this reduces programmer overhead. */ class clientStreamNode { CBDATA_CLASS(clientStreamNode); public: clientStreamNode(CSR * aReadfunc, CSCB * aCallback, CSD * aDetach, CSS * aStatus, ClientStreamData); ~clientStreamNode(); clientStreamNode *prev() const; clientStreamNode *next() const; void removeFromStream(); dlink_node node; dlink_list *head; /* sucks I know, but hey, the interface is limited */ CSR *readfunc; CSCB *callback; CSD *detach; /* tell this node the next one downstream wants no more data */ CSS *status; ClientStreamData data; /* Context for the node */ StoreIOBuffer readBuffer; /* what, where and how much this node wants */ }; /// \ingroup ClientStreamAPI void clientStreamInit(dlink_list *, CSR *, CSD *, CSS *, ClientStreamData, CSCB *, CSD *, ClientStreamData, StoreIOBuffer tailBuffer); /// \ingroup ClientStreamAPI void clientStreamInsertHead(dlink_list *, CSR *, CSCB *, CSD *, CSS *, ClientStreamData); /** \ingroup ClientStreamAPI * * Call back the next node the in chain with it's requested data. * Return data to the next node in the stream. * The data may be returned immediately, or may be delayed for a later scheduling cycle. * \param thisObject 'this' reference for the client stream \param http Superset of request data, being winnowed down over time. MUST NOT be NULL. \param rep Not NULL on the first call back only. Ownership is passed down the pipeline. Each node may alter the reply if appropriate. \param replyBuffer - buffer, length - where and how much. */ void clientStreamCallback(clientStreamNode *thisObject, ClientHttpRequest *http, HttpReply *rep, StoreIOBuffer replyBuffer); /** \ingroup ClientStreamAPI * * Triggers a read of data that satisfies the httpClientRequest * metainformation and (if appropriate) the offset,length and buffer * parameters. * \param thisObject 'this' reference for the client stream \param http Superset of request data, being winnowed down over time. MUST NOT be NULL. \param readBuffer - offset, length, buffer - what, how much and where. */ void clientStreamRead(clientStreamNode *thisObject, ClientHttpRequest *http, StoreIOBuffer readBuffer); /** \ingroup ClientStreamAPI * * Removes this node from a clientStream. The stream infrastructure handles the removal. * This node MUST have cleaned up all context data, UNLESS scheduled callbacks will take care of that. * Informs the prev node in the list of this nodes detachment. * \param thisObject 'this' reference for the client stream \param http MUST NOT be NULL. */ void clientStreamDetach(clientStreamNode *thisObject, ClientHttpRequest *http); /** \ingroup ClientStreamAPI * * Detaches the tail of the stream. CURRENTLY DOES NOT clean up the tail node data - * this must be done separately. Thus Abort may ONLY be called by the tail node. * \param thisObject 'this' reference for the client stream \param http MUST NOT be NULL. */ void clientStreamAbort(clientStreamNode *thisObject, ClientHttpRequest *http); /** \ingroup ClientStreamAPI * * Allows nodes to query the upstream nodes for : \li stream ABORTS - request cancelled for some reason. upstream will not accept further reads(). \li stream COMPLETION - upstream has completed and will not accept further reads(). \li stream UNPLANNED COMPLETION - upstream has completed, but not at a pre-planned location (used for keepalive checking), and will not accept further reads(). \li stream NONE - no special status, further reads permitted. * \param thisObject 'this' reference for the client stream \param http MUST NOT be NULL. */ clientStream_status_t clientStreamStatus(clientStreamNode *thisObject, ClientHttpRequest *http); #endif /* SQUID_SRC_CLIENTSTREAM_H */