Merge pull request #942 from joemfb/libh2o

rewrites http client and server to use libh2o
This commit is contained in:
Joe Bryan 2018-04-11 13:12:23 -07:00 committed by GitHub
commit 4842bb8c29
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 1311 additions and 2363 deletions

6
.gitmodules vendored
View File

@ -4,9 +4,6 @@
[submodule "subprojects/commonmark-legacy"] [submodule "subprojects/commonmark-legacy"]
path = subprojects/commonmark-legacy path = subprojects/commonmark-legacy
url = https://github.com/urbit/commonmark-legacy.git url = https://github.com/urbit/commonmark-legacy.git
[submodule "subprojects/http-parser-legacy"]
path = subprojects/http-parser-legacy
url = https://github.com/urbit/http-parser-legacy.git
[submodule "subprojects/ed25519"] [submodule "subprojects/ed25519"]
path = subprojects/ed25519 path = subprojects/ed25519
url = https://github.com/urbit/ed25519.git url = https://github.com/urbit/ed25519.git
@ -19,3 +16,6 @@
[submodule "subprojects/libuv"] [submodule "subprojects/libuv"]
path = subprojects/libuv path = subprojects/libuv
url = https://github.com/urbit/libuv.git url = https://github.com/urbit/libuv.git
[submodule "subprojects/h2o"]
path = subprojects/libh2o
url = https://github.com/urbit/h2o.git

View File

@ -90,3 +90,10 @@
c3_assert(!"memory lost"); \ c3_assert(!"memory lost"); \
} \ } \
rut;}) rut;})
/* c3_calloc(): asserting calloc
*/
#define c3_calloc(s) ({ \
void* rut = c3_malloc(s); \
memset(rut, 0, s); \
rut;})

View File

@ -232,6 +232,7 @@
# define c3__con c3_s3('c','o','n') # define c3__con c3_s3('c','o','n')
# define c3__cone c3_s4('c','o','n','e') # define c3__cone c3_s4('c','o','n','e')
# define c3__cong c3_s4('c','o','n','g') # define c3__cong c3_s4('c','o','n','g')
# define c3__conn c3_s4('c','o','n','n')
# define c3__cons c3_s4('c','o','n','s') # define c3__cons c3_s4('c','o','n','s')
# define c3__cook c3_s4('c','o','o','k') # define c3__cook c3_s4('c','o','o','k')
# define c3__cool c3_s4('c','o','o','l') # define c3__cool c3_s4('c','o','o','l')
@ -772,6 +773,7 @@
# define c3__oops c3_s4('o','o','p','s') # define c3__oops c3_s4('o','o','p','s')
# define c3__op c3_s2('o','p') # define c3__op c3_s2('o','p')
# define c3__open c3_s4('o','p','e','n') # define c3__open c3_s4('o','p','e','n')
# define c3__opts c3_s4('o','p','t','s')
# define c3__or c3_s2('o','r') # define c3__or c3_s2('o','r')
# define c3__ord c3_s3('o','r','d') # define c3__ord c3_s3('o','r','d')
# define c3__orth c3_s4('o','r','t','h') # define c3__orth c3_s4('o','r','t','h')

View File

