Files
vpp/src/plugins/http/http.h
Matus Fabian 8ca6ce6fe1 http: return more than data from server app
Server app could return headers in front of body/data buffer.
Offers apis for building and serialization of headers section.
HTTP layer now only add Date, Server and Content-Lengths headers,
rest is up to app. Well known header names are predefined.

Type: improvement

Change-Id: If778bdfc9acf6b0d11a48f0a745a3a56c96c2436
Signed-off-by: Matus Fabian <matfabia@cisco.com>
2024-07-23 15:22:34 +00:00

928 lines
32 KiB
C

/*
* Copyright (c) 2022 Cisco and/or its affiliates.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef SRC_PLUGINS_HTTP_HTTP_H_
#define SRC_PLUGINS_HTTP_HTTP_H_
#include <ctype.h>
#include <vnet/plugin/plugin.h>
#include <vpp/app/version.h>
#include <vppinfra/time_range.h>
#include <vnet/session/application_interface.h>
#include <vnet/session/application.h>
#include <http/http_buffer.h>
#define HTTP_DEBUG 0
#if HTTP_DEBUG
#define HTTP_DBG(_lvl, _fmt, _args...) \
if (_lvl <= HTTP_DEBUG) \
clib_warning (_fmt, ##_args)
#else
#define HTTP_DBG(_lvl, _fmt, _args...)
#endif
typedef struct http_conn_id_
{
union
{
session_handle_t app_session_handle;
u32 parent_app_api_ctx;
};
session_handle_t tc_session_handle;
u32 parent_app_wrk_index;
} http_conn_id_t;
STATIC_ASSERT (sizeof (http_conn_id_t) <= TRANSPORT_CONN_ID_LEN,
"ctx id must be less than TRANSPORT_CONN_ID_LEN");
typedef struct
{
char *base;
uword len;
} http_token_t;
#define http_token_lit(s) (s), sizeof (s) - 1
typedef enum http_conn_state_
{
HTTP_CONN_STATE_LISTEN,
HTTP_CONN_STATE_CONNECTING,
HTTP_CONN_STATE_ESTABLISHED,
HTTP_CONN_STATE_TRANSPORT_CLOSED,
HTTP_CONN_STATE_APP_CLOSED,
HTTP_CONN_STATE_CLOSED
} http_conn_state_t;
typedef enum http_state_
{
HTTP_STATE_IDLE = 0,
HTTP_STATE_WAIT_APP_METHOD,
HTTP_STATE_WAIT_CLIENT_METHOD,
HTTP_STATE_WAIT_SERVER_REPLY,
HTTP_STATE_WAIT_APP_REPLY,
HTTP_STATE_CLIENT_IO_MORE_DATA,
HTTP_STATE_APP_IO_MORE_DATA,
HTTP_N_STATES,
} http_state_t;
typedef enum http_req_method_
{
HTTP_REQ_GET = 0,
HTTP_REQ_POST,
} http_req_method_t;
typedef enum http_msg_type_
{
HTTP_MSG_REQUEST,
HTTP_MSG_REPLY
} http_msg_type_t;
typedef enum http_target_form_
{
HTTP_TARGET_ORIGIN_FORM,
HTTP_TARGET_ABSOLUTE_FORM,
HTTP_TARGET_AUTHORITY_FORM,
HTTP_TARGET_ASTERISK_FORM
} http_target_form_t;
#define foreach_http_content_type \
_ (APP_7Z, ".7z", "application / x - 7z - compressed") \
_ (APP_DOC, ".doc", "application / msword") \
_ (APP_DOCX, ".docx", \
"application / vnd.openxmlformats - " \
"officedocument.wordprocessingml.document") \
_ (APP_EPUB, ".epub", "application / epub + zip") \
_ (APP_FONT, ".eot", "application / vnd.ms - fontobject") \
_ (APP_JAR, ".jar", "application / java - archive") \
_ (APP_JSON, ".json", "application / json") \
_ (APP_JSON_LD, ".jsonld", "application / ld + json") \
_ (APP_MPKG, ".mpkg", "application / vnd.apple.installer + xml") \
_ (APP_ODP, ".odp", "application / vnd.oasis.opendocument.presentation") \
_ (APP_ODS, ".ods", "application / vnd.oasis.opendocument.spreadsheet") \
_ (APP_ODT, ".odt", "application / vnd.oasis.opendocument.text") \
_ (APP_OGX, ".ogx", "application / ogg") \
_ (APP_PDF, ".pdf", "application / pdf") \
_ (APP_PHP, ".php", "application / x - httpd - php") \
_ (APP_PPT, ".ppt", "application / vnd.ms - powerpoint") \
_ (APP_PPTX, ".pptx", "application / vnd.ms - powerpoint") \
_ (APP_RAR, ".rar", "application / vnd.rar") \
_ (APP_RTF, ".rtf", "application / rtf") \
_ (APP_SH, ".sh", "application / x - sh") \
_ (APP_TAR, ".tar", "application / x - tar") \
_ (APP_VSD, ".vsd", "application / vnd.visio") \
_ (APP_XHTML, ".xhtml", "application / xhtml + xml") \
_ (APP_XLS, ".xls", "application / vnd.ms - excel") \
_ (APP_XML, ".xml", "application / xml") \
_ (APP_XSLX, ".xlsx", \
"application / vnd.openxmlformats - officedocument.spreadsheetml.sheet") \
_ (APP_XUL, ".xul", "application / vnd.mozilla.xul + xml") \
_ (APP_ZIP, ".zip", "application / zip") \
_ (AUDIO_AAC, ".aac", "audio / aac") \
_ (AUDIO_CD, ".cda", "application / x - cdf") \
_ (AUDIO_WAV, ".wav", "audio / wav") \
_ (AUDIO_WEBA, ".weba", "audio / webm") \
_ (AUDO_MIDI, ".midi", "audio / midi") \
_ (AUDO_MID, ".mid", "audo / midi") \
_ (AUDO_MP3, ".mp3", "audio / mpeg") \
_ (AUDO_OGA, ".oga", "audio / ogg") \
_ (AUDO_OPUS, ".opus", "audio / opus") \
_ (APP_OCTET_STREAM, ".bin", "application / octet - stream") \
_ (BZIP2, ".bz2", "application / x - bzip2") \
_ (BZIP, ".bz", "application / x - bzip") \
_ (FONT_OTF, ".otf", "font / otf") \
_ (FONT_TTF, ".ttf", "font / ttf") \
_ (FONT_WOFF2, ".woff2", "font / woff2") \
_ (FONT_WOFF, ".woff", "font / woff") \
_ (GZIP, ".gz", "application / gzip") \
_ (IMAGE_AVIF, ".avif", "image / avif") \
_ (IMAGE_BMP, ".bmp", "image / bmp") \
_ (IMAGE_GIF, ".gif", "image / gif") \
_ (IMAGE_ICON, ".ico", "image / vnd.microsoft.icon") \
_ (IMAGE_JPEG, ".jpeg", "image / jpeg") \
_ (IMAGE_JPG, ".jpg", "image / jpeg") \
_ (IMAGE_PNG, ".png", "image / png") \
_ (IMAGE_SVG, ".svg", "image / svg + xml") \
_ (IMAGE_TIFF, ".tiff", "image / tiff") \
_ (IMAGE_TIF, ".tif", "image / tiff") \
_ (IMAGE_WEBP, ".webp", "image / webp") \
_ (SCRIPT_CSH, ".csh", "application / x - csh") \
_ (TEXT_ABIWORD, ".abw", "application / x - abiword") \
_ (TEXT_ARCHIVE, ".arc", "application / x - freearc") \
_ (TEXT_AZW, ".azw", "application / vnd.amazon.ebook") \
_ (TEXT_CALENDAR, ".ics", "text / calendar") \
_ (TEXT_CSS, ".css", "text / css") \
_ (TEXT_CSV, ".csv", "text / csv") \
_ (TEXT_HTM, ".htm", "text / html") \
_ (TEXT_HTML, ".html", "text / html") \
_ (TEXT_JS, ".js", "text / javascript") \
_ (TEXT_MJS, ".mjs", "text / javascript") \
_ (TEXT_PLAIN, ".txt", "text / plain") \
_ (VIDEO_3GP2, ".3g2", "video / 3gpp2") \
_ (VIDEO_3GP, ".3gp", "video / 3gpp") \
_ (VIDEO_AVI, ".avi", "video / x - msvideo") \
_ (VIDEO_MP4, ".mp4", "video / mp4") \
_ (VIDEO_MPEG, ".mpeg", "video / mpeg") \
_ (VIDEO_OGG, ".ogv", "video / ogg") \
_ (VIDEO_TS, ".ts", "video / mp2t") \
_ (VIDEO_WEBM, ".webm", "video / webm")
typedef enum http_content_type_
{
#define _(s, ext, str) HTTP_CONTENT_##s,
foreach_http_content_type
#undef _
} http_content_type_t;
#define foreach_http_status_code \
_ (100, CONTINUE, "100 Continue") \
_ (101, SWITCHING_PROTOCOLS, "101 Switching Protocols") \
_ (200, OK, "200 OK") \
_ (201, CREATED, "201 Created") \
_ (202, ACCEPTED, "202 Accepted") \
_ (203, NON_UTHORITATIVE_INFORMATION, "203 Non-Authoritative Information") \
_ (204, NO_CONTENT, "204 No Content") \
_ (205, RESET_CONTENT, "205 Reset Content") \
_ (206, PARTIAL_CONTENT, "206 Partial Content") \
_ (300, MULTIPLE_CHOICES, "300 Multiple Choices") \
_ (301, MOVED, "301 Moved Permanently") \
_ (302, FOUND, "302 Found") \
_ (303, SEE_OTHER, "303 See Other") \
_ (304, NOT_MODIFIED, "304 Not Modified") \
_ (305, USE_PROXY, "305 Use Proxy") \
_ (307, TEMPORARY_REDIRECT, "307 Temporary Redirect") \
_ (308, PERMANENT_REDIRECT, "308 Permanent Redirect") \
_ (400, BAD_REQUEST, "400 Bad Request") \
_ (401, UNAUTHORIZED, "401 Unauthorized") \
_ (402, PAYMENT_REQUIRED, "402 Payment Required") \
_ (403, FORBIDDEN, "403 Forbidden") \
_ (404, NOT_FOUND, "404 Not Found") \
_ (405, METHOD_NOT_ALLOWED, "405 Method Not Allowed") \
_ (406, NOT_ACCEPTABLE, "406 Not Acceptable") \
_ (407, PROXY_AUTHENTICATION_REQUIRED, "407 Proxy Authentication Required") \
_ (408, REQUEST_TIMEOUT, "408 Request Timeout") \
_ (409, CONFLICT, "409 Conflict") \
_ (410, GONE, "410 Gone") \
_ (411, LENGTH_REQUIRED, "411 Length Required") \
_ (412, PRECONDITION_FAILED, "412 Precondition Failed") \
_ (413, CONTENT_TOO_LARGE, "413 Content Too Large") \
_ (414, URI_TOO_LONG, "414 URI Too Long") \
_ (415, UNSUPPORTED_MEDIA_TYPE, "415 Unsupported Media Type") \
_ (416, RANGE_NOT_SATISFIABLE, "416 Range Not Satisfiable") \
_ (417, EXPECTATION_FAILED, "417 Expectation Failed") \
_ (421, MISDIRECTED_REQUEST, "421 Misdirected Request") \
_ (422, UNPROCESSABLE_CONTENT, "422 Unprocessable_Content") \
_ (426, UPGRADE_REQUIRED, "426 Upgrade Required") \
_ (500, INTERNAL_ERROR, "500 Internal Server Error") \
_ (501, NOT_IMPLEMENTED, "501 Not Implemented") \
_ (502, BAD_GATEWAY, "502 Bad Gateway") \
_ (503, SERVICE_UNAVAILABLE, "503 Service Unavailable") \
_ (504, GATEWAY_TIMEOUT, "504 Gateway Timeout") \
_ (505, HTTP_VERSION_NOT_SUPPORTED, "505 HTTP Version Not Supported")
typedef enum http_status_code_
{
#define _(c, s, str) HTTP_STATUS_##s,
foreach_http_status_code
#undef _
HTTP_N_STATUS
} http_status_code_t;
#define foreach_http_header_name \
_ (ACCEPT, "Accept") \
_ (ACCEPT_CHARSET, "Accept-Charset") \
_ (ACCEPT_ENCODING, "Accept-Encoding") \
_ (ACCEPT_LANGUAGE, "Accept-Language") \
_ (ACCEPT_RANGES, "Accept-Ranges") \
_ (ACCESS_CONTROL_ALLOW_CREDENTIALS, "Access-Control-Allow-Credentials") \
_ (ACCESS_CONTROL_ALLOW_HEADERS, "Access-Control-Allow-Headers") \
_ (ACCESS_CONTROL_ALLOW_METHODS, "Access-Control-Allow-Methods") \
_ (ACCESS_CONTROL_ALLOW_ORIGIN, "Access-Control-Allow-Origin") \
_ (ACCESS_CONTROL_EXPOSE_HEADERS, "Access-Control-Expose-Headers") \
_ (ACCESS_CONTROL_MAX_AGE, "Access-Control-Max-Age") \
_ (ACCESS_CONTROL_REQUEST_HEADERS, "Access-Control-Request-Headers") \
_ (ACCESS_CONTROL_REQUEST_METHOD, "Access-Control-Request-Method") \
_ (AGE, "Age") \
_ (ALLOW, "Allow") \
_ (ALPN, "ALPN") \
_ (ALT_SVC, "Alt-Svc") \
_ (ALT_USED, "Alt-Used") \
_ (ALTERNATES, "Alternates") \
_ (AUTHENTICATION_CONTROL, "Authentication-Control") \
_ (AUTHENTICATION_INFO, "Authentication-Info") \
_ (AUTHORIZATION, "Authorization") \
_ (CACHE_CONTROL, "Cache-Control") \
_ (CACHE_STATUS, "Cache-Status") \
_ (CAPSULE_PROTOCOL, "Capsule-Protocol") \
_ (CDN_CACHE_CONTROL, "CDN-Cache-Control") \
_ (CDN_LOOP, "CDN-Loop") \
_ (CLIENT_CERT, "Client-Cert") \
_ (CLIENT_CERT_CHAIN, "Client-Cert-Chain") \
_ (CLOSE, "Close") \
_ (CONNECTION, "Connection") \
_ (CONTENT_DIGEST, "Content-Digest") \
_ (CONTENT_DISPOSITION, "Content-Disposition") \
_ (CONTENT_ENCODING, "Content-Encoding") \
_ (CONTENT_LANGUAGE, "Content-Language") \
_ (CONTENT_LENGTH, "Content-Length") \
_ (CONTENT_LOCATION, "Content-Location") \
_ (CONTENT_RANGE, "Content-Range") \
_ (CONTENT_TYPE, "Content-Type") \
_ (COOKIE, "Cookie") \
_ (DATE, "Date") \
_ (DIGEST, "Digest") \
_ (DPOP, "DPoP") \
_ (DPOP_NONCE, "DPoP-Nonce") \
_ (EARLY_DATA, "Early-Data") \
_ (ETAG, "ETag") \
_ (EXPECT, "Expect") \
_ (EXPIRES, "Expires") \
_ (FORWARDED, "Forwarded") \
_ (FROM, "From") \
_ (HOST, "Host") \
_ (IF_MATCH, "If-Match") \
_ (IF_MODIFIED_SINCE, "If-Modified-Since") \
_ (IF_NONE_MATCH, "If-None-Match") \
_ (IF_RANGE, "If-Range") \
_ (IF_UNMODIFIED_SINCE, "If-Unmodified-Since") \
_ (KEEP_ALIVE, "Keep-Alive") \
_ (LAST_MODIFIED, "Last-Modified") \
_ (LINK, "Link") \
_ (LOCATION, "Location") \
_ (MAX_FORWARDS, "Max-Forwards") \
_ (ORIGIN, "Origin") \
_ (PRIORITY, "Priority") \
_ (PROXY_AUTHENTICATE, "Proxy-Authenticate") \
_ (PROXY_AUTHENTICATION_INFO, "Proxy-Authentication-Info") \
_ (PROXY_AUTHORIZATION, "Proxy-Authorization") \
_ (PROXY_STATUS, "Proxy-Status") \
_ (RANGE, "Range") \
_ (REFERER, "Referer") \
_ (REPR_DIGEST, "Repr-Digest") \
_ (SET_COOKIE, "Set-Cookie") \
_ (SIGNATURE, "Signature") \
_ (SIGNATURE_INPUT, "Signature-Input") \
_ (STRICT_TRANSPORT_SECURITY, "Strict-Transport-Security") \
_ (RETRY_AFTER, "Retry-After") \
_ (SERVER, "Server") \
_ (TE, "TE") \
_ (TRAILER, "Trailer") \
_ (TRANSFER_ENCODING, "Transfer-Encoding") \
_ (UPGRADE, "Upgrade") \
_ (USER_AGENT, "User-Agent") \
_ (VARY, "Vary") \
_ (VIA, "Via") \
_ (WANT_CONTENT_DIGEST, "Want-Content-Digest") \
_ (WANT_REPR_DIGEST, "Want-Repr-Digest") \
_ (WWW_AUTHENTICATE, "WWW-Authenticate")
typedef enum http_header_name_
{
#define _(sym, str) HTTP_HEADER_##sym,
foreach_http_header_name
#undef _
} http_header_name_t;
typedef enum http_msg_data_type_
{
HTTP_MSG_DATA_INLINE,
HTTP_MSG_DATA_PTR
} http_msg_data_type_t;
typedef struct http_msg_data_
{
http_msg_data_type_t type;
u64 len;
http_target_form_t target_form;
u32 target_path_offset;
u32 target_path_len;
u32 target_query_offset;
u32 target_query_len;
u32 headers_offset;
u32 headers_len;
u32 body_offset;
u32 body_len;
u8 data[0];
} http_msg_data_t;
typedef struct http_msg_
{
http_msg_type_t type;
union
{
http_req_method_t method_type;
http_status_code_t code;
};
http_content_type_t content_type;
http_msg_data_t data;
} http_msg_t;
typedef struct http_tc_
{
union
{
transport_connection_t connection;
http_conn_id_t c_http_conn_id;
};
#define h_tc_session_handle c_http_conn_id.tc_session_handle
#define h_pa_wrk_index c_http_conn_id.parent_app_wrk_index
#define h_pa_session_handle c_http_conn_id.app_session_handle
#define h_pa_app_api_ctx c_http_conn_id.parent_app_api_ctx
#define h_hc_index connection.c_index
http_conn_state_t state;
u32 timer_handle;
u8 *app_name;
/*
* Current request
*/
http_state_t http_state;
http_req_method_t method;
u8 *rx_buf;
u32 rx_buf_offset;
http_buffer_t tx_buf;
u32 to_recv;
u32 bytes_dequeued;
http_target_form_t target_form;
u32 target_path_offset;
u32 target_path_len;
u32 target_query_offset;
u32 target_query_len;
u32 headers_offset;
u32 headers_len;
u32 body_offset;
u32 body_len;
} http_conn_t;
typedef struct http_worker_
{
http_conn_t *conn_pool;
} http_worker_t;
typedef struct http_main_
{
http_worker_t *wrk;
http_conn_t *listener_pool;
u32 app_index;
clib_timebase_t timebase;
/*
* Runtime config
*/
u8 debug_level;
u8 is_init;
/*
* Config
*/
u64 first_seg_size;
u64 add_seg_size;
u32 fifo_size;
} http_main_t;
always_inline int
_validate_target_syntax (u8 *target, int is_query, int *is_encoded)
{
int i, encoded = 0;
static uword valid_chars[4] = {
/* !$&'()*+,-./0123456789:;= */
0x2fffffd200000000,
/* @ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz~ */
0x47fffffe87ffffff,
0x0000000000000000,
0x0000000000000000,
};
for (i = 0; i < vec_len (target); i++)
{
if (clib_bitmap_get_no_check (valid_chars, target[i]))
continue;
/* target was already split after first question mark,
* for query it is valid character */
if (is_query && target[i] == '?')
continue;
/* pct-encoded = "%" HEXDIG HEXDIG */
if (target[i] == '%')
{
if ((i + 2) > vec_len (target))
return -1;
if (!isxdigit (target[i + 1]) || !isxdigit (target[i + 2]))
return -1;
i += 2;
encoded = 1;
continue;
}
clib_warning ("invalid character %d", target[i]);
return -1;
}
if (is_encoded)
*is_encoded = encoded;
return 0;
}
/**
* An "absolute-path" rule validation (RFC9110 section 4.1).
*
* @param path Target path to validate.
* @param is_encoded Return flag that indicates if percent-encoded (optional).
*
* @return @c 0 on success.
*/
always_inline int
http_validate_abs_path_syntax (u8 *path, int *is_encoded)
{
return _validate_target_syntax (path, 0, is_encoded);
}
/**
* A "query" rule validation (RFC3986 section 2.1).
*
* @param query Target query to validate.
* @param is_encoded Return flag that indicates if percent-encoded (optional).
*
* @return @c 0 on success.
*/
always_inline int
http_validate_query_syntax (u8 *query, int *is_encoded)
{
return _validate_target_syntax (query, 1, is_encoded);
}
#define htoi(x) (isdigit (x) ? (x - '0') : (tolower (x) - 'a' + 10))
/**
* Decode percent-encoded data.
*
* @param src Data to decode.
*
* @return New vector with decoded data.
*
* The caller is always responsible to free the returned vector.
*/
always_inline u8 *
http_percent_decode (u8 *src)
{
int i;
u8 *decoded_uri = 0;
for (i = 0; i < vec_len (src); i++)
{
if (src[i] == '%')
{
u8 c = (htoi (src[i + 1]) << 4) | htoi (src[i + 2]);
vec_add1 (decoded_uri, c);
i += 2;
}
else
vec_add1 (decoded_uri, src[i]);
}
return decoded_uri;
}
/**
* Remove dot segments from path (RFC3986 section 5.2.4)
*
* @param path Path to sanitize.
*
* @return New vector with sanitized path.
*
* The caller is always responsible to free the returned vector.
*/
always_inline u8 *
http_path_remove_dot_segments (u8 *path)
{
u32 *segments = 0, *segments_len = 0, segment_len;
u8 *new_path = 0;
int i, ii;
if (!path)
return vec_new (u8, 0);
segments = vec_new (u32, 1);
/* first segment */
segments[0] = 0;
/* find all segments */
for (i = 1; i < (vec_len (path) - 1); i++)
{
if (path[i] == '/')
vec_add1 (segments, i + 1);
}
/* dummy tail */
vec_add1 (segments, vec_len (path));
/* scan all segments for "." and ".." */
segments_len = vec_new (u32, vec_len (segments) - 1);
for (i = 0; i < vec_len (segments_len); i++)
{
segment_len = segments[i + 1] - segments[i];
if (segment_len == 2 && path[segments[i]] == '.')
segment_len = 0;
else if (segment_len == 3 && path[segments[i]] == '.' &&
path[segments[i] + 1] == '.')
{
segment_len = 0;
/* remove parent (if any) */
for (ii = i - 1; ii >= 0; ii--)
{
if (segments_len[ii])
{
segments_len[ii] = 0;
break;
}
}
}
segments_len[i] = segment_len;
}
/* we might end with empty path, so return at least empty vector */
new_path = vec_new (u8, 0);
/* append all valid segments */
for (i = 0; i < vec_len (segments_len); i++)
{
if (segments_len[i])
vec_add (new_path, path + segments[i], segments_len[i]);
}
vec_free (segments);
vec_free (segments_len);
return new_path;
}
always_inline int
_parse_field_name (u8 **pos, u8 *end, u8 **field_name_start,
u32 *field_name_len)
{
u32 name_len = 0;
u8 *p;
static uword tchar[4] = {
/* !#$%'*+-.0123456789 */
0x03ff6cba00000000,
/* ABCDEFGHIJKLMNOPQRSTUVWXYZ^_`abcdefghijklmnopqrstuvwxyz|~ */
0x57ffffffc7fffffe,
0x0000000000000000,
0x0000000000000000,
};
p = *pos;
*field_name_start = p;
while (p != end)
{
if (clib_bitmap_get_no_check (tchar, *p))
{
name_len++;
p++;
}
else if (*p == ':')
{
if (name_len == 0)
{
clib_warning ("empty field name");
return -1;
}
*field_name_len = name_len;
p++;
*pos = p;
return 0;
}
else
{
clib_warning ("invalid character %d", *p);
return -1;
}
}
clib_warning ("field name end not found");
return -1;
}
always_inline int
_parse_field_value (u8 **pos, u8 *end, u8 **field_value_start,
u32 *field_value_len)
{
u32 value_len = 0;
u8 *p;
p = *pos;
/* skip leading whitespace */
while (1)
{
if (p == end)
{
clib_warning ("field value not found");
return -1;
}
else if (*p != ' ' && *p != '\t')
{
break;
}
p++;
}
*field_value_start = p;
while (p != end)
{
if (*p == '\r')
{
if ((end - p) < 1)
{
clib_warning ("incorrect field line end");
return -1;
}
p++;
if (*p == '\n')
{
if (value_len == 0)
{
clib_warning ("empty field value");
return -1;
}
p++;
*pos = p;
/* skip trailing whitespace */
p = *field_value_start + value_len - 1;
while (*p == ' ' || *p == '\t')
{
p--;
value_len--;
}
*field_value_len = value_len;
return 0;
}
clib_warning ("CR without LF");
return -1;
}
if (*p < ' ' && *p != '\t')
{
clib_warning ("invalid character %d", *p);
return -1;
}
p++;
value_len++;
}
clib_warning ("field value end not found");
return -1;
}
typedef struct
{
u8 *name;
u8 *value;
} http_header_ht_t;
typedef struct
{
http_token_t name;
http_token_t value;
} http_header_t;
typedef struct
{
http_header_ht_t *headers;
uword *value_by_name;
} http_header_table_t;
/**
* Free header table's memory.
*
* @param ht Header table to free.
*/
always_inline void
http_free_header_table (http_header_table_t *ht)
{
http_header_ht_t *header;
vec_foreach (header, ht->headers)
{
vec_free (header->name);
vec_free (header->value);
}
vec_free (ht->headers);
hash_free (ht->value_by_name);
clib_mem_free (ht);
}
/**
* Parse headers in given vector.
*
* @param headers Vector to parse.
* @param [out] header_table Parsed headers in case of success.
*
* @return @c 0 on success.
*
* The caller is responsible to free the returned @c header_table
* using @c http_free_header_table .
*/
always_inline int
http_parse_headers (u8 *headers, http_header_table_t **header_table)
{
u8 *pos, *end, *name_start, *value_start, *name;
u32 name_len, value_len;
int rv;
http_header_ht_t *header;
http_header_table_t *ht;
uword *p;
end = headers + vec_len (headers);
pos = headers;
ht = clib_mem_alloc (sizeof (*ht));
ht->value_by_name = hash_create_string (0, sizeof (uword));
ht->headers = 0;
do
{
rv = _parse_field_name (&pos, end, &name_start, &name_len);
if (rv != 0)
{
http_free_header_table (ht);
return rv;
}
rv = _parse_field_value (&pos, end, &value_start, &value_len);
if (rv != 0)
{
http_free_header_table (ht);
return rv;
}
name = vec_new (u8, name_len);
clib_memcpy (name, name_start, name_len);
vec_terminate_c_string (name);
/* check if header is repeated */
p = hash_get_mem (ht->value_by_name, name);
if (p)
{
/* if yes combine values */
header = vec_elt_at_index (ht->headers, p[0]);
vec_pop (header->value); /* drop null byte */
header->value = format (header->value, ", %U%c", format_ascii_bytes,
value_start, value_len, 0);
vec_free (name);
continue;
}
/* or create new record */
vec_add2 (ht->headers, header, sizeof (*header));
header->name = name;
header->value = vec_new (u8, value_len);
clib_memcpy (header->value, value_start, value_len);
vec_terminate_c_string (header->value);
hash_set_mem (ht->value_by_name, header->name, header - ht->headers);
}
while (pos != end);
*header_table = ht;
return 0;
}
/**
* Try to find given header name in header table.
*
* @param header_table Header table to search.
* @param name Header name to match.
*
* @return Header's value in case of success, @c 0 otherwise.
*/
always_inline const char *
http_get_header (http_header_table_t *header_table, const char *name)
{
uword *p;
http_header_ht_t *header;
p = hash_get_mem (header_table->value_by_name, name);
if (p)
{
header = vec_elt_at_index (header_table->headers, p[0]);
return (const char *) header->value;
}
return 0;
}
/**
* Add header to the list.
*
* @param headers Header list.
* @param name Pointer to header's name buffer.
* @param name_len Length of the name.
* @param value Pointer to header's value buffer.
* @param value_len Length of the value.
*
* @note Headers added at protocol layer: Date, Server, Content-Length
*/
always_inline void
http_add_header (http_header_t **headers, const char *name, uword name_len,
const char *value, uword value_len)
{
http_header_t *header;
vec_add2 (*headers, header, 1);
header->name.base = (char *) name;
header->name.len = name_len;
header->value.base = (char *) value;
header->value.len = value_len;
}
/**
* Serialize the header list.
*
* @param headers Header list to serialize.
*
* @return New vector with serialized headers.
*
* The caller is always responsible to free the returned vector.
*/
always_inline u8 *
http_serialize_headers (http_header_t *headers)
{
u8 *headers_buf = 0, *dst;
u32 headers_buf_len = 2;
http_header_t *header;
vec_foreach (header, headers)
headers_buf_len += header->name.len + header->value.len + 4;
vec_validate (headers_buf, headers_buf_len - 1);
dst = headers_buf;
vec_foreach (header, headers)
{
clib_memcpy (dst, header->name.base, header->name.len);
dst += header->name.len;
*dst++ = ':';
*dst++ = ' ';
clib_memcpy (dst, header->value.base, header->value.len);
dst += header->value.len;
*dst++ = '\r';
*dst++ = '\n';
}
*dst++ = '\r';
*dst = '\n';
return headers_buf;
}
#endif /* SRC_PLUGINS_HTTP_HTTP_H_ */
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/