mirror of
https://github.com/wg/wrk
synced 2026-06-10 00:55:51 +08:00
Compare commits
4 Commits
+483
-246
File diff suppressed because it is too large
Load Diff
+45
-19
@@ -24,8 +24,10 @@
|
|||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Also update SONAME in the Makefile whenever you change these. */
|
||||||
#define HTTP_PARSER_VERSION_MAJOR 2
|
#define HTTP_PARSER_VERSION_MAJOR 2
|
||||||
#define HTTP_PARSER_VERSION_MINOR 1
|
#define HTTP_PARSER_VERSION_MINOR 4
|
||||||
|
#define HTTP_PARSER_VERSION_PATCH 2
|
||||||
|
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#if defined(_WIN32) && !defined(__MINGW32__) && (!defined(_MSC_VER) || _MSC_VER<1600)
|
#if defined(_WIN32) && !defined(__MINGW32__) && (!defined(_MSC_VER) || _MSC_VER<1600)
|
||||||
@@ -50,9 +52,16 @@ typedef unsigned __int64 uint64_t;
|
|||||||
# define HTTP_PARSER_STRICT 1
|
# define HTTP_PARSER_STRICT 1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Maximium header size allowed */
|
/* Maximium header size allowed. If the macro is not defined
|
||||||
#define HTTP_MAX_HEADER_SIZE (80*1024)
|
* before including this header then the default is used. To
|
||||||
|
* change the maximum header size, define the macro in the build
|
||||||
|
* environment (e.g. -DHTTP_MAX_HEADER_SIZE=<value>). To remove
|
||||||
|
* the effective limit on the size of the header, define the macro
|
||||||
|
* to a very large number (e.g. -DHTTP_MAX_HEADER_SIZE=0x7fffffff)
|
||||||
|
*/
|
||||||
|
#ifndef HTTP_MAX_HEADER_SIZE
|
||||||
|
# define HTTP_MAX_HEADER_SIZE (80*1024)
|
||||||
|
#endif
|
||||||
|
|
||||||
typedef struct http_parser http_parser;
|
typedef struct http_parser http_parser;
|
||||||
typedef struct http_parser_settings http_parser_settings;
|
typedef struct http_parser_settings http_parser_settings;
|
||||||
@@ -67,7 +76,7 @@ typedef struct http_parser_settings http_parser_settings;
|
|||||||
* HEAD request which may contain 'Content-Length' or 'Transfer-Encoding:
|
* HEAD request which may contain 'Content-Length' or 'Transfer-Encoding:
|
||||||
* chunked' headers that indicate the presence of a body.
|
* chunked' headers that indicate the presence of a body.
|
||||||
*
|
*
|
||||||
* http_data_cb does not return data chunks. It will be call arbitrarally
|
* http_data_cb does not return data chunks. It will be called arbitrarily
|
||||||
* many times for each string. E.G. you might get 10 callbacks for "on_url"
|
* many times for each string. E.G. you might get 10 callbacks for "on_url"
|
||||||
* each providing just a few characters more data.
|
* each providing just a few characters more data.
|
||||||
*/
|
*/
|
||||||
@@ -108,6 +117,8 @@ typedef int (*http_cb) (http_parser*);
|
|||||||
/* RFC-5789 */ \
|
/* RFC-5789 */ \
|
||||||
XX(24, PATCH, PATCH) \
|
XX(24, PATCH, PATCH) \
|
||||||
XX(25, PURGE, PURGE) \
|
XX(25, PURGE, PURGE) \
|
||||||
|
/* CalDAV */ \
|
||||||
|
XX(26, MKCALENDAR, MKCALENDAR) \
|
||||||
|
|
||||||
enum http_method
|
enum http_method
|
||||||
{
|
{
|
||||||
@@ -125,9 +136,10 @@ enum flags
|
|||||||
{ F_CHUNKED = 1 << 0
|
{ F_CHUNKED = 1 << 0
|
||||||
, F_CONNECTION_KEEP_ALIVE = 1 << 1
|
, F_CONNECTION_KEEP_ALIVE = 1 << 1
|
||||||
, F_CONNECTION_CLOSE = 1 << 2
|
, F_CONNECTION_CLOSE = 1 << 2
|
||||||
, F_TRAILING = 1 << 3
|
, F_CONNECTION_UPGRADE = 1 << 3
|
||||||
, F_UPGRADE = 1 << 4
|
, F_TRAILING = 1 << 4
|
||||||
, F_SKIPBODY = 1 << 5
|
, F_UPGRADE = 1 << 5
|
||||||
|
, F_SKIPBODY = 1 << 6
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@@ -141,13 +153,13 @@ enum flags
|
|||||||
\
|
\
|
||||||
/* Callback-related errors */ \
|
/* Callback-related errors */ \
|
||||||
XX(CB_message_begin, "the on_message_begin callback failed") \
|
XX(CB_message_begin, "the on_message_begin callback failed") \
|
||||||
XX(CB_status_complete, "the on_status_complete callback failed") \
|
|
||||||
XX(CB_url, "the on_url callback failed") \
|
XX(CB_url, "the on_url callback failed") \
|
||||||
XX(CB_header_field, "the on_header_field callback failed") \
|
XX(CB_header_field, "the on_header_field callback failed") \
|
||||||
XX(CB_header_value, "the on_header_value callback failed") \
|
XX(CB_header_value, "the on_header_value callback failed") \
|
||||||
XX(CB_headers_complete, "the on_headers_complete callback failed") \
|
XX(CB_headers_complete, "the on_headers_complete callback failed") \
|
||||||
XX(CB_body, "the on_body callback failed") \
|
XX(CB_body, "the on_body callback failed") \
|
||||||
XX(CB_message_complete, "the on_message_complete callback failed") \
|
XX(CB_message_complete, "the on_message_complete callback failed") \
|
||||||
|
XX(CB_status, "the on_status callback failed") \
|
||||||
\
|
\
|
||||||
/* Parsing-related errors */ \
|
/* Parsing-related errors */ \
|
||||||
XX(INVALID_EOF_STATE, "stream ended at an unexpected time") \
|
XX(INVALID_EOF_STATE, "stream ended at an unexpected time") \
|
||||||
@@ -191,11 +203,11 @@ enum http_errno {
|
|||||||
|
|
||||||
struct http_parser {
|
struct http_parser {
|
||||||
/** PRIVATE **/
|
/** PRIVATE **/
|
||||||
unsigned char type : 2; /* enum http_parser_type */
|
unsigned int type : 2; /* enum http_parser_type */
|
||||||
unsigned char flags : 6; /* F_* values from 'flags' enum; semi-public */
|
unsigned int flags : 6; /* F_* values from 'flags' enum; semi-public */
|
||||||
unsigned char state; /* enum state from http_parser.c */
|
unsigned int state : 8; /* enum state from http_parser.c */
|
||||||
unsigned char header_state; /* enum header_state from http_parser.c */
|
unsigned int header_state : 8; /* enum header_state from http_parser.c */
|
||||||
unsigned char index; /* index into current matcher */
|
unsigned int index : 8; /* index into current matcher */
|
||||||
|
|
||||||
uint32_t nread; /* # bytes read in various scenarios */
|
uint32_t nread; /* # bytes read in various scenarios */
|
||||||
uint64_t content_length; /* # bytes in body (0 if no Content-Length header) */
|
uint64_t content_length; /* # bytes in body (0 if no Content-Length header) */
|
||||||
@@ -203,16 +215,16 @@ struct http_parser {
|
|||||||
/** READ-ONLY **/
|
/** READ-ONLY **/
|
||||||
unsigned short http_major;
|
unsigned short http_major;
|
||||||
unsigned short http_minor;
|
unsigned short http_minor;
|
||||||
unsigned short status_code; /* responses only */
|
unsigned int status_code : 16; /* responses only */
|
||||||
unsigned char method; /* requests only */
|
unsigned int method : 8; /* requests only */
|
||||||
unsigned char http_errno : 7;
|
unsigned int http_errno : 7;
|
||||||
|
|
||||||
/* 1 = Upgrade header was present and the parser has exited because of that.
|
/* 1 = Upgrade header was present and the parser has exited because of that.
|
||||||
* 0 = No upgrade header present.
|
* 0 = No upgrade header present.
|
||||||
* Should be checked when http_parser_execute() returns in addition to
|
* Should be checked when http_parser_execute() returns in addition to
|
||||||
* error checking.
|
* error checking.
|
||||||
*/
|
*/
|
||||||
unsigned char upgrade : 1;
|
unsigned int upgrade : 1;
|
||||||
|
|
||||||
/** PUBLIC **/
|
/** PUBLIC **/
|
||||||
void *data; /* A pointer to get hook to the "connection" or "socket" object */
|
void *data; /* A pointer to get hook to the "connection" or "socket" object */
|
||||||
@@ -222,7 +234,7 @@ struct http_parser {
|
|||||||
struct http_parser_settings {
|
struct http_parser_settings {
|
||||||
http_cb on_message_begin;
|
http_cb on_message_begin;
|
||||||
http_data_cb on_url;
|
http_data_cb on_url;
|
||||||
http_cb on_status_complete;
|
http_data_cb on_status;
|
||||||
http_data_cb on_header_field;
|
http_data_cb on_header_field;
|
||||||
http_data_cb on_header_value;
|
http_data_cb on_header_value;
|
||||||
http_cb on_headers_complete;
|
http_cb on_headers_complete;
|
||||||
@@ -261,9 +273,23 @@ struct http_parser_url {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* Returns the library version. Bits 16-23 contain the major version number,
|
||||||
|
* bits 8-15 the minor version number and bits 0-7 the patch level.
|
||||||
|
* Usage example:
|
||||||
|
*
|
||||||
|
* unsigned long version = http_parser_version();
|
||||||
|
* unsigned major = (version >> 16) & 255;
|
||||||
|
* unsigned minor = (version >> 8) & 255;
|
||||||
|
* unsigned patch = version & 255;
|
||||||
|
* printf("http_parser v%u.%u.%u\n", major, minor, patch);
|
||||||
|
*/
|
||||||
|
unsigned long http_parser_version(void);
|
||||||
|
|
||||||
void http_parser_init(http_parser *parser, enum http_parser_type type);
|
void http_parser_init(http_parser *parser, enum http_parser_type type);
|
||||||
|
|
||||||
|
|
||||||
|
/* Executes the parser. Returns number of parsed bytes. Sets
|
||||||
|
* `parser->http_errno` on error. */
|
||||||
size_t http_parser_execute(http_parser *parser,
|
size_t http_parser_execute(http_parser *parser,
|
||||||
const http_parser_settings *settings,
|
const http_parser_settings *settings,
|
||||||
const char *data,
|
const char *data,
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ status sock_close(connection *c) {
|
|||||||
status sock_read(connection *c, size_t *n) {
|
status sock_read(connection *c, size_t *n) {
|
||||||
ssize_t r = read(c->fd, c->buf, sizeof(c->buf));
|
ssize_t r = read(c->fd, c->buf, sizeof(c->buf));
|
||||||
*n = (size_t) r;
|
*n = (size_t) r;
|
||||||
return r > 0 ? OK : ERROR;
|
return r >= 0 ? OK : ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
status sock_write(connection *c, char *buf, size_t len, size_t *n) {
|
status sock_write(connection *c, char *buf, size_t len, size_t *n) {
|
||||||
|
|||||||
@@ -458,15 +458,18 @@ static void socket_connected(aeEventLoop *loop, int fd, void *data, int mask) {
|
|||||||
error:
|
error:
|
||||||
c->thread->errors.connect++;
|
c->thread->errors.connect++;
|
||||||
reconnect_socket(c->thread, c);
|
reconnect_socket(c->thread, c);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void socket_writeable(aeEventLoop *loop, int fd, void *data, int mask) {
|
static void socket_writeable(aeEventLoop *loop, int fd, void *data, int mask) {
|
||||||
connection *c = data;
|
connection *c = data;
|
||||||
thread *thread = c->thread;
|
thread *thread = c->thread;
|
||||||
|
|
||||||
if (!c->written && cfg.dynamic) {
|
if (!c->written) {
|
||||||
script_request(thread->L, &c->request, &c->length);
|
if (cfg.dynamic) {
|
||||||
|
script_request(thread->L, &c->request, &c->length);
|
||||||
|
}
|
||||||
|
c->start = time_us();
|
||||||
|
c->pending = cfg.pipeline;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *buf = c->request + c->written;
|
char *buf = c->request + c->written;
|
||||||
@@ -479,11 +482,6 @@ static void socket_writeable(aeEventLoop *loop, int fd, void *data, int mask) {
|
|||||||
case RETRY: return;
|
case RETRY: return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!c->written) {
|
|
||||||
c->start = time_us();
|
|
||||||
c->pending = cfg.pipeline;
|
|
||||||
}
|
|
||||||
|
|
||||||
c->written += n;
|
c->written += n;
|
||||||
if (c->written == c->length) {
|
if (c->written == c->length) {
|
||||||
c->written = 0;
|
c->written = 0;
|
||||||
@@ -553,7 +551,8 @@ static struct option longopts[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static int parse_args(struct config *cfg, char **url, char **headers, int argc, char **argv) {
|
static int parse_args(struct config *cfg, char **url, char **headers, int argc, char **argv) {
|
||||||
char c, **header = headers;
|
char **header = headers;
|
||||||
|
int c;
|
||||||
|
|
||||||
memset(cfg, 0, sizeof(struct config));
|
memset(cfg, 0, sizeof(struct config));
|
||||||
cfg->threads = 2;
|
cfg->threads = 2;
|
||||||
|
|||||||
Reference in New Issue
Block a user