1
0
mirror of https://github.com/stefanocasazza/ULib.git synced 2025-10-26 19:57:22 +08:00
ULib/include/ulib/internal/chttp.h
stefanocasazza dfbea95583 sync
2019-04-02 15:38:18 +02:00

343 lines
16 KiB
C

/* ============================================================================
*
* LIBRARY
* ULib - c++ library
*
* FILENAME
* chttp.h - HTTP definition for C binding
*
* AUTHOR
* Stefano Casazza
*
* ============================================================================ */
#ifndef ULIB_CHTTP_H
#define ULIB_CHTTP_H 1
/**
* -------------------------------------------------------------------------------------------------------
* _ _ _
* | |__ | |_| |_ _ __
* | '_ \| __| __| '_ \
* | | | | |_| |_| |_) |
* |_| |_|\__|\__| .__/
* |_|
*
* ---------------------------------------------------------------------------------------------------------
* HTTP message handler
*
* The status code is a three-digit integer, and the first digit identifies the general category of response
* ---------------------------------------------------------------------------------------------------------
*/
/* 1xx indicates an informational message only */
#define HTTP_CONTINUE 100
#define HTTP_SWITCH_PROT 101
#define HTTP_STATUS_102 102 /* "Processing" */
/* 2xx indicates success of some kind */
#define HTTP_OK 200
#define HTTP_CREATED 201
#define HTTP_ACCEPTED 202
#define HTTP_NOT_AUTHORITATIVE 203
#define HTTP_NO_CONTENT 204
#define HTTP_RESET 205
#define HTTP_PARTIAL 206
#define HTTP_STATUS_207 207 /* "Multi-Status" */
#define HTTP_OPTIONS_RESPONSE 222 /* only internal use, not standard */
/* 3xx redirects the client to another URL */
#define HTTP_MULT_CHOICE 300
#define HTTP_MOVED_PERM 301
#define HTTP_MOVED_TEMP 302
#define HTTP_FOUND 302
#define HTTP_SEE_OTHER 303
#define HTTP_NOT_MODIFIED 304
#define HTTP_USE_PROXY 305
#define HTTP_TEMP_REDIR 307
#define HTTP_PERMANENT_REDIRECT 308
/* 4xx indicates an error on the client's part */
#define HTTP_BAD_REQUEST 400
#define HTTP_UNAUTHORIZED 401
#define HTTP_PAYMENT_REQUIRED 402
#define HTTP_FORBIDDEN 403
#define HTTP_NOT_FOUND 404
#define HTTP_BAD_METHOD 405
#define HTTP_NOT_ACCEPTABLE 406
#define HTTP_PROXY_AUTH 407
#define HTTP_CLIENT_TIMEOUT 408
#define HTTP_CONFLICT 409
#define HTTP_GONE 410
#define HTTP_LENGTH_REQUIRED 411
#define HTTP_PRECON_FAILED 412
#define HTTP_ENTITY_TOO_LARGE 413
#define HTTP_REQ_TOO_LONG 414
#define HTTP_UNSUPPORTED_TYPE 415
#define HTTP_REQ_RANGE_NOT_OK 416
#define HTTP_EXPECTATION_FAILED 417
#define HTTP_STATUS_418 418 /* "I'm a teapot" */
#define HTTP_UNPROCESSABLE_ENTITY 422
#define HTTP_STATUS_423 423 /* "Locked" */
#define HTTP_STATUS_424 424 /* "Failed Dependency" */
#define HTTP_STATUS_425 425 /* "Unordered Collection" */
#define HTTP_STATUS_426 426 /* "Upgrade Required" */
#define HTTP_PRECONDITION_REQUIRED 428
#define HTTP_TOO_MANY_REQUESTS 429
#define HTTP_REQUEST_HEADER_FIELDS_TOO_LARGE 431
/* 5xx indicates an error on the server's part */
#define HTTP_INTERNAL_ERROR 500
#define HTTP_NOT_IMPLEMENTED 501
#define HTTP_BAD_GATEWAY 502
#define HTTP_UNAVAILABLE 503
#define HTTP_GATEWAY_TIMEOUT 504
#define HTTP_VERSION 505
#define HTTP_STATUS_506 506 /* "Variant Also Negotiates" */
#define HTTP_STATUS_507 507 /* "Insufficient Storage" */
#define HTTP_STATUS_509 509 /* "Bandwidth Limit Exceeded" */
#define HTTP_STATUS_510 510 /* "Not Extended" */
#define HTTP_NETWORK_AUTHENTICATION_REQUIRED 511
#define U_IS_HTTP_INFO(x) (((x)>=100)&&((x)<200)) /* is the status code informational */
#define U_IS_HTTP_SUCCESS(x) (((x)>=200)&&((x)<300)) /* is the status code OK ? */
#define U_IS_HTTP_REDIRECT(x) (((x)>=300)&&((x)<400)) /* is the status code a redirect */
#define U_IS_HTTP_ERROR(x) (((x)>=400)&&((x)<600)) /* is the status code a error (client or server) */
#define U_IS_HTTP_CLIENT_ERROR(x) (((x)>=400)&&((x)<500)) /* is the status code a client error */
#define U_IS_HTTP_SERVER_ERROR(x) (((x)>=500)&&((x)<600)) /* is the status code a server error */
#define U_IS_HTTP_VALID_RESPONSE(x) (((x)>=100)&&((x)<600)) /* is the status code a (potentially) valid response code ? */
/* should the status code drop the connection ? */
#define U_STATUS_DROPS_CONNECTION(x) \
(((x) == HTTP_MOVED_TEMP) || \
((x) == HTTP_BAD_REQUEST) || \
((x) == HTTP_CLIENT_TIMEOUT) || \
((x) == HTTP_LENGTH_REQUIRED) || \
((x) == HTTP_PRECON_FAILED) || \
((x) == HTTP_ENTITY_TOO_LARGE) || \
((x) == HTTP_REQ_TOO_LONG) || \
((x) == HTTP_INTERNAL_ERROR) || \
((x) == HTTP_UNAVAILABLE) || \
((x) == HTTP_NETWORK_AUTHENTICATION_REQUIRED) || \
((x) == HTTP_NOT_IMPLEMENTED))
/**
* HTTP header representation
*
* sizeof(struct uhttpinfo) 64bit == 144
*/
typedef struct uhttpinfo {
const char* uri;
const char* query;
const char* host;
const char* range;
const char* cookie;
const char* accept;
const char* referer;
const char* ip_client;
const char* user_agent;
const char* content_type;
const char* accept_language;
/* RESET == 52 */
uint16_t nResponseCode, cookie_len, referer_len, user_agent_len;
uint32_t if_modified_since, startHeader, endHeader, clength, uri_len, query_len, method_type;
unsigned char flag[16];
} uhttpinfo;
/**
* GET is used to retrieve a resource on the server.
* HEAD is used to retrieve some information about the document, but don't need the document itself.
* POST says that you are providing some information of your own (i.e., in forms). This may change the state of the server in some way, such as creating a record in a database.
* PUT is used to replace or create a new document on the server.
* DELETE is used to remove a document on the server.
* TRACE is used for protocol debugging purposes.
* OPTIONS is used when the client looks for other methods which can be used on the document.
* CONNECT is used when a client needs to talk to an HTTPS server through a proxy server.
*
* Other HTTP methods that you may see (LINK, UNLINK, and PATCH) are less clearly defined
*/
enum HTTPMethodType {
/* request methods */
HTTP_GET = 0x00000001,
HTTP_HEAD = 0x00000002,
HTTP_POST = 0x00000004,
HTTP_PUT = 0x00000008,
HTTP_DELETE = 0x00000010,
HTTP_OPTIONS = 0x00000020,
/* pathological */
HTTP_TRACE = 0x00000040,
HTTP_CONNECT = 0x00000080,
/* webdav */
HTTP_COPY = 0x00000100,
HTTP_MOVE = 0x00000200,
HTTP_LOCK = 0x00000400,
HTTP_UNLOCK = 0x00000800,
HTTP_MKCOL = 0x00001000,
HTTP_SEARCH = 0x00002000,
HTTP_PROPFIND = 0x00004000,
HTTP_PROPPATCH = 0x00008000,
/* rfc-5789 */
HTTP_PATCH = 0x00010000,
HTTP_PURGE = 0x00020000,
/* subversion */
HTTP_MERGE = 0x00040000,
HTTP_REPORT = 0x00080000,
HTTP_CHECKOUT = 0x00100000,
HTTP_MKACTIVITY = 0x00200000,
/* upnp */
HTTP_NOTIFY = 0x00400000,
HTTP_MSEARCH = 0x00800000,
HTTP_SUBSCRIBE = 0x01000000,
HTTP_UNSUBSCRIBE = 0x02000000
};
typedef struct uhttpmethodtype {
const char* name;
uint32_t len;
} uhttpmethodtype;
typedef struct uclientimage_info {
union uucflag64 flag;
struct uhttpmethodtype http_method_list[26];
struct uhttpinfo http_info;
} uclientimage_info;
#ifdef __cplusplus
extern "C" {
#endif
extern U_EXPORT uclientimage_info u_clientimage_info;
#ifdef __cplusplus
}
#endif
#define U_PARALLELIZATION_CHILD 1
#define U_PARALLELIZATION_PARENT 2
#define U_http_info u_clientimage_info.http_info
#define U_http_method_list u_clientimage_info.http_method_list
#define U_http_method_type u_clientimage_info.http_info.method_type
#define U_line_terminator_len u_clientimage_info.flag.c[0]
#define U_ClientImage_state u_clientimage_info.flag.c[1]
#define U_ClientImage_close u_clientimage_info.flag.c[2]
#define U_ClientImage_request u_clientimage_info.flag.c[3]
#define U_ClientImage_pipeline u_clientimage_info.flag.c[4]
#define U_ClientImage_data_missing u_clientimage_info.flag.c[5]
#define U_ClientImage_parallelization u_clientimage_info.flag.c[6]
#define U_ClientImage_advise_for_parallelization u_clientimage_info.flag.c[7]
#define U_http_version u_clientimage_info.http_info.flag[ 0]
#define U_http_method_num u_clientimage_info.http_info.flag[ 1]
#define U_http_host_len u_clientimage_info.http_info.flag[ 2]
#define U_http_host_vlen u_clientimage_info.http_info.flag[ 3]
#define U_http_range_len u_clientimage_info.http_info.flag[ 4]
#define U_http_accept_len u_clientimage_info.http_info.flag[ 5]
#define U_http_uri_offset u_clientimage_info.http_info.flag[ 6]
#define U_http_websocket_len u_clientimage_info.http_info.flag[ 7]
#define U_http2_settings_len u_clientimage_info.http_info.flag[ 8]
#define U_http_ip_client_len u_clientimage_info.http_info.flag[ 9]
#define U_http_content_type_len u_clientimage_info.http_info.flag[10]
#define U_http_accept_language_len u_clientimage_info.http_info.flag[11]
#define U_http_flag u_clientimage_info.http_info.flag[12]
#define U_http_usp_flag u_clientimage_info.http_info.flag[13]
#define U_http_len_user1 u_clientimage_info.http_info.flag[14]
#define U_http_len_user2 u_clientimage_info.http_info.flag[15]
enum HttpRequestType {
HTTP_IS_KEEP_ALIVE = 0x01,
HTTP_IS_DATA_CHUNKED = 0x02,
HTTP_IS_ACCEPT_GZIP = 0x04,
HTTP_IS_ACCEPT_BROTLI = 0x08,
HTTP_IS_NOCACHE_FILE = 0x10,
HTTP_IS_REQUEST_NOSTAT = 0x20,
HTTP_METHOD_NOT_IMPLEMENTED = 0x40
};
#define U_http_keep_alive ((U_http_flag & HTTP_IS_KEEP_ALIVE) != 0)
#define U_http_data_chunked ((U_http_flag & HTTP_IS_DATA_CHUNKED) != 0)
#define U_http_is_accept_gzip ((U_http_flag & HTTP_IS_ACCEPT_GZIP) != 0)
#define U_http_is_accept_brotli ((U_http_flag & HTTP_IS_ACCEPT_BROTLI) != 0)
#define U_http_is_nocache_file ((U_http_flag & HTTP_IS_NOCACHE_FILE) != 0)
#define U_http_is_request_nostat ((U_http_flag & HTTP_IS_REQUEST_NOSTAT) != 0)
#define U_http_method_not_implemented ((U_http_flag & HTTP_METHOD_NOT_IMPLEMENTED) != 0)
#define U_HTTP_INFO_INIT(c) (void) U_SYSCALL(memset, "%p,%d,%u", &(u_clientimage_info.http_info), c, sizeof(uhttpinfo))
#define U_HTTP_INFO_RESET(c) (void) U_SYSCALL(memset, "%p,%d,%u", &(u_clientimage_info.http_info.nResponseCode), c, 52)
#define U_HTTP_URI_TO_PARAM u_clientimage_info.http_info.uri, u_clientimage_info.http_info.uri_len
#define U_HTTP_URI_TO_TRACE u_clientimage_info.http_info.uri_len, u_clientimage_info.http_info.uri
#define U_HTTP_QUERY_TO_PARAM u_clientimage_info.http_info.query, u_clientimage_info.http_info.query_len
#define U_HTTP_QUERY_TO_TRACE u_clientimage_info.http_info.query_len, u_clientimage_info.http_info.query
#define U_HTTP_URI_QUERY_LEN (u_clientimage_info.http_info.uri_len + u_clientimage_info.http_info.query_len + (u_clientimage_info.http_info.query_len ? 1 : 0))
#define U_HTTP_URI_QUERY_TO_PARAM u_clientimage_info.http_info.uri, U_HTTP_URI_QUERY_LEN
#define U_HTTP_URI_QUERY_TO_TRACE U_HTTP_URI_QUERY_LEN, u_clientimage_info.http_info.uri
#define U_HTTP_CTYPE_TO_PARAM u_clientimage_info.http_info.content_type, U_http_content_type_len
#define U_HTTP_CTYPE_TO_TRACE U_http_content_type_len, u_clientimage_info.http_info.content_type
#define U_HTTP_RANGE_TO_PARAM u_clientimage_info.http_info.range, U_http_range_len
#define U_HTTP_RANGE_TO_TRACE U_http_range_len, u_clientimage_info.http_info.range
#define U_HTTP_COOKIE_TO_PARAM u_clientimage_info.http_info.cookie, u_clientimage_info.http_info.cookie_len
#define U_HTTP_COOKIE_TO_TRACE u_clientimage_info.http_info.cookie_len, u_clientimage_info.http_info.cookie
#define U_HTTP_REFERER_TO_PARAM u_clientimage_info.http_info.referer, u_clientimage_info.http_info.referer_len
#define U_HTTP_REFERER_TO_TRACE u_clientimage_info.http_info.referer_len, u_clientimage_info.http_info.referer
#define U_HTTP_IP_CLIENT_TO_PARAM u_clientimage_info.http_info.ip_client, U_http_ip_client_len
#define U_HTTP_IP_CLIENT_TO_TRACE U_http_ip_client_len, u_clientimage_info.http_info.ip_client
#define U_HTTP_USER_AGENT_TO_PARAM u_clientimage_info.http_info.user_agent, u_clientimage_info.http_info.user_agent_len
#define U_HTTP_USER_AGENT_TO_TRACE u_clientimage_info.http_info.user_agent_len, u_clientimage_info.http_info.user_agent
#define U_HTTP_ACCEPT_TO_PARAM u_clientimage_info.http_info.accept, U_http_accept_len
#define U_HTTP_ACCEPT_TO_TRACE U_http_accept_len, u_clientimage_info.http_info.accept
#define U_HTTP_ACCEPT_LANGUAGE_TO_PARAM u_clientimage_info.http_info.accept_language, U_http_accept_language_len
#define U_HTTP_ACCEPT_LANGUAGE_TO_TRACE U_http_accept_language_len, u_clientimage_info.http_info.accept_language
#define U_HTTP_METHOD_TO_PARAM U_HTTP_METHOD_NUM_TO_PARAM(U_http_method_num)
#define U_HTTP_METHOD_TO_TRACE U_HTTP_METHOD_NUM_TO_TRACE(U_http_method_num)
#define U_HTTP_METHOD_NUM_TO_PARAM(num) u_clientimage_info.http_method_list[num].name, u_clientimage_info.http_method_list[num].len
#define U_HTTP_METHOD_NUM_TO_TRACE(num) u_clientimage_info.http_method_list[num].len, u_clientimage_info.http_method_list[num].name
#define U_HTTP_URI_MEMEQ(str) (memcmp(u_clientimage_info.http_info.uri, U_CONSTANT_TO_PARAM(str)) == 0)
#define U_HTTP_URI_STREQ(str) U_STREQ(u_clientimage_info.http_info.uri, u_clientimage_info.http_info.uri_len, str)
#define U_HTTP_HOST_STREQ(str) (U_http_host_len ? U_STREQ(u_clientimage_info.http_info.host, U_http_host_len, str) : false)
#define U_HTTP_REFERER_STREQ(str) (u_clientimage_info.http_info.referer_len ? U_STREQ(u_clientimage_info.http_info.referer,u_clientimage_info.http_info.referer_len, str) : false)
#define U_HTTP_CTYPE_MEMEQ(str) (U_http_content_type_len ? memcmp(u_clientimage_info.http_info.content_type, U_CONSTANT_TO_PARAM(str)) == 0 : false)
#define U_HTTP_CTYPE_STREQ(str) (U_http_content_type_len ? U_STREQ(u_clientimage_info.http_info.content_type, U_http_content_type_len, str) : false)
#define U_HTTP_QUERY_MEMEQ(str) (u_clientimage_info.http_info.query_len ? memcmp(u_clientimage_info.http_info.query, U_CONSTANT_TO_PARAM(str)) == 0 : false)
#define U_HTTP_QUERY_STREQ(str) (u_clientimage_info.http_info.query_len ? U_STREQ(u_clientimage_info.http_info.query, u_clientimage_info.http_info.query_len, str) : false)
#define U_HTTP_USER_AGENT_MEMEQ(str) (u_clientimage_info.http_info.user_agent_len ? memcmp(u_clientimage_info.http_info.user_agent,U_CONSTANT_TO_PARAM(str)) == 0 : false)
#define U_HTTP_USER_AGENT_STREQ(str) (u_clientimage_info.http_info.user_agent_len \
? U_STREQ(u_clientimage_info.http_info.user_agent,u_clientimage_info.http_info.user_agent_len, str) \
: false)
/**
* The hostname of your server from header's request.
* The difference between U_HTTP_HOST_.. and U_HTTP_VHOST_.. is that
* U_HTTP_HOST_.. can include the :PORT text, and U_HTTP_VHOST_.. only the name
*/
#define U_HTTP_HOST_TO_PARAM u_clientimage_info.http_info.host, U_http_host_len
#define U_HTTP_HOST_TO_TRACE U_http_host_len, u_clientimage_info.http_info.host
#define U_HTTP_VHOST_TO_PARAM u_clientimage_info.http_info.host, U_http_host_vlen
#define U_HTTP_VHOST_TO_TRACE U_http_host_vlen, u_clientimage_info.http_info.host
#endif