@ -2,6 +2,9 @@
** **
** This file is in the public domain. ** This file is in the public domain.
*/ */
#include "h2o.h"
/** Quasi-tunable parameters. /** Quasi-tunable parameters.
**/ **/
/* First kernel this executable can boot. /* First kernel this executable can boot.
@ -19,7 +22,9 @@
*/ */
typedef struct _u3_hhed { typedef struct _u3_hhed {
struct _u3_hhed* nex_u; struct _u3_hhed* nex_u;
c3_w nam_w;
c3_c* nam_c; c3_c* nam_c;
c3_w val_w;
c3_c* val_c; c3_c* val_c;
} u3_hhed; } u3_hhed;
@ -31,85 +36,38 @@
c3_y hun_y[0]; c3_y hun_y[0];
} u3_hbod; } u3_hbod;
/* u3_hrat: http parser state.
*/
typedef enum {
u3_hreq_non,
u3_hreq_nam,
u3_hreq_val
} u3_hrat;
/* u3_csat: client connection state.
*/
typedef enum {
u3_csat_dead = 0, // connection dead
u3_csat_addr = 1, // connection addressed
u3_csat_clyr = 2, // connection open in cleartext
u3_csat_crop = 3, // connection open, ssl needs hs
u3_csat_sing = 4, // connection handshaking ssl
u3_csat_cryp = 5, // connection open, ssl open
} u3_csat;
/* u3_hmet: http method. Matches jhttp encoding.
*/
typedef enum {
u3_hmet_delete,
u3_hmet_get,
u3_hmet_head,
u3_hmet_post,
u3_hmet_put,
u3_hmet_nop, // virtual method
u3_hmet_other // ie, unsupported
} u3_hmet;
/* u3_hreq: incoming http request. /* u3_hreq: incoming http request.
*/ */
typedef struct _u3_hreq { typedef struct _u3_hreq {
struct _u3_hcon* hon_u; // connection h2o_req_t* rec_u; // h2o request
c3_w seq_l; // sequence within connection c3_w seq_l; // sequence within connection
u3_hmet met_e; // method struct _u3_hcon* hon_u; // connection backlink
u3_hrat rat_e; // parser state struct _u3_hreq* nex_u; // next in connection's list
c3_c* url_c; // url
c3_w ipf_w; // ipv4
c3_o liv; // keepalive
c3_o end; // all responses added
u3_hhed* hed_u; // headers
u3_hbod* bod_u; // body parts (exit)
u3_hbod* dob_u; // body parts (entry)
struct _u3_hreq* nex_u; // next in request queue
u3_hbod* rub_u; // exit of write queue
u3_hbod* bur_u; // entry of write queue
} u3_hreq; } u3_hreq;
/* u3_hrep: outgoing http response.
*/
typedef struct _u3_hrep {
c3_w sev_l; // server number
c3_w coq_l; // connection number
c3_w seq_l; // request number
c3_w sas_w; // status
u3_hhed* hed_u; // headers
u3_hbod* bod_u; // body (one part)
} u3_hrep;
/* u3_hcon: incoming http connection. /* u3_hcon: incoming http connection.
*/ */
typedef struct _u3_hcon { typedef struct _u3_hcon {
uv_tcp_t wax_u; // event handler state uv_tcp_t wax_u; // client stream handler
h2o_conn_t* con_u; // h2o connection
h2o_socket_t* sok_u; // h2o connection socket
c3_w ipf_w; // client ipv4
c3_w coq_l; // connection number c3_w coq_l; // connection number
c3_w seq_l; // next request number c3_w seq_l; // next request number
struct _u3_http* htp_u; // backlink to server struct _u3_http* htp_u; // server backlink
struct _u3_hreq* req_u; // request list
struct _u3_hcon* nex_u; // next in server's list struct _u3_hcon* nex_u; // next in server's list
struct _u3_hreq* ruc_u; // request under construction
struct _u3_hreq* req_u; // exit of request queue
struct _u3_hreq* qer_u; // entry of request queue
void* par_u; // struct http_parser *
} u3_hcon; } u3_hcon;
/* u3_http: http server. /* u3_http: http server.
*/ */
typedef struct _u3_http { typedef struct _u3_http {
uv_tcp_t wax_u; // event handler state uv_tcp_t wax_u; // server stream handler
h2o_globalconf_t* fig_u; // h2o global config
h2o_context_t* ctx_u; // h2o ctx
h2o_accept_ctx_t* cep_u; // h2o accept ctx (wat for?)
h2o_hostconf_t* hos_u; // h2o host config
h2o_handler_t* han_u; // h2o request handler
c3_w sev_l; // server number c3_w sev_l; // server number
c3_w coq_l; // next connection number c3_w coq_l; // next connection number
c3_w por_w; // running port c3_w por_w; // running port
@ -119,13 +77,20 @@
struct _u3_http* nex_u; // next in list struct _u3_http* nex_u; // next in list
} u3_http; } u3_http;
/* u3_csat: client connection state.
*/
typedef enum {
u3_csat_init = 0, // initialized
u3_csat_addr = 1, // address resolution begun
u3_csat_quit = 2, // cancellation requested
u3_csat_ripe = 3 // passed to libh2o
} u3_csat;
/* u3_cres: response to http client. /* u3_cres: response to http client.
*/ */
typedef struct _u3_cres { typedef struct _u3_cres {
u3_hrat rat_e; // parser state
void* par_u; // struct http_parser *
c3_w sas_w; // status code c3_w sas_w; // status code
u3_hhed* hed_u; // headers u3_noun hed; // headers
u3_hbod* bod_u; // exit of body queue u3_hbod* bod_u; // exit of body queue
u3_hbod* dob_u; // entry of body queue u3_hbod* dob_u; // entry of body queue
} u3_cres; } u3_cres;
@ -134,59 +99,42 @@
*/ */
typedef struct _u3_creq { // client request typedef struct _u3_creq { // client request
c3_l num_l; // request number c3_l num_l; // request number
h2o_http1client_t* cli_u; // h2o client
u3_csat sat_e; // connection state
c3_o sec; // yes == https
c3_w ipf_w; // IP
c3_c* ipf_c; // IP (string)
c3_c* hot_c; // host c3_c* hot_c; // host
c3_s por_s; // port c3_s por_s; // port
c3_c* por_c; // port (string)
c3_m met_m; // method
c3_c* url_c; // url c3_c* url_c; // url
c3_o sec; // yes == https
u3_hmet met_e; // method
u3_hhed* hed_u; // headers u3_hhed* hed_u; // headers
u3_hbod* bod_u; // body u3_hbod* bod_u; // body
u3_cres* res_u; // nascent response
struct _u3_ccon* coc_u; // parent connection
struct _u3_creq* nex_u; // next in queue
} u3_creq;
/* u3_sslx: per-connection ssl context.
*/
typedef struct _u3_sslx {
void* ssl_u; // struct SSL*
void* rio_u; // struct BIO* for read
void* wio_u; // struct BIO* for write
} u3_sslx;
/* u3_ccon: outgoing http connection.
*/
typedef struct _u3_ccon { // client connection
uv_tcp_t wax_u; // i/o handler state
uv_connect_t cot_u; // connection handler state
uv_getaddrinfo_t adr_u; // resolver state
u3_sslx ssl; // ssl state
u3_csat sat_e; // connection state
c3_c* hot_c; // hostname
c3_s por_s; // port
c3_w ipf_w; // IP
c3_o sec; // yes == https
u3_hbod* rub_u; // exit of send queue u3_hbod* rub_u; // exit of send queue
u3_hbod* bur_u; // entry of send queue u3_hbod* bur_u; // entry of send queue
u3_creq* ceq_u; // exit of request queue h2o_iovec_t* vec_u; // send-buffer array
u3_creq* qec_u; // entry of request queue u3_cres* res_u; // nascent response
struct _u3_ccon* pre_u; // previous in list struct _u3_creq* nex_u; // next in list
struct _u3_ccon* nex_u; // next in list struct _u3_creq* pre_u; // previous in list
} u3_ccon; } u3_creq;
/* u3_chot: foreign host (not yet used). /* u3_chot: foreign host (not yet used).
*/ */
typedef struct _u3_chot { typedef struct _u3_chot {
c3_w ipf_w; // ip address (or 0) c3_w ipf_w; // ip address (or 0)
c3_c* hot_c; // hostname (no port) (or 0) c3_c* hot_c; // hostname (no port) (or 0)
struct _u3_ccon* ins_u; // insecure connection (or 0) void* ins_u; // insecure connection (or 0)
struct _u3_ccon* sec_u; // secure connection (or 0) void* sec_u; // secure connection (or 0)
} u3_chot; } u3_chot;
/* u3_cttp: http client. /* u3_cttp: http client.
*/ */
typedef struct _u3_cttp { typedef struct _u3_cttp {
struct _u3_ccon* coc_u; // connection list u3_creq* ceq_u; // request list
h2o_http1client_ctx_t* //
ctx_u; // h2o client ctx
void* tls_u; // client SSL_CTX*
} u3_cttp; } u3_cttp;
/* u3_apac: ames packet, coming or going. /* u3_apac: ames packet, coming or going.
@ -581,12 +529,11 @@
u3_behn teh_u; // behn timer u3_behn teh_u; // behn timer
c3_o liv; // if u3_no, shut down c3_o liv; // if u3_no, shut down
c3_i xit_i; // exit code for shutdown c3_i xit_i; // exit code for shutdown
void* ssl_u; // struct SSL_CTX* void* tls_u; // server SSL_CTX*
} u3_host; // host == computer == process } u3_host; // host == computer == process
# define u3L u3_Host.lup_u // global event loop # define u3L u3_Host.lup_u // global event loop
# define u3Z (&(u3_Raft)) # define u3Z (&(u3_Raft))
# define u3S u3_Host.ssl_u
/** Global variables. /** Global variables.
**/ **/

