static void on_setup_ostream(h2o_filter_t *self, h2o_req_t *req, h2o_ostream_t **slot) { struct rproxy_t *rproxy; ssize_t reproxy_header_index; h2o_buf_t reproxy_url; /* do nothing unless 200 */ if (req->res.status != 200) goto SkipMe; if ((reproxy_header_index = h2o_find_header(&req->res.headers, H2O_TOKEN_X_REPROXY_URL, -1)) == -1) goto SkipMe; reproxy_url = req->res.headers.entries[reproxy_header_index].value; h2o_delete_header(&req->res.headers, reproxy_header_index); /* setup */ rproxy = (void*)h2o_add_ostream(req, sizeof(struct rproxy_t), slot); rproxy->filter = self; rproxy->super.do_send = send_chunk; rproxy->reproxy_url = h2o_strdup(&req->pool, reproxy_url.base, reproxy_url.len).base; /* next ostream is setup when send_chunk receives EOS */ return; SkipMe: h2o_setup_next_ostream(self, req, slot); }
static void on_setup_ostream(h2o_filter_t *self, h2o_req_t *req, h2o_ostream_t **slot) { h2o_iovec_t dest, method; ssize_t xru_index; /* obtain x-reproxy-url header, or skip to next ostream */ if ((xru_index = h2o_find_header(&req->res.headers, H2O_TOKEN_X_REPROXY_URL, -1)) == -1) { h2o_setup_next_ostream(req, slot); return; } dest = req->res.headers.entries[xru_index].value; h2o_delete_header(&req->res.headers, xru_index); /* setup params */ switch (req->res.status) { case 307: case 308: method = req->method; break; default: method = h2o_iovec_init(H2O_STRLIT("GET")); req->entity = (h2o_iovec_t){NULL}; break; } /* request internal redirect (is deferred) */ h2o_send_redirect_internal(req, method, dest.base, dest.len, 0); /* setup filter (that swallows the response until the timeout gets fired) */ h2o_ostream_t *ostream = h2o_add_ostream(req, H2O_ALIGNOF(*ostream), sizeof(*ostream), slot); ostream->do_send = on_send; }
static void on_setup_ostream(h2o_filter_t *self, h2o_req_t *req, h2o_ostream_t **slot) { chunked_encoder_t *encoder; /* do nothing if content-length is known */ if (req->res.content_length != SIZE_MAX) goto Next; /* RFC 2616 4.4 states that the following status codes (and response to a HEAD method) should not include message body */ if ((100 <= req->res.status && req->res.status <= 199) || req->res.status == 204 || req->res.status == 304) goto Next; else if (h2o_memis(req->method.base, req->method.len, H2O_STRLIT("HEAD")) == 0) goto Next; /* we cannot handle certain responses (like 101 switching protocols) */ if (req->res.status != 200) { req->http1_is_persistent = 0; goto Next; } /* skip if content-encoding header is being set */ if (h2o_find_header(&req->res.headers, H2O_TOKEN_TRANSFER_ENCODING, -1) != -1) goto Next; /* set content-encoding header */ h2o_add_header(&req->pool, &req->res.headers, H2O_TOKEN_TRANSFER_ENCODING, H2O_STRLIT("chunked")); /* setup filter */ encoder = (void*)h2o_add_ostream(req, sizeof(chunked_encoder_t), slot); encoder->super.do_send = send_chunk; slot = &encoder->super.next; Next: h2o_setup_next_ostream(self, req, slot); }
static void on_setup_ostream(h2o_filter_t *_self, h2o_req_t *req, h2o_ostream_t **slot) { struct st_compress_filter_t *self = (void *)_self; struct st_compress_encoder_t *encoder; int compressible_types; h2o_compress_context_t *compressor; ssize_t i; if (req->version < 0x101) goto Next; if (req->res.status != 200) goto Next; if (h2o_memis(req->input.method.base, req->input.method.len, H2O_STRLIT("HEAD"))) goto Next; if (req->res.mime_attr == NULL) h2o_req_fill_mime_attributes(req); if (!req->res.mime_attr->is_compressible) goto Next; if (req->res.content_length < self->args.min_size) goto Next; /* skip if failed to gather the list of compressible types */ if ((compressible_types = h2o_get_compressible_types(&req->headers)) == 0) goto Next; /* skip if content-encoding header is being set (as well as obtain the location of accept-ranges) */ size_t content_encoding_header_index = -1, accept_ranges_header_index = -1; for (i = 0; i != req->res.headers.size; ++i) { if (req->res.headers.entries[i].name == &H2O_TOKEN_CONTENT_ENCODING->buf) content_encoding_header_index = i; else if (req->res.headers.entries[i].name == &H2O_TOKEN_ACCEPT_RANGES->buf) accept_ranges_header_index = i; else continue; } if (content_encoding_header_index != -1) goto Next; /* open the compressor */ #ifndef _MSC_VER #else #define H2O_USE_BROTLI 0 #endif #if H2O_USE_BROTLI if (self->args.brotli.quality != -1 && (compressible_types & H2O_COMPRESSIBLE_BROTLI) != 0) { compressor = h2o_compress_brotli_open(&req->pool, self->args.brotli.quality, req->res.content_length); } else #endif if (self->args.gzip.quality != -1 && (compressible_types & H2O_COMPRESSIBLE_GZIP) != 0) { compressor = h2o_compress_gzip_open(&req->pool, self->args.gzip.quality); } else { goto Next; } /* adjust the response headers */ req->res.content_length = SIZE_MAX; h2o_add_header(&req->pool, &req->res.headers, H2O_TOKEN_CONTENT_ENCODING, compressor->name.base, compressor->name.len); h2o_set_header_token(&req->pool, &req->res.headers, H2O_TOKEN_VARY, H2O_STRLIT("accept-encoding")); if (accept_ranges_header_index != -1) { req->res.headers.entries[accept_ranges_header_index].value = h2o_iovec_init(H2O_STRLIT("none")); } else { h2o_add_header(&req->pool, &req->res.headers, H2O_TOKEN_ACCEPT_RANGES, H2O_STRLIT("none")); } /* setup filter */ encoder = (void *)h2o_add_ostream(req, sizeof(*encoder), slot); encoder->super.do_send = do_send; slot = &encoder->super.next; encoder->compressor = compressor; /* adjust preferred chunk size (compress by 8192 bytes) */ if (req->preferred_chunk_size > BUF_SIZE) req->preferred_chunk_size = BUF_SIZE; Next: h2o_setup_next_ostream(req, slot); }
static void on_setup_ostream(h2o_filter_t *self, h2o_req_t *req, h2o_ostream_t **slot) { gzip_encoder_t *encoder; ssize_t i; if (req->version < 0x101) goto Next; if (req->res.status != 200) goto Next; if (h2o_memis(req->input.method.base, req->input.method.len, H2O_STRLIT("HEAD"))) goto Next; if (req->res.mime_attr == NULL) h2o_req_fill_mime_attributes(req); if (!req->res.mime_attr->is_compressible) goto Next; /* 100 is a rough estimate */ if (req->res.content_length <= 100) goto Next; /* skip if no accept-encoding is set */ if ((i = h2o_find_header(&req->headers, H2O_TOKEN_ACCEPT_ENCODING, -1)) == -1) goto Next; if (!h2o_contains_token(req->headers.entries[i].value.base, req->headers.entries[i].value.len, H2O_STRLIT("gzip"), ',')) goto Next; /* skip if content-encoding header is being set (as well as obtain the location of accept-ranges */ size_t content_encoding_header_index = -1, accept_ranges_header_index = -1; for (i = 0; i != req->res.headers.size; ++i) { if (req->res.headers.entries[i].name == &H2O_TOKEN_CONTENT_ENCODING->buf) content_encoding_header_index = i; else if (req->res.headers.entries[i].name == &H2O_TOKEN_ACCEPT_RANGES->buf) accept_ranges_header_index = i; else continue; } if (content_encoding_header_index != -1) goto Next; /* adjust the response headers */ req->res.content_length = SIZE_MAX; h2o_add_header(&req->pool, &req->res.headers, H2O_TOKEN_CONTENT_ENCODING, H2O_STRLIT("gzip")); h2o_add_header_token(&req->pool, &req->res.headers, H2O_TOKEN_VARY, H2O_STRLIT("accept-encoding")); if (accept_ranges_header_index != -1) { req->res.headers.entries[accept_ranges_header_index].value = h2o_iovec_init(H2O_STRLIT("none")); } else { h2o_add_header(&req->pool, &req->res.headers, H2O_TOKEN_ACCEPT_RANGES, H2O_STRLIT("none")); } /* setup filter */ encoder = (void *)h2o_add_ostream(req, sizeof(gzip_encoder_t), slot); encoder->super.do_send = send_gzip; encoder->super.stop = stop_gzip; slot = &encoder->super.next; encoder->bufs.capacity = 0; encoder->bufs.size = 0; encoder->zstream.zalloc = gzip_encoder_alloc; encoder->zstream.zfree = gzip_encoder_free; encoder->zstream.opaque = encoder; /* adjust preferred chunk size (compress by 8192 bytes) */ if (req->preferred_chunk_size > BUF_SIZE) req->preferred_chunk_size = BUF_SIZE; Next: h2o_setup_next_ostream(req, slot); }