shrub/vere/http.c

901 lines
21 KiB
C
Raw Normal View History

2013-09-29 00:21:18 +04:00
/* v/http.c
**
*/
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdint.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <uv.h>
#include <errno.h>
#include "all.h"
#include "vere/vere.h"
2013-09-29 00:21:18 +04:00
2017-11-22 09:55:39 +03:00
#include "h2o.h"
2017-11-27 09:34:47 +03:00
#include <openssl/ssl.h>
#include <openssl/err.h>
#include <openssl/rand.h>
// XX rename
2017-11-25 01:12:09 +03:00
typedef struct _h2hed {
struct _h2hed* nex_u;
c3_w nam_w;
c3_c* nam_c;
c3_w val_w;
c3_c* val_c;
} h2hed;
/* forward declarations
*/
static void _http_request(u3_hreq* req_u, u3_noun recq);
static void _http_request_kill(u3_hreq* req_u);
static void _http_req_link(u3_hcon* hon_u, u3_hreq* req_u);
static void _http_req_unlink(u3_hreq* req_u);
static void _http_conn_link(u3_http* htp_u, u3_hcon* hon_u);
static void _http_conn_unlink(u3_hcon* hon_u);
static const c3_i TCP_BACKLOG = 16;
// XX u3_Host.tls_u ?
static SSL_CTX* tls_u = 0;
/* _http_vec_to_meth(): convert h2o_iovec_t to meth
2017-11-22 09:55:39 +03:00
*/
static u3_weak
_http_vec_to_meth(h2o_iovec_t vec_u)
2017-11-22 09:55:39 +03:00
{
2017-11-25 00:32:11 +03:00
return ( 0 == strncmp(vec_u.base, "GET", vec_u.len) ) ? c3__get :
( 0 == strncmp(vec_u.base, "PUT", vec_u.len) ) ? c3__put :
( 0 == strncmp(vec_u.base, "POST", vec_u.len) ) ? c3__post :
( 0 == strncmp(vec_u.base, "HEAD", vec_u.len) ) ? c3__head :
( 0 == strncmp(vec_u.base, "CONNECT", vec_u.len) ) ? c3__conn :
( 0 == strncmp(vec_u.base, "DELETE", vec_u.len) ) ? c3__delt :
( 0 == strncmp(vec_u.base, "OPTIONS", vec_u.len) ) ? c3__opts :
( 0 == strncmp(vec_u.base, "TRACE", vec_u.len) ) ? c3__trac :
// TODO ??
// ( 0 == strncmp(vec_u.base, "PATCH", vec_u.len) ) ? c3__patc :
u3_none;
2017-11-22 09:55:39 +03:00
}
/* _http_vec_to_atom(): convert h2o_iovec_t to atom (cord)
2017-11-22 09:55:39 +03:00
*/
static u3_noun
_http_vec_to_atom(h2o_iovec_t vec_u)
2017-11-22 09:55:39 +03:00
{
// XX portable?
2017-11-22 09:55:39 +03:00
return u3i_bytes(vec_u.len, (const c3_y*)vec_u.base);
}
/* _http_vec_to_octs(): convert h2o_iovec_t to (unit octs)
2017-11-22 09:55:39 +03:00
*/
static u3_noun
_http_vec_to_octs(h2o_iovec_t vec_u)
2017-11-22 09:55:39 +03:00
{
if ( 0 == vec_u.len ) {
return u3_nul;
}
// XX correct size_t -> atom?
return u3nt(u3_nul, u3i_chubs(1, (const c3_d*)&vec_u.len),
_http_vec_to_atom(vec_u));
2017-11-22 09:55:39 +03:00
}
/* _http_vec_from_octs(): convert (unit octs) to h2o_iovec_t
2017-11-22 09:55:39 +03:00
*/
static h2o_iovec_t*
_http_vec_from_octs(u3_noun oct)
2017-11-22 09:55:39 +03:00
{
h2o_iovec_t* vec_u = c3_malloc(sizeof(*vec_u));
2017-11-22 09:55:39 +03:00
if ( u3_nul == oct ) {
vec_u->base = 0;
2017-11-22 09:55:39 +03:00
vec_u->len = 0;
}
else {
2018-03-03 08:41:35 +03:00
// 2GB max
if ( c3n == u3a_is_cat(u3h(u3t(oct))) ) {
u3m_bail(c3__fail);
2017-11-28 06:42:16 +03:00
}
2017-11-22 09:55:39 +03:00
vec_u->len = u3h(u3t(oct));
vec_u->base = c3_malloc(vec_u->len);
u3r_bytes(0, vec_u->len, (c3_y*)vec_u->base, u3t(u3t(oct)));
2017-11-22 09:55:39 +03:00
}
u3z(oct);
return vec_u;
}
/* _http_heds_to_noun(): convert h2o_headers_t to (list (pair @t @t))
*/
static u3_noun
_http_heds_to_noun(h2o_headers_t* hed_u)
{
u3_noun hed = u3_nul;
2018-03-03 08:41:35 +03:00
c3_d dex_d = hed_u->size;
h2o_header_t deh;
2018-03-03 08:41:35 +03:00
while ( 0 < dex_d ) {
deh = hed_u->entries[--dex_d];
hed = u3nc(u3nc(_http_vec_to_atom(*deh.name),
_http_vec_to_atom(deh.value)), hed);
}
return hed;
}
/* _http_heds_from_noun(): convert (list (pair @t @t)) to h2hed
*/
2017-11-25 01:12:09 +03:00
static h2hed*
_http_heds_from_noun(u3_noun hed)
2017-11-23 06:44:22 +03:00
{
2017-11-25 01:12:09 +03:00
u3_noun deh = hed;
h2hed* hed_u = 0;
while ( u3_nul != hed ) {
u3_noun nam = u3h(u3h(hed));
u3_noun val = u3t(u3h(hed));
2017-11-23 06:44:22 +03:00
2017-11-25 01:12:09 +03:00
h2hed* nex_u = c3_malloc(sizeof(*nex_u));
nex_u->nam_w = u3r_met(3, nam);
nex_u->val_w = u3r_met(3, val);
nex_u->nam_c = c3_malloc(nex_u->nam_w);
nex_u->val_c = c3_malloc(nex_u->val_w);
2017-11-23 06:44:22 +03:00
2017-11-25 01:12:09 +03:00
u3r_bytes(0, nex_u->nam_w, (c3_y*)nex_u->nam_c, nam);
u3r_bytes(0, nex_u->val_w, (c3_y*)nex_u->val_c, val);
2017-11-25 01:12:09 +03:00
nex_u->nex_u = hed_u;
hed_u = nex_u;
hed = u3t(hed);
}
2017-11-23 06:44:22 +03:00
2017-11-25 01:12:09 +03:00
u3z(deh);
return hed_u;
2017-11-23 06:44:22 +03:00
}
2017-11-22 09:55:39 +03:00
/* _http_req_to_httq(): convert h2o_req_t to httq
2017-11-22 09:55:39 +03:00
*/
static u3_weak
_http_req_to_httq(h2o_req_t* rec_u)
2017-11-22 09:55:39 +03:00
{
u3_noun med = _http_vec_to_meth(rec_u->method);
if ( u3_none == med ) {
return u3_none;
2017-11-22 09:55:39 +03:00
}
u3_noun url = _http_vec_to_atom(rec_u->path);
u3_noun hed = _http_heds_to_noun(&rec_u->headers);
// restore host header
hed = u3nc(u3nc(u3i_string("host"),
_http_vec_to_atom(rec_u->authority)),
hed);
u3_noun bod = _http_vec_to_octs(rec_u->entity);
2017-11-22 09:55:39 +03:00
return u3nq(med, url, hed, bod);
2017-11-22 09:55:39 +03:00
}
/* _http_req_free(): free http request.
2013-09-29 00:21:18 +04:00
*/
2014-08-21 02:09:51 +04:00
static void
_http_req_free(u3_hreq* req_u)
2013-09-29 00:21:18 +04:00
{
_http_req_unlink(req_u);
free(req_u);
2013-09-29 00:21:18 +04:00
}
/* _http_req_new(): receive http request.
2013-09-29 00:21:18 +04:00
*/
static u3_hreq*
_http_req_new(u3_hcon* hon_u, h2o_req_t* rec_u)
2013-09-29 00:21:18 +04:00
{
u3_hreq* req_u = c3_malloc(sizeof(*req_u));
req_u->rec_u = rec_u;
_http_req_link(hon_u, req_u);
2013-09-29 00:21:18 +04:00
return req_u;
2013-09-29 00:21:18 +04:00
}
/* _http_send_response(): write httr to h2o_req_t->res and send
2013-09-29 00:21:18 +04:00
*/
static void
_http_send_response(u3_hreq* req_u, u3_noun sas, u3_noun hed, u3_noun bod)
2013-09-29 00:21:18 +04:00
{
h2o_req_t* rec_u = req_u->rec_u;
2013-09-29 00:21:18 +04:00
rec_u->res.status = sas;
rec_u->res.reason = (sas < 200) ? "Weird" :
(sas < 300) ? "OK" :
(sas < 400) ? "Moved" :
(sas < 500) ? "Missing" :
"Hosed";
2013-09-29 00:21:18 +04:00
h2hed* hed_u = _http_heds_from_noun(u3k(hed));
h2hed* deh_u = hed_u;
2013-09-29 00:21:18 +04:00
while ( 0 != hed_u ) {
h2o_add_header_by_str(&rec_u->pool, &rec_u->res.headers,
hed_u->nam_c, hed_u->nam_w, 0, 0,
hed_u->val_c, hed_u->val_w);
hed_u = hed_u->nex_u;
2013-09-29 00:21:18 +04:00
}
2017-12-09 06:56:57 +03:00
// XX free req_u on disponse (rec_u should be freed by h2o)
static h2o_generator_t gen_u = {NULL, NULL};
h2o_start_response(rec_u, &gen_u);
2013-09-29 00:21:18 +04:00
h2o_iovec_t* bod_u = _http_vec_from_octs(u3k(bod));
2017-12-09 06:56:57 +03:00
rec_u->res.content_length = bod_u->len;
2018-03-03 08:41:35 +03:00
h2o_send(rec_u, bod_u, 1, H2O_SEND_STATE_FINAL);
2013-09-29 00:21:18 +04:00
_http_req_free(req_u);
2013-09-29 00:21:18 +04:00
// XX allocate on &req_u->pool and skip these?
free(bod_u->base);
free(bod_u);
while ( 0 != deh_u ) {
h2hed* duh_u = deh_u;
deh_u = deh_u->nex_u;
free(duh_u->nam_c);
free(duh_u->val_c);
free(duh_u);
2013-09-29 00:21:18 +04:00
}
u3z(sas); u3z(hed); u3z(bod);
2013-09-29 00:21:18 +04:00
}
// for casting and retrieving h2o_socket_t; see st_h2o_http1_conn_t
typedef struct _h2o_con_http1 {
h2o_conn_t con_u; // h2o connection
h2o_socket_t* sok_u; // h2o connection socket
} h2o_con_http1;
2013-09-29 00:21:18 +04:00
// for casting and retrieving u3_hcon; see st_h2o_uv_socket_t
typedef struct _h2o_sok_uv {
h2o_socket_t sok_u; // h2o connection socket
struct {
uv_stream_t *stream; // client stream handler (u3_hcon)
uv_close_cb close_cb;
} uv;
} h2o_sok_uv;
2013-09-29 00:21:18 +04:00
/* _http_conn_from_req(); retrieve connection from h2o http1 request.
2013-09-29 00:21:18 +04:00
*/
static u3_hcon*
_http_conn_from_req(h2o_req_t* rec_u)
2013-09-29 00:21:18 +04:00
{
// XX HTTP2 wat do?
h2o_con_http1* noc_u = (h2o_con_http1*)rec_u->conn;
h2o_sok_uv* kos_u = (h2o_sok_uv*)noc_u->sok_u;
return (u3_hcon*)kos_u->uv.stream;
};
2013-09-29 00:21:18 +04:00
/* _http_handle_new_req(); handle incoming http request from h2o.
2013-09-29 00:21:18 +04:00
*/
static c3_i
_http_handle_new_req(h2o_handler_t* han_u, h2o_req_t* rec_u)
2013-09-29 00:21:18 +04:00
{
u3_weak recq = _http_req_to_httq(rec_u);
2013-09-29 00:21:18 +04:00
if ( u3_none == recq ) {
if ( (u3C.wag_w & u3o_verbose) ) {
uL(fprintf(uH, "strange %.*s request\n", (int)rec_u->method.len,
rec_u->method.base));
}
2013-09-29 00:21:18 +04:00
static h2o_generator_t gen_u = {NULL, NULL};
rec_u->res.status = 400;
rec_u->res.reason = "Bad Request";
h2o_start_response(rec_u, &gen_u);
2018-03-03 08:41:35 +03:00
h2o_send(rec_u, 0, 0, H2O_SEND_STATE_FINAL);
2013-09-29 00:21:18 +04:00
}
else {
u3_hcon* hon_u = _http_conn_from_req(rec_u);
u3_hreq* req_u = _http_req_new(hon_u, rec_u);
_http_request(req_u, recq);
2013-09-29 00:21:18 +04:00
}
return 0;
2013-09-29 00:21:18 +04:00
}
/* _http_conn_free_early(): free http connection on failure.
*/
static void
_http_conn_free_early(uv_handle_t* han_t)
{
u3_hcon* hon_u = (u3_hcon*)han_t;
free(hon_u);
}
2013-09-29 00:21:18 +04:00
/* _http_conn_free(): free http connection on close.
*/
static void
_http_conn_free(uv_handle_t* han_t)
{
u3_hcon* hon_u = (u3_hcon*)han_t;
2013-09-29 00:21:18 +04:00
while ( 0 != hon_u->req_u ) {
u3_hreq* req_u = hon_u->req_u;
u3_hreq* nex_u = req_u->nex_u;
2013-09-29 00:21:18 +04:00
_http_request_kill(req_u);
2013-09-29 00:21:18 +04:00
_http_req_free(req_u);
hon_u->req_u = nex_u;
}
_http_conn_unlink(hon_u);
2013-09-29 00:21:18 +04:00
free(hon_u);
}
/* _http_conn_new(): create and accept http connection.
2013-09-29 00:21:18 +04:00
*/
static void
_http_conn_new(u3_http* htp_u)
2013-09-29 00:21:18 +04:00
{
// TODO where?
// u3_lo_open();
2013-09-29 00:21:18 +04:00
u3_hcon* hon_u = c3_malloc(sizeof(*hon_u));
hon_u->seq_l = 1;
hon_u->req_u = 0;
2013-09-29 00:21:18 +04:00
uv_tcp_init(u3L, &hon_u->wax_u);
2013-09-29 00:21:18 +04:00
2018-03-03 08:41:35 +03:00
c3_i sas_i;
if ( 0 != (sas_i = uv_accept((uv_stream_t*)&htp_u->wax_u,
(uv_stream_t*)&hon_u->wax_u)) ) {
if ( (u3C.wag_w & u3o_verbose) ) {
uL(fprintf(uH, "http: accept: %s\n", uv_strerror(sas_i)));
}
uv_close((uv_handle_t*)&hon_u->wax_u,
(uv_close_cb)_http_conn_free_early);
return;
2013-09-29 00:21:18 +04:00
}
_http_conn_link(htp_u, hon_u);
2013-09-29 00:21:18 +04:00
hon_u->sok_u = h2o_uv_socket_create((uv_stream_t*)&hon_u->wax_u,
(uv_close_cb)_http_conn_free);
h2o_accept(htp_u->cep_u, hon_u->sok_u);
2013-09-29 00:21:18 +04:00
// capture h2o connection (XX fragile)
hon_u->con_u = (h2o_conn_t*)hon_u->sok_u->data;
2013-09-29 00:21:18 +04:00
struct sockaddr_in adr_u;
h2o_socket_getpeername(hon_u->sok_u, (struct sockaddr*)&adr_u);
hon_u->ipf_w = ( adr_u.sin_family != AF_INET ) ?
0 : ntohl(adr_u.sin_addr.s_addr);
2013-09-29 00:21:18 +04:00
// TODO where?
// u3_lo_shut(c3y);
2013-09-29 00:21:18 +04:00
}
/* _http_listen_cb(): uv_connection_cb for uv_listen
2013-09-29 00:21:18 +04:00
*/
static void
_http_listen_cb(uv_stream_t* str_u, c3_i sas_i)
2013-09-29 00:21:18 +04:00
{
u3_http* htp_u = (u3_http*)str_u;
if ( 0 != sas_i ) {
2017-11-28 06:42:44 +03:00
uL(fprintf(uH, "http: listen_cb: %s\n", uv_strerror(sas_i)));
2013-09-29 00:21:18 +04:00
}
2014-03-03 23:51:35 +04:00
else {
_http_conn_new(htp_u);
2014-03-03 23:51:35 +04:00
}
2013-09-29 00:21:18 +04:00
}
/* _http_req_find(): find http request in connection by sequence.
2013-09-29 00:21:18 +04:00
*/
static u3_hreq*
_http_req_find(u3_hcon* hon_u, c3_w seq_l)
2013-09-29 00:21:18 +04:00
{
u3_hreq* req_u = hon_u->req_u;
2013-09-29 00:21:18 +04:00
// XX glories of linear search
//
while ( req_u ) {
if ( seq_l == req_u->seq_l ) {
return req_u;
}
req_u = req_u->nex_u;
2013-09-29 00:21:18 +04:00
}
return 0;
}
/* _http_req_link(): link http request to connection
2013-09-29 00:21:18 +04:00
*/
static void
_http_req_link(u3_hcon* hon_u, u3_hreq* req_u)
2013-09-29 00:21:18 +04:00
{
req_u->hon_u = hon_u;
req_u->seq_l = hon_u->seq_l++;
req_u->nex_u = hon_u->req_u;
hon_u->req_u = req_u;
2013-09-29 00:21:18 +04:00
}
/* _http_req_unlink(): remove http request from connection
2013-09-29 00:21:18 +04:00
*/
static void
_http_req_unlink(u3_hreq* req_u)
2013-09-29 00:21:18 +04:00
{
u3_hcon* hon_u = req_u->hon_u;
2013-09-29 00:21:18 +04:00
if ( hon_u->req_u == req_u ) {
hon_u->req_u = req_u->nex_u;
}
else {
u3_hreq* pre_u = hon_u->req_u;
// XX glories of linear search
//
while ( pre_u ) {
if ( pre_u->nex_u == req_u ) {
pre_u->nex_u = req_u->nex_u;
2013-09-29 00:21:18 +04:00
}
else pre_u = pre_u->nex_u;
2013-09-29 00:21:18 +04:00
}
}
}
/* _http_conn_find(): find http connection in server by sequence.
*/
static u3_hcon*
_http_conn_find(u3_http *htp_u, c3_w coq_l)
2017-11-22 09:55:39 +03:00
{
u3_hcon* hon_u = htp_u->hon_u;
// XX glories of linear search
//
while ( hon_u ) {
if ( coq_l == hon_u->coq_l ) {
return hon_u;
}
hon_u = hon_u->nex_u;
}
2017-11-22 09:55:39 +03:00
return 0;
}
/* _http_conn_link(): link http request to connection
2017-11-23 06:43:42 +03:00
*/
static void
_http_conn_link(u3_http* htp_u, u3_hcon* hon_u)
2017-11-22 09:55:39 +03:00
{
hon_u->htp_u = htp_u;
hon_u->coq_l = htp_u->coq_l++;
hon_u->nex_u = htp_u->hon_u;
htp_u->hon_u = hon_u;
2017-11-22 09:55:39 +03:00
}
/* _http_conn_unlink(): remove http request from connection
2013-09-29 00:21:18 +04:00
*/
static void
_http_conn_unlink(u3_hcon* hon_u)
2013-09-29 00:21:18 +04:00
{
u3_http* htp_u = hon_u->htp_u;
2013-09-29 00:21:18 +04:00
if ( htp_u->hon_u == hon_u ) {
htp_u->hon_u = hon_u->nex_u;
2013-09-29 00:21:18 +04:00
}
else {
u3_hcon *pre_u = htp_u->hon_u;
2013-09-29 00:21:18 +04:00
// XX glories of linear search
//
while ( pre_u ) {
if ( pre_u->nex_u == hon_u ) {
pre_u->nex_u = hon_u->nex_u;
}
else pre_u = pre_u->nex_u;
2013-09-29 00:21:18 +04:00
}
}
}
/* _http_serv_find(): find http server by sequence.
2014-03-20 02:40:40 +04:00
*/
static u3_http*
2014-03-20 02:40:40 +04:00
_http_serv_find(c3_l sev_l)
{
u3_http* htp_u = u3_Host.htp_u;
2014-03-20 02:40:40 +04:00
// XX glories of linear search
//
while ( htp_u ) {
if ( sev_l == htp_u->sev_l ) {
return htp_u;
}
htp_u = htp_u->nex_u;
}
return 0;
}
// XX serv_link and serv_unlink
2013-09-29 00:21:18 +04:00
// XX rename
2013-09-29 00:21:18 +04:00
/* _http_pox_to_noun(): translate srv/con/req to path noun (pox).
*/
2014-09-11 04:01:32 +04:00
static u3_noun
2013-09-29 00:21:18 +04:00
_http_pox_to_noun(c3_w sev_l, c3_w coq_l, c3_w seq_l)
{
return u3nt(u3_blip, c3__http,
u3nq(u3dc("scot", c3_s2('u','v'), sev_l),
u3dc("scot", c3_s2('u','d'), coq_l),
u3dc("scot", c3_s2('u','d'), seq_l),
u3_nul));
2013-09-29 00:21:18 +04:00
}
/* _http_request(): dispatch http request to %eyre
2013-09-29 00:21:18 +04:00
*/
static void
_http_request(u3_hreq* req_u, u3_noun recq)
2013-09-29 00:21:18 +04:00
{
u3_noun pox = _http_pox_to_noun(req_u->hon_u->htp_u->sev_l,
req_u->hon_u->coq_l,
req_u->seq_l);
2013-09-29 00:21:18 +04:00
u3_noun typ = _(req_u->hon_u->htp_u->lop) ? c3__chis : c3__this;
2013-09-29 00:21:18 +04:00
u3v_plan(pox, u3nq(typ,
req_u->hon_u->htp_u->sec,
u3nc(c3y, u3i_words(1, &req_u->hon_u->ipf_w)),
recq));
2013-09-29 00:21:18 +04:00
}
/* _http_request_kill(): kill http request in %eyre.
*/
static void
_http_request_kill(u3_hreq* req_u)
{
2014-09-11 04:01:32 +04:00
u3_noun pox = _http_pox_to_noun(req_u->hon_u->htp_u->sev_l,
req_u->hon_u->coq_l,
req_u->seq_l);
2014-11-06 03:20:01 +03:00
u3v_plan(pox, u3nc(c3__thud, u3_nul));
}
/* _http_respond(): attach %eyre response to open request.
2013-09-29 00:21:18 +04:00
*/
static void
_http_respond(c3_l sev_l, c3_l coq_l, c3_l seq_l, u3_noun rep)
2013-09-29 00:21:18 +04:00
{
u3_http* htp_u;
u3_hcon* hon_u;
u3_hreq* req_u;
2017-11-28 06:58:52 +03:00
c3_w bug_w = u3C.wag_w & u3o_verbose;
if ( !(htp_u = _http_serv_find(sev_l)) ) {
2017-11-28 06:58:52 +03:00
if ( bug_w ) {
uL(fprintf(uH, "http: server not found: %x\r\n", sev_l));
2015-11-19 05:31:12 +03:00
}
2014-03-20 02:40:40 +04:00
}
2017-11-28 06:58:52 +03:00
else if ( !(hon_u = _http_conn_find(htp_u, coq_l)) ) {
if ( bug_w ) {
uL(fprintf(uH, "http: connection not found: %x/%d\r\n", sev_l, coq_l));
2015-11-19 05:31:12 +03:00
}
2013-09-29 00:21:18 +04:00
}
2017-11-28 06:58:52 +03:00
else if ( !(req_u = _http_req_find(hon_u, seq_l)) ) {
if ( bug_w ) {
uL(fprintf(uH, "http: request not found: %x/%d/%d\r\n", sev_l, coq_l, seq_l));
2015-11-19 05:31:12 +03:00
}
2013-09-29 00:21:18 +04:00
}
2017-11-28 06:58:52 +03:00
else {
u3_noun p_rep, q_rep, r_rep;
2013-09-29 00:21:18 +04:00
if ( c3n == u3r_trel(rep, &p_rep, &q_rep, &r_rep) ) {
uL(fprintf(uH, "http: strange response\n"));
}
else {
_http_send_response(req_u, u3k(p_rep), u3k(q_rep), u3k(r_rep));
}
}
2013-10-30 22:42:24 +04:00
u3z(rep);
2013-10-30 22:42:24 +04:00
}
/* _http_init_h2o(): initialize h2o ctx and handlers for server.
2013-09-29 00:21:18 +04:00
*/
static void
_http_init_h2o(u3_http* htp_u)
2013-09-29 00:21:18 +04:00
{
2018-03-03 08:41:35 +03:00
htp_u->fig_u = c3_calloc(sizeof(*htp_u->fig_u));
h2o_config_init(htp_u->fig_u);
// XX use u3_Host.ops_u.nam_c? Or ship.urbit.org? Multiple hosts?
// see https://github.com/urbit/urbit/issues/914
htp_u->hos_u = h2o_config_register_host(htp_u->fig_u,
h2o_iovec_init(H2O_STRLIT("default")),
htp_u->por_w);
2018-03-03 08:41:35 +03:00
htp_u->ctx_u = c3_calloc(sizeof(*htp_u->ctx_u));
htp_u->cep_u = c3_calloc(sizeof(*htp_u->cep_u));
htp_u->cep_u->ctx = (h2o_context_t*)htp_u->ctx_u;
htp_u->cep_u->hosts = htp_u->fig_u->hosts;
2013-09-29 00:21:18 +04:00
2017-11-27 09:34:47 +03:00
if ( c3y == htp_u->sec ) {
htp_u->cep_u->ssl_ctx = tls_u;
}
2018-03-03 08:41:35 +03:00
htp_u->han_u = h2o_create_handler(&htp_u->hos_u->fallback_path, sizeof(*htp_u->han_u));
htp_u->han_u->on_req = _http_handle_new_req;
2018-03-03 08:41:35 +03:00
h2o_context_init(htp_u->ctx_u, u3L, htp_u->fig_u);
2013-09-29 00:21:18 +04:00
}
/* _http_start(): start http server.
*/
static void
_http_start(u3_http* htp_u)
2013-09-29 00:21:18 +04:00
{
struct sockaddr_in adr_u;
memset(&adr_u, 0, sizeof(adr_u));
adr_u.sin_family = AF_INET;
2016-02-11 22:44:28 +03:00
if ( c3y == htp_u->lop ) {
inet_pton(AF_INET, "127.0.0.1", &adr_u.sin_addr);
2016-02-11 22:44:28 +03:00
}
else {
adr_u.sin_addr.s_addr = INADDR_ANY;
2016-02-11 22:44:28 +03:00
}
2013-09-29 00:21:18 +04:00
2017-11-27 09:34:47 +03:00
if ( c3y == htp_u->sec && 0 == tls_u ) {
uL(fprintf(uH, "http: secure server error: no tls config\n"));
htp_u->por_w = 0;
2017-12-09 05:18:02 +03:00
return;
2017-11-27 09:34:47 +03:00
}
uv_tcp_init(u3L, &htp_u->wax_u);
2013-09-29 00:21:18 +04:00
/* Try ascending ports.
*/
while ( 1 ) {
2018-03-03 08:41:35 +03:00
c3_i sas_i;
2013-09-29 00:21:18 +04:00
adr_u.sin_port = htons(htp_u->por_w);
2018-03-03 08:41:35 +03:00
sas_i = uv_tcp_bind(&htp_u->wax_u, (const struct sockaddr*)&adr_u, 0);
2013-09-29 00:21:18 +04:00
2018-03-03 08:41:35 +03:00
if ( 0 != sas_i ||
0 != (sas_i = uv_listen((uv_stream_t*)&htp_u->wax_u,
TCP_BACKLOG, _http_listen_cb)) ) {
if ( UV_EADDRINUSE == sas_i ) {
htp_u->por_w++;
2013-09-29 00:21:18 +04:00
continue;
}
2018-03-03 08:41:35 +03:00
uL(fprintf(uH, "http: listen: %s\n", uv_strerror(sas_i)));
htp_u->por_w = 0;
2018-03-03 08:41:35 +03:00
return;
2013-09-29 00:21:18 +04:00
}
2018-03-03 08:41:35 +03:00
_http_init_h2o(htp_u);
2016-02-11 22:44:28 +03:00
uL(fprintf(uH, "http: live (%s, %s) on %d\n",
2017-11-27 09:34:47 +03:00
(c3y == htp_u->sec) ? "secure" : "insecure",
2016-02-11 22:44:28 +03:00
(c3y == htp_u->lop) ? "loopback" : "public",
2014-03-20 02:40:40 +04:00
htp_u->por_w));
2013-09-29 00:21:18 +04:00
break;
}
}
2017-11-27 09:34:47 +03:00
/* _http_init_tls: initialize OpenSSL context
*/
static SSL_CTX*
_http_init_tls()
{
SSL_CTX* tls_u = c3_malloc(sizeof(*tls_u));
SSL_library_init();
SSL_load_error_strings();
tls_u = SSL_CTX_new(TLSv1_2_server_method());
SSL_CTX_set_options(tls_u, SSL_OP_NO_SSLv2);
// SSL_CTX_set_verify(tls_u, SSL_VERIFY_NONE, NULL);
SSL_CTX_set_default_verify_paths(tls_u);
SSL_CTX_set_session_cache_mode(tls_u, SSL_SESS_CACHE_OFF);
SSL_CTX_set_cipher_list(tls_u,
"ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:"
"ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:"
"RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS");
c3_c pub_c[2048];
c3_c pir_c[2048];
c3_i ret_i;
ret_i = snprintf(pub_c, 2048, "%s/.urb/tls/certificate.pem", u3_Host.dir_c);
c3_assert(ret_i < 2048);
ret_i = snprintf(pir_c, 2048, "%s/.urb/tls/private.pem", u3_Host.dir_c);
c3_assert(ret_i < 2048);
// TODO: SSL_CTX_use_certificate_chain_file ?
if (SSL_CTX_use_certificate_file(tls_u, pub_c, SSL_FILETYPE_PEM) <= 0) {
uL(fprintf(uH, "https: failed to load certificate\n"));
// c3_assert(0);
return 0;
}
if (SSL_CTX_use_PrivateKey_file(tls_u, pir_c, SSL_FILETYPE_PEM) <= 0 ) {
uL(fprintf(uH, "https: failed to load private key\n"));
// c3_assert(0);
return 0;
}
return tls_u;
}
2016-12-12 23:37:43 +03:00
/* _http_write_ports_file(): update .http.ports
*/
void
_http_write_ports_file(c3_c *pax_c)
{
c3_i pal_i;
c3_c *paf_c;
c3_i por_i;
2016-12-12 23:37:43 +03:00
u3_http *htp_u;
pal_i = strlen(pax_c) + 13; /* includes NUL */
paf_c = u3a_malloc(pal_i);
snprintf(paf_c, pal_i, "%s/%s", pax_c, ".http.ports");
2016-12-12 23:37:43 +03:00
por_i = open(paf_c, O_WRONLY | O_CREAT | O_TRUNC, 0666);
u3a_free(paf_c);
2016-12-12 23:37:43 +03:00
for ( htp_u = u3_Host.htp_u; htp_u; htp_u = htp_u->nex_u ) {
if ( 0 < htp_u->por_w ) {
dprintf(por_i, "%u %s %s\n", htp_u->por_w,
(c3y == htp_u->sec) ? "secure" : "insecure",
(c3y == htp_u->lop) ? "loopback" : "public");
}
}
2016-12-12 23:37:43 +03:00
c3_sync(por_i);
close(por_i);
2016-12-12 23:37:43 +03:00
}
/* _http_release_ports_file(): remove .http.ports
2016-12-12 23:37:43 +03:00
*/
void
_http_release_ports_file(c3_c *pax_c)
{
c3_i pal_i;
c3_c *paf_c;
2016-12-12 23:37:43 +03:00
pal_i = strlen(pax_c) + 13; /* includes NUL */
paf_c = u3a_malloc(pal_i);
snprintf(paf_c, pal_i, "%s/%s", pax_c, ".http.ports");
2016-12-12 23:37:43 +03:00
unlink(paf_c);
u3a_free(paf_c);
}
// XX rename (is a card an effect?)
/* u3_http_ef_bake(): notify %eyre that we're live
*/
void
u3_http_ef_bake(void)
{
u3_noun pax = u3nq(u3_blip, c3__http, u3k(u3A->sen), u3_nul);
u3v_plan(pax, u3nc(c3__born, u3_nul));
}
/* u3_http_ef_thou(): send %thou from %eyre as http response.
*/
void
u3_http_ef_thou(c3_l sev_l,
c3_l coq_l,
c3_l seq_l,
u3_noun rep)
{
2017-11-28 05:51:44 +03:00
_http_respond(sev_l, coq_l, seq_l, rep);
2016-12-12 23:37:43 +03:00
}
2014-09-11 04:01:32 +04:00
/* u3_http_io_init(): initialize http I/O.
2013-09-29 00:21:18 +04:00
*/
void
2014-09-11 04:01:32 +04:00
u3_http_io_init()
2013-09-29 00:21:18 +04:00
{
2016-02-11 22:44:28 +03:00
// Lens port
{
u3_http *htp_u = c3_malloc(sizeof(*htp_u));
2016-02-11 22:44:28 +03:00
htp_u->sev_l = u3A->sev_l + 2;
htp_u->coq_l = 1;
htp_u->por_w = 12321;
htp_u->sec = c3n;
htp_u->lop = c3y;
2017-11-27 09:34:30 +03:00
htp_u->cep_u = 0;
htp_u->hos_u = 0;
2016-02-11 22:44:28 +03:00
htp_u->hon_u = 0;
htp_u->nex_u = 0;
htp_u->nex_u = u3_Host.htp_u;
u3_Host.htp_u = htp_u;
2016-02-11 22:44:28 +03:00
}
2017-11-27 09:34:30 +03:00
// Secure port.
2014-03-20 02:40:40 +04:00
{
u3_http *htp_u = c3_malloc(sizeof(*htp_u));
2014-03-20 02:40:40 +04:00
2014-09-11 04:01:32 +04:00
htp_u->sev_l = u3A->sev_l + 1;
2014-03-20 02:40:40 +04:00
htp_u->coq_l = 1;
htp_u->por_w = 8443;
2014-11-05 04:18:47 +03:00
htp_u->sec = c3y;
2016-02-11 22:44:28 +03:00
htp_u->lop = c3n;
2014-03-20 02:40:40 +04:00
2017-11-27 09:34:30 +03:00
htp_u->cep_u = 0;
htp_u->hos_u = 0;
2014-03-20 02:40:40 +04:00
htp_u->hon_u = 0;
htp_u->nex_u = 0;
htp_u->nex_u = u3_Host.htp_u;
u3_Host.htp_u = htp_u;
2014-03-20 02:40:40 +04:00
}
2017-11-27 09:34:30 +03:00
// Insecure port.
2014-03-20 02:40:40 +04:00
{
u3_http* htp_u = c3_malloc(sizeof(*htp_u));
2014-03-20 02:40:40 +04:00
2014-09-11 04:01:32 +04:00
htp_u->sev_l = u3A->sev_l;
2014-03-20 02:40:40 +04:00
htp_u->coq_l = 1;
htp_u->por_w = 8080;
2014-11-05 04:18:47 +03:00
htp_u->sec = c3n;
2016-02-11 22:44:28 +03:00
htp_u->lop = c3n;
2013-09-29 00:21:18 +04:00
htp_u->cep_u = 0;
htp_u->hos_u = 0;
2014-03-20 02:40:40 +04:00
htp_u->hon_u = 0;
htp_u->nex_u = 0;
2013-09-29 00:21:18 +04:00
htp_u->nex_u = u3_Host.htp_u;
u3_Host.htp_u = htp_u;
2014-03-20 02:40:40 +04:00
}
2014-02-27 04:40:53 +04:00
2017-11-27 09:34:47 +03:00
tls_u = _http_init_tls();
// XX why is this here?
2014-09-11 04:01:32 +04:00
u3_Host.ctp_u.coc_u = 0;
2013-09-29 00:21:18 +04:00
}
/* u3_http_io_talk(): start http I/O.
2014-01-17 12:12:05 +04:00
*/
void
2014-09-11 04:01:32 +04:00
u3_http_io_talk()
2014-01-17 12:12:05 +04:00
{
u3_http* htp_u;
2014-01-17 12:12:05 +04:00
for ( htp_u = u3_Host.htp_u; htp_u; htp_u = htp_u->nex_u ) {
2014-03-20 02:40:40 +04:00
_http_start(htp_u);
}
2016-12-12 23:37:43 +03:00
_http_write_ports_file(u3_Host.dir_c);
2014-01-17 12:12:05 +04:00
}
2014-09-11 04:01:32 +04:00
/* u3_http_io_poll(): poll kernel for http I/O.
2013-09-29 00:21:18 +04:00
*/
void
2014-09-11 04:01:32 +04:00
u3_http_io_poll(void)
2013-09-29 00:21:18 +04:00
{
}
2014-09-11 04:01:32 +04:00
/* u3_http_io_exit(): shut down http.
2013-09-29 00:21:18 +04:00
*/
void
2014-09-11 04:01:32 +04:00
u3_http_io_exit(void)
2013-09-29 00:21:18 +04:00
{
// XX shutdown servers cleanly
2016-12-12 23:37:43 +03:00
_http_release_ports_file(u3_Host.dir_c);
2013-09-29 00:21:18 +04:00
}