View File

@ -302,9 +302,9 @@ urbitscrypt_dep = dependency('libscrypt', version: '>=0.1.21', fallback: ['libsc
ed25519_dep = dependency('ed25519', version: '>=0.1.0', fallback: ['ed25519', 'ed25519_dep']) ed25519_dep = dependency('ed25519', version: '>=0.1.0', fallback: ['ed25519', 'ed25519_dep'])
murmur3_dep = dependency('murmur3', version: '>=0.1.0', fallback: ['murmur3', 'murmur3_dep']) murmur3_dep = dependency('murmur3', version: '>=0.1.0', fallback: ['murmur3', 'murmur3_dep'])
http_parser_dep = dependency('http-parser', version: '0.1.0', fallback: ['http-parser-legacy', 'http_parser_dep'])
softfloat3_dep = dependency('softfloat3', version: '>=3.0.0', fallback: ['softfloat3', 'softfloat3_dep']) softfloat3_dep = dependency('softfloat3', version: '>=3.0.0', fallback: ['softfloat3', 'softfloat3_dep'])
libuv_dep = dependency('libuv', version: '>=1.8.0', fallback:['libuv', 'libuv_dep']) libuv_dep = dependency('libuv', version: '>=1.8.0', fallback:['libuv', 'libuv_dep'])
libh2o_dep = dependency('libh2o', version: '>=0.13.3', fallback: ['libh2o', 'libh2o_dep'])
opt_flags = [] opt_flags = []
if get_option('debug') if get_option('debug')
@ -321,12 +321,12 @@ link_args: os_link_flags,
dependencies: [openssl_dep, dependencies: [openssl_dep,
curl_dep, curl_dep,
libuv_dep, libuv_dep,
libh2o_dep,
cmark_dep, cmark_dep,
gmp_dep, gmp_dep,
sigsegv_dep, sigsegv_dep,
urbitscrypt_dep, urbitscrypt_dep,
ed25519_dep, ed25519_dep,
murmur3_dep, murmur3_dep,
http_parser_dep,
softfloat3_dep] + os_deps, softfloat3_dep] + os_deps,
install: true) install: true)

