void h2o_proxy_register_reverse_proxy(h2o_pathconf_t *pathconf, h2o_url_t *upstream, h2o_proxy_config_vars_t *config) { struct rp_handler_t *self = (void *)h2o_create_handler(pathconf, sizeof(*self)); self->super.on_context_init = on_context_init; self->super.on_context_dispose = on_context_dispose; self->super.dispose = on_handler_dispose; self->super.on_req = on_req; if (config->keepalive_timeout != 0) { self->sockpool = h2o_mem_alloc(sizeof(*self->sockpool)); struct sockaddr_un sa; const char *to_sa_err; int is_ssl = upstream->scheme == &H2O_URL_SCHEME_HTTPS; if ((to_sa_err = h2o_url_host_to_sun(upstream->host, &sa)) == h2o_url_host_to_sun_err_is_not_unix_socket) { h2o_socketpool_init_by_hostport(self->sockpool, upstream->host, h2o_url_get_port(upstream), is_ssl, SIZE_MAX /* FIXME */); } else { assert(to_sa_err == NULL); h2o_socketpool_init_by_address(self->sockpool, (void *)&sa, sizeof(sa), is_ssl, SIZE_MAX /* FIXME */); } } h2o_url_copy(NULL, &self->upstream, upstream); h2o_strtolower(self->upstream.host.base, self->upstream.host.len); self->config = *config; if (self->config.ssl_ctx != NULL) CRYPTO_add(&self->config.ssl_ctx->references, 1, CRYPTO_LOCK_SSL_CTX); }
h2o_socketpool_target_t *h2o_socketpool_create_target(h2o_url_t *origin, h2o_socketpool_target_conf_t *lb_target_conf) { struct sockaddr_storage sa; socklen_t salen; h2o_socketpool_target_t *target = h2o_mem_alloc(sizeof(*target)); h2o_url_copy(NULL, &target->url, origin); assert(target->url.host.base[target->url.host.len] == '\0'); /* needs to be null-terminated in order to be used in SNI */ target->type = detect_target_type(origin, &sa, &salen); if (!(target->type == H2O_SOCKETPOOL_TYPE_SOCKADDR && sa.ss_family == AF_UNIX)) { h2o_strtolower(target->url.authority.base, target->url.authority.len); h2o_strtolower(target->url.host.base, target->url.host.len); } switch (target->type) { case H2O_SOCKETPOOL_TYPE_NAMED: target->peer.named_serv.base = h2o_mem_alloc(sizeof(H2O_UINT16_LONGEST_STR)); target->peer.named_serv.len = sprintf(target->peer.named_serv.base, "%u", (unsigned)h2o_url_get_port(&target->url)); break; case H2O_SOCKETPOOL_TYPE_SOCKADDR: assert(salen <= sizeof(target->peer.sockaddr.bytes)); memcpy(&target->peer.sockaddr.bytes, &sa, salen); target->peer.sockaddr.len = salen; break; } target->_shared.leased_count = 0; if (lb_target_conf != NULL) target->conf.weight_m1 = lb_target_conf->weight_m1; else { target->conf.weight_m1 = 0; } h2o_linklist_init_anchor(&target->_shared.sockets); return target; }