blob: 0b549510d2ae056ab086efb18d49cae759575569 [file] [log] [blame]
Christopher Fauletd7c91962015-04-30 11:48:27 +02001/*
2 * include/types/filteers.h
3 * This file defines everything related to stream filters.
4 *
5 * Copyright (C) 2015 Qualys Inc., Christopher Faulet <cfaulet@qualys.com>
6 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation, version 2.1
10 * exclusively.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21#ifndef _TYPES_FILTERS_H
22#define _TYPES_FILTERS_H
23
24#include <common/config.h>
25#include <common/mini-clist.h>
26
27struct http_msg;
28struct proxy;
29struct stream;
30struct channel;
Christopher Faulet443ea1a2016-02-04 13:40:26 +010031struct flt_conf;
Christopher Fauletd7c91962015-04-30 11:48:27 +020032struct filter;
33
34/* Descriptor for a "filter" keyword. The ->parse() function returns 0 in case
35 * of success, or a combination of ERR_* flags if an error is encountered. The
Christopher Fauletb3f4e142016-03-07 12:46:38 +010036 * function pointer can be NULL if not implemented.
Christopher Fauletd7c91962015-04-30 11:48:27 +020037 */
38struct flt_kw {
39 const char *kw;
40 int (*parse)(char **args, int *cur_arg, struct proxy *px,
Thierry Fournier3610c392016-04-13 18:27:51 +020041 struct flt_conf *fconf, char **err, void *private);
42 void *private;
Christopher Fauletd7c91962015-04-30 11:48:27 +020043};
44
45/*
46 * A keyword list. It is a NULL-terminated array of keywords. It embeds a struct
47 * list in order to be linked to other lists, allowing it to easily be declared
48 * where it is needed, and linked without duplicating data nor allocating
49 * memory. It is also possible to indicate a scope for the keywords.
50 */
51struct flt_kw_list {
52 const char *scope;
53 struct list list;
54 struct flt_kw kw[VAR_ARRAY];
55};
56
57/*
Christopher Fauletd7c91962015-04-30 11:48:27 +020058 * Callbacks available on a filter:
59 *
60 * - init : Initializes the filter for a proxy. Returns a
61 * negative value if an error occurs.
62 * - deinit : Cleans up what the init function has done.
63 * - check : Check the filter config for a proxy. Returns the
64 * number of errors encountered.
Christopher Faulet71a6a8e2017-07-27 16:33:28 +020065 * - init_per_thread : Initializes the filter for a proxy for a specific
66 * thread. Returns a negative value if an error
67 * occurs.
68 * - deinit_per_thread : Cleans up what the init_per_thread funcion has
69 * done.
Christopher Fauletd7c91962015-04-30 11:48:27 +020070 *
71 *
Christopher Faulet31ed32d2016-06-21 11:42:37 +020072 * - attach : Called after a filter instance creation, when it is
73 * attached to a stream. This happens when the stream
74 * is started for filters defined on the stream's
75 * frontend and when the backend is set for filters
76 * declared on the stream's backend.
77 * Returns a negative value if an error occurs, 0 if
78 * the filter must be ignored for the stream, any other
79 * value otherwise.
Christopher Fauletd7c91962015-04-30 11:48:27 +020080 * - stream_start : Called when a stream is started. This callback will
Christopher Faulet31ed32d2016-06-21 11:42:37 +020081 * only be called for filters defined on the stream's
82 * frontend.
83 * Returns a negative value if an error occurs, any
84 * other value otherwise.
85 * - stream_set_backend : Called when a backend is set for a stream. This
86 * callbacks will be called for all filters attached
87 * to a stream (frontend and backend).
Christopher Fauletd7c91962015-04-30 11:48:27 +020088 * Returns a negative value if an error occurs, any
89 * other value otherwise.
90 * - stream_stop : Called when a stream is stopped. This callback will
Christopher Faulet31ed32d2016-06-21 11:42:37 +020091 * only be called for filters defined on the stream's
92 * frontend.
93 * - detach : Called when a filter instance is detached from a
94 * stream, before its destruction. This happens when
95 * the stream is stopped for filters defined on the
96 * stream's frontend and when the analyze ends for
97 * filters defined on the stream's backend.
Christopher Fauleta00d8172016-11-10 14:58:05 +010098 * - check_timeouts : Called when a a stream is woken up because of an
99 * expired timer.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200100 *
101 *
102 * - channel_start_analyze: Called when a filter starts to analyze a channel.
103 * Returns a negative value if an error occurs, 0 if
104 * it needs to wait, any other value otherwise.
Christopher Faulet3a394fa2016-05-11 17:13:39 +0200105 * - channel_pre_analyze : Called before each analyzer attached to a channel,
Christopher Fauletd7c91962015-04-30 11:48:27 +0200106 * expects analyzers responsible for data sending.
107 * Returns a negative value if an error occurs, 0 if
108 * it needs to wait, any other value otherwise.
Christopher Faulet3a394fa2016-05-11 17:13:39 +0200109 * - channel_post_analyze: Called after each analyzer attached to a channel,
110 * expects analyzers responsible for data sending.
111 * Returns a negative value if an error occurs,
112 * any other value otherwise.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200113 * - channel_end_analyze : Called when all other analyzers have finished their
114 * processing.
115 * Returns a negative value if an error occurs, 0 if
116 * it needs to wait, any other value otherwise.
117 *
118 *
Christopher Faulet1339d742016-05-11 16:48:33 +0200119 * - http_headers : Called before the body parsing, after all HTTP
120 * headers was parsed and analyzed.
121 * Returns a negative value if an error occurs, 0 if
122 * it needs to wait, any other value otherwise.
Christopher Faulet75bc9132018-11-30 15:18:09 +0100123 * - http_payload : Called when some data can be consumed.
124 * Returns a negative value if an error occurs, else
125 * the number of forwarded bytes.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200126 * - http_end : Called when all the request/response has been
127 * processed and all body data has been forwarded.
128 * Returns a negative value if an error occurs, 0 if
129 * it needs to wait for some reason, any other value
130 * otherwise.
131 * - http_reset : Called when the HTTP message is reseted. It happens
Olivier Houcharda254a372019-04-05 15:30:12 +0200132 * either when a 100-continue response is received.
133 * that can be detected if s->txn->status is 10X, or
134 * if we're attempting a L7 retry.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200135 * Returns nothing.
136 * - http_reply : Called when, at any time, HA proxy decides to stop
137 * the HTTP message's processing and to send a message
138 * to the client (mainly, when an error or a redirect
139 * occur).
140 * Returns nothing.
Christopher Fauletb2e58492019-11-12 11:13:01 +0100141 *
142 *
143 * - tcp_payload : Called when some data can be consumed.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200144 * Returns a negative value if an error occurs, else
Christopher Fauletb2e58492019-11-12 11:13:01 +0100145 * the number of forwarded bytes.
Christopher Fauletd7c91962015-04-30 11:48:27 +0200146 */
147struct flt_ops {
148 /*
149 * Callbacks to manage the filter lifecycle
150 */
Christopher Faulet71a6a8e2017-07-27 16:33:28 +0200151 int (*init) (struct proxy *p, struct flt_conf *fconf);
152 void (*deinit) (struct proxy *p, struct flt_conf *fconf);
153 int (*check) (struct proxy *p, struct flt_conf *fconf);
154 int (*init_per_thread) (struct proxy *p, struct flt_conf *fconf);
155 void (*deinit_per_thread)(struct proxy *p, struct flt_conf *fconf);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200156 /*
157 * Stream callbacks
158 */
Christopher Faulet31ed32d2016-06-21 11:42:37 +0200159 int (*attach) (struct stream *s, struct filter *f);
160 int (*stream_start) (struct stream *s, struct filter *f);
161 int (*stream_set_backend)(struct stream *s, struct filter *f, struct proxy *be);
162 void (*stream_stop) (struct stream *s, struct filter *f);
163 void (*detach) (struct stream *s, struct filter *f);
Christopher Fauleta00d8172016-11-10 14:58:05 +0100164 void (*check_timeouts) (struct stream *s, struct filter *f);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200165 /*
166 * Channel callbacks
167 */
168 int (*channel_start_analyze)(struct stream *s, struct filter *f, struct channel *chn);
Christopher Faulet3a394fa2016-05-11 17:13:39 +0200169 int (*channel_pre_analyze) (struct stream *s, struct filter *f, struct channel *chn, unsigned int an_bit);
170 int (*channel_post_analyze) (struct stream *s, struct filter *f, struct channel *chn, unsigned int an_bit);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200171 int (*channel_end_analyze) (struct stream *s, struct filter *f, struct channel *chn);
172
173 /*
174 * HTTP callbacks
175 */
Christopher Faulet1339d742016-05-11 16:48:33 +0200176 int (*http_headers) (struct stream *s, struct filter *f, struct http_msg *msg);
Christopher Faulet75bc9132018-11-30 15:18:09 +0100177 int (*http_payload) (struct stream *s, struct filter *f, struct http_msg *msg,
178 unsigned int offset, unsigned int len);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200179 int (*http_end) (struct stream *s, struct filter *f, struct http_msg *msg);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200180
Christopher Faulet309c6412015-12-02 09:57:32 +0100181 void (*http_reset) (struct stream *s, struct filter *f, struct http_msg *msg);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200182 void (*http_reply) (struct stream *s, struct filter *f, short status,
Willy Tarreau83061a82018-07-13 11:56:34 +0200183 const struct buffer *msg);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200184
185 /*
186 * TCP callbacks
187 */
Christopher Fauletb2e58492019-11-12 11:13:01 +0100188 int (*tcp_payload) (struct stream *s, struct filter *f, struct channel *chn,
189 unsigned int offset, unsigned int len);
Christopher Fauletd7c91962015-04-30 11:48:27 +0200190};
191
Christopher Faulet75bc9132018-11-30 15:18:09 +0100192/* Flags set on a filter config */
193#define FLT_CFG_FL_HTX 0x00000001 /* The filter can filter HTX streams */
194
Christopher Fauletda02e172015-12-04 09:25:05 +0100195/* Flags set on a filter instance */
196#define FLT_FL_IS_BACKEND_FILTER 0x0001 /* The filter is a backend filter */
197#define FLT_FL_IS_REQ_DATA_FILTER 0x0002 /* The filter will parse data on the request channel */
198#define FLT_FL_IS_RSP_DATA_FILTER 0x0004 /* The filter will parse data on the response channel */
199
Christopher Fauletda02e172015-12-04 09:25:05 +0100200/* Flags set on the stream, common to all filters attached to its stream */
201#define STRM_FLT_FL_HAS_FILTERS 0x0001 /* The stream has at least one filter */
202
Christopher Fauletd7c91962015-04-30 11:48:27 +0200203/*
Christopher Faulet443ea1a2016-02-04 13:40:26 +0100204 * Structure representing the filter configuration, attached to a proxy and
205 * accessible from a filter when instantiated in a stream
206 */
207struct flt_conf {
208 const char *id; /* The filter id */
209 struct flt_ops *ops; /* The filter callbacks */
210 void *conf; /* The filter configuration */
211 struct list list; /* Next filter for the same proxy */
Christopher Faulet75bc9132018-11-30 15:18:09 +0100212 unsigned int flags; /* FLT_CFG_FL_* */
Christopher Faulet443ea1a2016-02-04 13:40:26 +0100213};
214
215/*
216 * Structure reprensenting a filter instance attached to a stream
Christopher Fauletd7c91962015-04-30 11:48:27 +0200217 *
218 * 2D-Array fields are used to store info per channel. The first index stands
219 * for the request channel, and the second one for the response channel.
220 * Especially, <next> and <fwd> are offets representing amount of data that the
221 * filter are, respectively, parsed and forwarded on a channel. Filters can
222 * access these values using FLT_NXT and FLT_FWD macros.
223 */
224struct filter {
Christopher Faulet443ea1a2016-02-04 13:40:26 +0100225 struct flt_conf *config; /* the filter's configuration */
Christopher Fauletd7c91962015-04-30 11:48:27 +0200226 void *ctx; /* The filter context (opaque) */
Christopher Fauletda02e172015-12-04 09:25:05 +0100227 unsigned short flags; /* FLT_FL_* */
Christopher Fauletb2e58492019-11-12 11:13:01 +0100228 unsigned long long offset[2]; /* Offset of input data already filtered for a specific channel
Christopher Fauletd7c91962015-04-30 11:48:27 +0200229 * 0: request channel, 1: response channel */
Christopher Faulet3a394fa2016-05-11 17:13:39 +0200230 unsigned int pre_analyzers; /* bit field indicating analyzers to pre-process */
231 unsigned int post_analyzers; /* bit field indicating analyzers to post-process */
Christopher Fauletd7c91962015-04-30 11:48:27 +0200232 struct list list; /* Next filter for the same proxy/stream */
233};
234
Christopher Fauletda02e172015-12-04 09:25:05 +0100235/*
236 * Structure reprensenting the "global" state of filters attached to a stream.
237 */
Christopher Fauletfcf035c2015-12-03 11:48:03 +0100238struct strm_flt {
Christopher Fauletda02e172015-12-04 09:25:05 +0100239 struct list filters; /* List of filters attached to a stream */
240 struct filter *current[2]; /* From which filter resume processing, for a specific channel.
241 * This is used for resumable callbacks only,
242 * If NULL, we start from the first filter.
243 * 0: request channel, 1: response channel */
244 unsigned short flags; /* STRM_FL_* */
Joseph Herlantb35ea682018-11-15 12:24:23 -0800245 unsigned char nb_req_data_filters; /* Number of data filters registered on the request channel */
246 unsigned char nb_rsp_data_filters; /* Number of data filters registered on the response channel */
Christopher Faulet75bc9132018-11-30 15:18:09 +0100247 unsigned long long offset[2];
Christopher Fauletfcf035c2015-12-03 11:48:03 +0100248};
249
Christopher Fauletd7c91962015-04-30 11:48:27 +0200250#endif /* _TYPES_FILTERS_H */
251
252/*
253 * Local variables:
254 * c-indent-level: 8
255 * c-basic-offset: 8
256 * End:
257 */