@ -1 +0,0 @@
Subproject commit 90d2cd1a9049a93c291b604fc3c9c4c84cf9bfe2

1
subprojects/libh2o Submodule

@ -0,0 +1 @@
Subproject commit fcf8df4af0c31477e05c55c380c4d7324311368d

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -19,6 +19,9 @@
#include <term.h> #include <term.h>
#include <dirent.h> #include <dirent.h>
#include <openssl/ssl.h> #include <openssl/ssl.h>
#include <openssl/rand.h>
#include "h2o.h"
#define U3_GLOBAL #define U3_GLOBAL
#define C3_GLOBAL #define C3_GLOBAL
@ -475,6 +478,7 @@ report(void)
printf("openssl: %s\n", SSLeay_version(SSLEAY_VERSION)); printf("openssl: %s\n", SSLeay_version(SSLEAY_VERSION));
printf("curses: %s\n", curses_version()); printf("curses: %s\n", curses_version());
printf("libuv: %s\n", uv_version_string()); printf("libuv: %s\n", uv_version_string());
printf("libh2o: %d.%d.%d\n", H2O_LIBRARY_VERSION_MAJOR, H2O_LIBRARY_VERSION_MINOR, H2O_LIBRARY_VERSION_PATCH);
} }
void void
@ -624,6 +628,24 @@ main(c3_i argc,
#endif #endif
} }
SSL_library_init();
SSL_load_error_strings();
{
c3_i rad;
c3_y buf[4096];
// RAND_status, at least on OS X, never returns true.
// 4096 bytes should be enough entropy for anyone, right?
rad = open("/dev/urandom", O_RDONLY);
if ( 4096 != read(rad, &buf, 4096) ) {
perror("rand-seed");
exit(1);
}
RAND_seed(buf, 4096);
close(rad);
}
// u3e_grab("main", u3_none); // u3e_grab("main", u3_none);
// //
u3_lo_loop(); u3_lo_loop();