2013-08-27 19:47:22 +02:00
|
|
|
/*
|
|
|
|
* nghttp2 - HTTP/2.0 C Library
|
|
|
|
*
|
|
|
|
* Copyright (c) 2012 Tatsuhiro Tsujikawa
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
* a copy of this software and associated documentation files (the
|
|
|
|
* "Software"), to deal in the Software without restriction, including
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
* the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be
|
|
|
|
* included in all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
#include "http2.h"
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
namespace nghttp2 {
|
|
|
|
|
|
|
|
namespace http2 {
|
|
|
|
|
2013-10-02 16:29:44 +02:00
|
|
|
std::string get_status_string(unsigned int status_code)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
|
|
|
switch(status_code) {
|
|
|
|
case 100: return "100 Continue";
|
|
|
|
case 101: return "101 Switching Protocols";
|
|
|
|
case 200: return "200 OK";
|
|
|
|
case 201: return "201 Created";
|
|
|
|
case 202: return "202 Accepted";
|
|
|
|
case 203: return "203 Non-Authoritative Information";
|
|
|
|
case 204: return "204 No Content";
|
|
|
|
case 205: return "205 Reset Content";
|
|
|
|
case 206: return "206 Partial Content";
|
|
|
|
case 300: return "300 Multiple Choices";
|
|
|
|
case 301: return "301 Moved Permanently";
|
|
|
|
case 302: return "302 Found";
|
|
|
|
case 303: return "303 See Other";
|
|
|
|
case 304: return "304 Not Modified";
|
|
|
|
case 305: return "305 Use Proxy";
|
|
|
|
// case 306: return "306 (Unused)";
|
|
|
|
case 307: return "307 Temporary Redirect";
|
|
|
|
case 400: return "400 Bad Request";
|
|
|
|
case 401: return "401 Unauthorized";
|
|
|
|
case 402: return "402 Payment Required";
|
|
|
|
case 403: return "403 Forbidden";
|
|
|
|
case 404: return "404 Not Found";
|
|
|
|
case 405: return "405 Method Not Allowed";
|
|
|
|
case 406: return "406 Not Acceptable";
|
|
|
|
case 407: return "407 Proxy Authentication Required";
|
|
|
|
case 408: return "408 Request Timeout";
|
|
|
|
case 409: return "409 Conflict";
|
|
|
|
case 410: return "410 Gone";
|
|
|
|
case 411: return "411 Length Required";
|
|
|
|
case 412: return "412 Precondition Failed";
|
|
|
|
case 413: return "413 Request Entity Too Large";
|
|
|
|
case 414: return "414 Request-URI Too Long";
|
|
|
|
case 415: return "415 Unsupported Media Type";
|
|
|
|
case 416: return "416 Requested Range Not Satisfiable";
|
|
|
|
case 417: return "417 Expectation Failed";
|
|
|
|
case 500: return "500 Internal Server Error";
|
|
|
|
case 501: return "501 Not Implemented";
|
|
|
|
case 502: return "502 Bad Gateway";
|
|
|
|
case 503: return "503 Service Unavailable";
|
|
|
|
case 504: return "504 Gateway Timeout";
|
|
|
|
case 505: return "505 HTTP Version Not Supported";
|
2013-10-02 16:29:44 +02:00
|
|
|
default: return util::utos(status_code);
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void capitalize(std::string& s, size_t offset)
|
|
|
|
{
|
|
|
|
s[offset] = util::upcase(s[offset]);
|
|
|
|
for(size_t i = offset+1, eoi = s.size(); i < eoi; ++i) {
|
|
|
|
if(s[i-1] == '-') {
|
|
|
|
s[i] = util::upcase(s[i]);
|
|
|
|
} else {
|
|
|
|
s[i] = util::lowcase(s[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-16 18:16:53 +01:00
|
|
|
bool lws(const char *value)
|
2013-09-11 16:24:32 +02:00
|
|
|
{
|
2013-12-25 16:08:42 +01:00
|
|
|
for(; *value; ++value) {
|
|
|
|
switch(*value) {
|
|
|
|
case '\t':
|
|
|
|
case ' ':
|
|
|
|
continue;
|
|
|
|
default:
|
2014-01-16 18:16:53 +01:00
|
|
|
return false;
|
2013-09-11 16:24:32 +02:00
|
|
|
}
|
|
|
|
}
|
2014-01-16 18:16:53 +01:00
|
|
|
return true;
|
2013-09-11 16:24:32 +02:00
|
|
|
}
|
|
|
|
|
2013-08-27 19:47:22 +02:00
|
|
|
void sanitize_header_value(std::string& s, size_t offset)
|
|
|
|
{
|
2013-12-25 16:08:42 +01:00
|
|
|
// Since both nghttp2 and spdylay do not allow \n and \r in header
|
|
|
|
// values, we don't have to do this anymore.
|
|
|
|
|
|
|
|
// for(size_t i = offset, eoi = s.size(); i < eoi; ++i) {
|
|
|
|
// if(s[i] == '\r' || s[i] == '\n') {
|
|
|
|
// s[i] = ' ';
|
|
|
|
// }
|
|
|
|
// }
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
|
2013-12-21 09:49:31 +01:00
|
|
|
void copy_url_component(std::string& dest, const http_parser_url *u, int field,
|
2013-08-27 19:47:22 +02:00
|
|
|
const char* url)
|
|
|
|
{
|
|
|
|
if(u->field_set & (1 << field)) {
|
|
|
|
dest.assign(url+u->field_data[field].off, u->field_data[field].len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool check_http2_allowed_header(const char *name)
|
|
|
|
{
|
|
|
|
return check_http2_allowed_header(reinterpret_cast<const uint8_t*>(name),
|
|
|
|
strlen(name));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool check_http2_allowed_header(const uint8_t *name, size_t namelen)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
!util::strieq("connection", name, namelen) &&
|
|
|
|
!util::strieq("host", name, namelen) &&
|
|
|
|
!util::strieq("keep-alive", name, namelen) &&
|
|
|
|
!util::strieq("proxy-connection", name, namelen) &&
|
|
|
|
!util::strieq("te", name, namelen) &&
|
|
|
|
!util::strieq("transfer-encoding", name, namelen) &&
|
|
|
|
!util::strieq("upgrade", name, namelen);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const char *DISALLOWED_HD[] = {
|
|
|
|
"connection",
|
|
|
|
"keep-alive",
|
|
|
|
"proxy-connection",
|
|
|
|
"te",
|
|
|
|
"transfer-encoding",
|
|
|
|
"upgrade",
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
size_t DISALLOWED_HDLEN = sizeof(DISALLOWED_HD)/sizeof(DISALLOWED_HD[0]);
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const char *IGN_HD[] = {
|
|
|
|
"connection",
|
|
|
|
"expect",
|
|
|
|
"http2-settings",
|
|
|
|
"keep-alive",
|
|
|
|
"proxy-connection",
|
|
|
|
"te",
|
|
|
|
"transfer-encoding",
|
|
|
|
"upgrade",
|
|
|
|
"via",
|
|
|
|
"x-forwarded-for",
|
|
|
|
"x-forwarded-proto",
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
size_t IGN_HDLEN = sizeof(IGN_HD)/sizeof(IGN_HD[0]);
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const char *HTTP1_IGN_HD[] = {
|
|
|
|
"connection",
|
2013-11-16 13:15:55 +01:00
|
|
|
"cookie",
|
2013-08-27 19:47:22 +02:00
|
|
|
"expect",
|
|
|
|
"http2-settings",
|
|
|
|
"keep-alive",
|
|
|
|
"proxy-connection",
|
|
|
|
"upgrade",
|
|
|
|
"via",
|
|
|
|
"x-forwarded-for",
|
|
|
|
"x-forwarded-proto",
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
size_t HTTP1_IGN_HDLEN = sizeof(HTTP1_IGN_HD)/sizeof(HTTP1_IGN_HD[0]);
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
2013-12-05 13:54:36 +01:00
|
|
|
auto nv_name_less = [](const nghttp2_nv& lhs, const nghttp2_nv& rhs)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2013-12-05 13:54:36 +01:00
|
|
|
return nghttp2_nv_compare_name(&lhs, &rhs) < 0;
|
2013-08-27 19:47:22 +02:00
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
bool name_less(const Headers::value_type& lhs,
|
|
|
|
const Headers::value_type& rhs)
|
|
|
|
{
|
|
|
|
return lhs.first < rhs.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool check_http2_headers(const Headers& nva)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
|
|
|
for(size_t i = 0; i < DISALLOWED_HDLEN; ++i) {
|
2014-01-16 15:41:13 +01:00
|
|
|
if(std::binary_search(std::begin(nva), std::end(nva),
|
|
|
|
std::make_pair(DISALLOWED_HD[i], ""), name_less)) {
|
2013-08-27 19:47:22 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
void normalize_headers(Headers& nva)
|
|
|
|
{
|
|
|
|
for(auto& kv : nva) {
|
|
|
|
util::inp_strlower(kv.first);
|
|
|
|
}
|
|
|
|
std::stable_sort(std::begin(nva), std::end(nva), name_less);
|
|
|
|
}
|
|
|
|
|
2013-12-05 13:54:36 +01:00
|
|
|
std::vector<nghttp2_nv> sort_nva(const nghttp2_nv *nva, size_t nvlen)
|
2013-11-13 15:56:02 +01:00
|
|
|
{
|
2013-12-05 13:54:36 +01:00
|
|
|
auto v = std::vector<nghttp2_nv>(&nva[0], &nva[nvlen]);
|
|
|
|
std::sort(std::begin(v), std::end(v), nv_name_less);
|
|
|
|
auto res = std::vector<nghttp2_nv>();
|
2013-11-13 15:56:02 +01:00
|
|
|
res.reserve(nvlen);
|
|
|
|
for(size_t i = 0; i < nvlen; ++i) {
|
2013-12-05 13:54:36 +01:00
|
|
|
if(v[i].valuelen == 0) {
|
|
|
|
res.push_back(v[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
auto j = v[i].value;
|
|
|
|
auto end = v[i].value + v[i].valuelen;
|
|
|
|
for(;;) {
|
|
|
|
// Skip 0 length value
|
2013-12-22 07:33:37 +01:00
|
|
|
j = std::find_if(j, end,
|
|
|
|
[](uint8_t c)
|
|
|
|
{
|
|
|
|
return c != '\0';
|
|
|
|
});
|
2013-12-05 13:54:36 +01:00
|
|
|
if(j == end) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
auto l = std::find(j, end, '\0');
|
|
|
|
res.push_back({v[i].name, j, v[i].namelen, static_cast<uint16_t>(l-j)});
|
|
|
|
j = l;
|
|
|
|
}
|
2013-11-13 15:56:02 +01:00
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
Headers::value_type to_header(const uint8_t *name, size_t namelen,
|
|
|
|
const uint8_t *value, size_t valuelen)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
return std::make_pair(std::string(reinterpret_cast<const char*>(name),
|
|
|
|
namelen),
|
|
|
|
std::string(reinterpret_cast<const char*>(value),
|
|
|
|
valuelen));
|
|
|
|
}
|
|
|
|
|
|
|
|
void split_add_header(Headers& nva,
|
|
|
|
const uint8_t *name, size_t namelen,
|
|
|
|
const uint8_t *value, size_t valuelen)
|
|
|
|
{
|
|
|
|
if(valuelen == 0) {
|
|
|
|
nva.push_back(to_header(name, namelen, value, valuelen));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto j = value;
|
|
|
|
auto end = value + valuelen;
|
|
|
|
for(;;) {
|
|
|
|
// Skip 0 length value
|
|
|
|
j = std::find_if(j, end,
|
|
|
|
[](uint8_t c)
|
|
|
|
{
|
|
|
|
return c != '\0';
|
|
|
|
});
|
|
|
|
if(j == end) {
|
|
|
|
break;
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
2014-01-16 15:41:13 +01:00
|
|
|
auto l = std::find(j, end, '\0');
|
|
|
|
nva.push_back(to_header(name, namelen, j, l-j));
|
|
|
|
j = l;
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
const Headers::value_type* get_unique_header(const Headers& nva,
|
|
|
|
const char *name)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
auto nv = Headers::value_type(name, "");
|
|
|
|
auto i = std::lower_bound(std::begin(nva), std::end(nva), nv, name_less);
|
|
|
|
if(i != std::end(nva) && (*i).first == nv.first) {
|
|
|
|
auto j = i + 1;
|
|
|
|
if(j == std::end(nva) || (*j).first != nv.first) {
|
|
|
|
return &(*i);
|
|
|
|
}
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
const Headers::value_type* get_header(const Headers& nva, const char *name)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
auto nv = Headers::value_type(name, "");
|
|
|
|
auto i = std::lower_bound(std::begin(nva), std::end(nva), nv, name_less);
|
|
|
|
if(i != std::end(nva) && (*i).first == nv.first) {
|
|
|
|
return &(*i);
|
2013-10-25 14:50:56 +02:00
|
|
|
}
|
2014-01-16 15:41:13 +01:00
|
|
|
return nullptr;
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
std::string value_to_str(const Headers::value_type *nv)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2013-10-25 14:50:56 +02:00
|
|
|
if(nv) {
|
2014-01-16 15:41:13 +01:00
|
|
|
return nv->second;
|
2013-10-25 14:50:56 +02:00
|
|
|
}
|
|
|
|
return "";
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
bool value_lws(const Headers::value_type *nv)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
return (*nv).second.find_first_not_of("\t ") == std::string::npos;
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
bool non_empty_value(const Headers::value_type *nv)
|
2013-10-25 14:50:56 +02:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
return nv && !value_lws(nv);
|
2013-10-25 14:50:56 +02:00
|
|
|
}
|
|
|
|
|
2013-11-28 13:36:04 +01:00
|
|
|
nghttp2_nv make_nv(const std::string& name, const std::string& value)
|
|
|
|
{
|
|
|
|
return {
|
|
|
|
(uint8_t*)name.c_str(),
|
|
|
|
(uint8_t*)value.c_str(),
|
|
|
|
(uint16_t)name.size(), (uint16_t)value.size()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
Headers concat_norm_headers(Headers headers)
|
2013-12-06 16:32:14 +01:00
|
|
|
{
|
2014-01-16 15:41:13 +01:00
|
|
|
auto res = Headers();
|
2013-12-06 16:32:14 +01:00
|
|
|
res.reserve(headers.size());
|
|
|
|
for(auto& kv : headers) {
|
|
|
|
if(!res.empty() && res.back().first == kv.first &&
|
|
|
|
kv.first != "cookie" && kv.first != "set-cookie") {
|
|
|
|
if(!kv.second.empty()) {
|
|
|
|
res.back().second.append(1, '\0');
|
|
|
|
res.back().second += kv.second;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
res.push_back(std::move(kv));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2013-11-28 13:36:04 +01:00
|
|
|
void copy_norm_headers_to_nva
|
2014-01-16 15:41:13 +01:00
|
|
|
(std::vector<nghttp2_nv>& nva, const Headers& headers)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
2013-10-23 16:18:24 +02:00
|
|
|
size_t i, j;
|
2013-08-27 19:47:22 +02:00
|
|
|
for(i = 0, j = 0; i < headers.size() && j < IGN_HDLEN;) {
|
|
|
|
int rv = strcmp(headers[i].first.c_str(), IGN_HD[j]);
|
|
|
|
if(rv < 0) {
|
2014-01-16 15:41:13 +01:00
|
|
|
if(!headers[i].first.empty() && headers[i].first.c_str()[0] != ':') {
|
|
|
|
nva.push_back(make_nv(headers[i].first, headers[i].second));
|
|
|
|
}
|
2013-08-27 19:47:22 +02:00
|
|
|
++i;
|
|
|
|
} else if(rv > 0) {
|
|
|
|
++j;
|
|
|
|
} else {
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(; i < headers.size(); ++i) {
|
2014-01-16 15:41:13 +01:00
|
|
|
if(!headers[i].first.empty() && headers[i].first.c_str()[0] != ':') {
|
|
|
|
nva.push_back(make_nv(headers[i].first, headers[i].second));
|
|
|
|
}
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void build_http1_headers_from_norm_headers
|
2014-01-16 15:41:13 +01:00
|
|
|
(std::string& hdrs, const Headers& headers)
|
2013-08-27 19:47:22 +02:00
|
|
|
{
|
|
|
|
size_t i, j;
|
|
|
|
for(i = 0, j = 0; i < headers.size() && j < HTTP1_IGN_HDLEN;) {
|
|
|
|
int rv = strcmp(headers[i].first.c_str(), HTTP1_IGN_HD[j]);
|
|
|
|
if(rv < 0) {
|
2014-01-16 15:41:13 +01:00
|
|
|
if(!headers[i].first.empty() && headers[i].first.c_str()[0] != ':') {
|
|
|
|
hdrs += headers[i].first;
|
|
|
|
capitalize(hdrs, hdrs.size()-headers[i].first.size());
|
|
|
|
hdrs += ": ";
|
|
|
|
hdrs += headers[i].second;
|
|
|
|
sanitize_header_value(hdrs, hdrs.size() - headers[i].second.size());
|
|
|
|
hdrs += "\r\n";
|
|
|
|
}
|
2013-08-27 19:47:22 +02:00
|
|
|
++i;
|
|
|
|
} else if(rv > 0) {
|
|
|
|
++j;
|
|
|
|
} else {
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(; i < headers.size(); ++i) {
|
2014-01-16 15:41:13 +01:00
|
|
|
if(!headers[i].first.empty() && headers[i].first.c_str()[0] != ':') {
|
|
|
|
hdrs += headers[i].first;
|
|
|
|
capitalize(hdrs, hdrs.size()-headers[i].first.size());
|
|
|
|
hdrs += ": ";
|
|
|
|
hdrs += headers[i].second;
|
|
|
|
sanitize_header_value(hdrs, hdrs.size() - headers[i].second.size());
|
|
|
|
hdrs += "\r\n";
|
|
|
|
}
|
2013-08-27 19:47:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-29 16:51:01 +01:00
|
|
|
int32_t determine_window_update_transmission(nghttp2_session *session,
|
|
|
|
int32_t stream_id)
|
|
|
|
{
|
|
|
|
int32_t recv_length, window_size;
|
|
|
|
if(stream_id == 0) {
|
|
|
|
recv_length = nghttp2_session_get_effective_recv_data_length(session);
|
|
|
|
window_size = nghttp2_session_get_effective_local_window_size(session);
|
|
|
|
} else {
|
|
|
|
recv_length = nghttp2_session_get_stream_effective_recv_data_length
|
|
|
|
(session, stream_id);
|
|
|
|
window_size = nghttp2_session_get_stream_effective_local_window_size
|
|
|
|
(session, stream_id);
|
|
|
|
}
|
|
|
|
if(recv_length != -1 && window_size != -1) {
|
|
|
|
if(recv_length >= window_size / 2) {
|
|
|
|
return recv_length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-11-17 15:52:19 +01:00
|
|
|
void dump_nv(FILE *out, const char **nv)
|
|
|
|
{
|
|
|
|
for(size_t i = 0; nv[i]; i += 2) {
|
|
|
|
fwrite(nv[i], strlen(nv[i]), 1, out);
|
|
|
|
fwrite(": ", 2, 1, out);
|
|
|
|
fwrite(nv[i+1], strlen(nv[i+1]), 1, out);
|
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
}
|
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
fflush(out);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dump_nv(FILE *out, const nghttp2_nv *nva, size_t nvlen)
|
|
|
|
{
|
2013-12-06 16:32:14 +01:00
|
|
|
// |nva| may have NULL-concatenated header fields
|
|
|
|
auto v = sort_nva(nva, nvlen);
|
|
|
|
for(auto& nv : v) {
|
|
|
|
fwrite(nv.name, nv.namelen, 1, out);
|
2013-11-17 15:52:19 +01:00
|
|
|
fwrite(": ", 2, 1, out);
|
2013-12-06 16:32:14 +01:00
|
|
|
fwrite(nv.value, nv.valuelen, 1, out);
|
2013-11-17 15:52:19 +01:00
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
}
|
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
fflush(out);
|
|
|
|
}
|
|
|
|
|
2014-01-16 15:41:13 +01:00
|
|
|
void dump_nv(FILE *out, const Headers& nva)
|
|
|
|
{
|
|
|
|
for(auto& nv : nva) {
|
|
|
|
fwrite(nv.first.c_str(), nv.first.size(), 1, out);
|
|
|
|
fwrite(": ", 2, 1, out);
|
|
|
|
fwrite(nv.second.c_str(), nv.second.size(), 1, out);
|
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
}
|
|
|
|
fwrite("\n", 1, 1, out);
|
|
|
|
fflush(out);
|
|
|
|
}
|
|
|
|
|
2013-12-21 09:49:31 +01:00
|
|
|
std::string rewrite_location_uri(const std::string& uri,
|
|
|
|
const http_parser_url& u,
|
|
|
|
const std::string& request_host,
|
|
|
|
const std::string& upstream_scheme,
|
2013-12-21 10:35:53 +01:00
|
|
|
uint16_t upstream_port)
|
2013-12-21 09:49:31 +01:00
|
|
|
{
|
|
|
|
// We just rewrite host and optionally port. We don't rewrite https
|
|
|
|
// link. Not sure it happens in practice.
|
|
|
|
if(u.field_set & (1 << UF_SCHEMA)) {
|
|
|
|
auto field = &u.field_data[UF_SCHEMA];
|
|
|
|
if(!util::streq("http", &uri[field->off], field->len)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if((u.field_set & (1 << UF_HOST)) == 0) {
|
|
|
|
return "";
|
|
|
|
}
|
2013-12-21 10:26:11 +01:00
|
|
|
auto field = &u.field_data[UF_HOST];
|
|
|
|
if(!util::startsWith(std::begin(request_host), std::end(request_host),
|
|
|
|
&uri[field->off], &uri[field->off] + field->len) ||
|
|
|
|
(request_host.size() != field->len &&
|
|
|
|
request_host[field->len] != ':')) {
|
2013-12-21 09:49:31 +01:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
std::string res = upstream_scheme;
|
|
|
|
res += "://";
|
|
|
|
res.append(&uri[field->off], field->len);
|
|
|
|
if(upstream_scheme == "http") {
|
|
|
|
if(upstream_port != 80) {
|
|
|
|
res += ":";
|
|
|
|
res += util::utos(upstream_port);
|
|
|
|
}
|
|
|
|
} else if(upstream_scheme == "https") {
|
|
|
|
if(upstream_port != 443) {
|
|
|
|
res += ":";
|
|
|
|
res += util::utos(upstream_port);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(u.field_set & (1 << UF_PATH)) {
|
|
|
|
field = &u.field_data[UF_PATH];
|
|
|
|
res.append(&uri[field->off], field->len);
|
|
|
|
}
|
|
|
|
if(u.field_set & (1 << UF_QUERY)) {
|
|
|
|
field = &u.field_data[UF_QUERY];
|
|
|
|
res += "?";
|
|
|
|
res.append(&uri[field->off], field->len);
|
|
|
|
}
|
|
|
|
if(u.field_set & (1 << UF_FRAGMENT)) {
|
|
|
|
field = &u.field_data[UF_FRAGMENT];
|
|
|
|
res += "#";
|
|
|
|
res.append(&uri[field->off], field->len);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2014-01-16 18:16:53 +01:00
|
|
|
namespace {
|
|
|
|
int VALID_HD_NAME_CHARS[] = {
|
|
|
|
0 /* NUL */, 0 /* SOH */, 0 /* STX */, 0 /* ETX */,
|
|
|
|
0 /* EOT */, 0 /* ENQ */, 0 /* ACK */, 0 /* BEL */,
|
|
|
|
0 /* BS */, 0 /* HT */, 0 /* LF */, 0 /* VT */,
|
|
|
|
0 /* FF */, 0 /* CR */, 0 /* SO */, 0 /* SI */,
|
|
|
|
0 /* DLE */, 0 /* DC1 */, 0 /* DC2 */, 0 /* DC3 */,
|
|
|
|
0 /* DC4 */, 0 /* NAK */, 0 /* SYN */, 0 /* ETB */,
|
|
|
|
0 /* CAN */, 0 /* EM */, 0 /* SUB */, 0 /* ESC */,
|
|
|
|
0 /* FS */, 0 /* GS */, 0 /* RS */, 0 /* US */,
|
|
|
|
0 /* SPC */, 1 /* ! */, 0 /* " */, 1 /* # */,
|
|
|
|
1 /* $ */, 1 /* % */, 1 /* & */, 1 /* ' */,
|
|
|
|
0 /* ( */, 0 /* ) */, 1 /* * */, 1 /* + */,
|
|
|
|
0 /* , */, 1 /* - */, 1 /* . */, 0 /* / */,
|
|
|
|
1 /* 0 */, 1 /* 1 */, 1 /* 2 */, 1 /* 3 */,
|
|
|
|
1 /* 4 */, 1 /* 5 */, 1 /* 6 */, 1 /* 7 */,
|
|
|
|
1 /* 8 */, 1 /* 9 */, 0 /* : */, 0 /* ; */,
|
|
|
|
0 /* < */, 0 /* = */, 0 /* > */, 0 /* ? */,
|
|
|
|
0 /* @ */, 0 /* A */, 0 /* B */, 0 /* C */,
|
|
|
|
0 /* D */, 0 /* E */, 0 /* F */, 0 /* G */,
|
|
|
|
0 /* H */, 0 /* I */, 0 /* J */, 0 /* K */,
|
|
|
|
0 /* L */, 0 /* M */, 0 /* N */, 0 /* O */,
|
|
|
|
0 /* P */, 0 /* Q */, 0 /* R */, 0 /* S */,
|
|
|
|
0 /* T */, 0 /* U */, 0 /* V */, 0 /* W */,
|
|
|
|
0 /* X */, 0 /* Y */, 0 /* Z */, 0 /* [ */,
|
|
|
|
0 /* \ */, 0 /* ] */, 1 /* ^ */, 1 /* _ */,
|
|
|
|
1 /* ` */, 1 /* a */, 1 /* b */, 1 /* c */,
|
|
|
|
1 /* d */, 1 /* e */, 1 /* f */, 1 /* g */,
|
|
|
|
1 /* h */, 1 /* i */, 1 /* j */, 1 /* k */,
|
|
|
|
1 /* l */, 1 /* m */, 1 /* n */, 1 /* o */,
|
|
|
|
1 /* p */, 1 /* q */, 1 /* r */, 1 /* s */,
|
|
|
|
1 /* t */, 1 /* u */, 1 /* v */, 1 /* w */,
|
|
|
|
1 /* x */, 1 /* y */, 1 /* z */, 0 /* { */,
|
|
|
|
1 /* | */, 0 /* } */, 1 /* ~ */, 0 /* DEL */,
|
|
|
|
0 /* 0x80 */, 0 /* 0x81 */, 0 /* 0x82 */, 0 /* 0x83 */,
|
|
|
|
0 /* 0x84 */, 0 /* 0x85 */, 0 /* 0x86 */, 0 /* 0x87 */,
|
|
|
|
0 /* 0x88 */, 0 /* 0x89 */, 0 /* 0x8a */, 0 /* 0x8b */,
|
|
|
|
0 /* 0x8c */, 0 /* 0x8d */, 0 /* 0x8e */, 0 /* 0x8f */,
|
|
|
|
0 /* 0x90 */, 0 /* 0x91 */, 0 /* 0x92 */, 0 /* 0x93 */,
|
|
|
|
0 /* 0x94 */, 0 /* 0x95 */, 0 /* 0x96 */, 0 /* 0x97 */,
|
|
|
|
0 /* 0x98 */, 0 /* 0x99 */, 0 /* 0x9a */, 0 /* 0x9b */,
|
|
|
|
0 /* 0x9c */, 0 /* 0x9d */, 0 /* 0x9e */, 0 /* 0x9f */,
|
|
|
|
0 /* 0xa0 */, 0 /* 0xa1 */, 0 /* 0xa2 */, 0 /* 0xa3 */,
|
|
|
|
0 /* 0xa4 */, 0 /* 0xa5 */, 0 /* 0xa6 */, 0 /* 0xa7 */,
|
|
|
|
0 /* 0xa8 */, 0 /* 0xa9 */, 0 /* 0xaa */, 0 /* 0xab */,
|
|
|
|
0 /* 0xac */, 0 /* 0xad */, 0 /* 0xae */, 0 /* 0xaf */,
|
|
|
|
0 /* 0xb0 */, 0 /* 0xb1 */, 0 /* 0xb2 */, 0 /* 0xb3 */,
|
|
|
|
0 /* 0xb4 */, 0 /* 0xb5 */, 0 /* 0xb6 */, 0 /* 0xb7 */,
|
|
|
|
0 /* 0xb8 */, 0 /* 0xb9 */, 0 /* 0xba */, 0 /* 0xbb */,
|
|
|
|
0 /* 0xbc */, 0 /* 0xbd */, 0 /* 0xbe */, 0 /* 0xbf */,
|
|
|
|
0 /* 0xc0 */, 0 /* 0xc1 */, 0 /* 0xc2 */, 0 /* 0xc3 */,
|
|
|
|
0 /* 0xc4 */, 0 /* 0xc5 */, 0 /* 0xc6 */, 0 /* 0xc7 */,
|
|
|
|
0 /* 0xc8 */, 0 /* 0xc9 */, 0 /* 0xca */, 0 /* 0xcb */,
|
|
|
|
0 /* 0xcc */, 0 /* 0xcd */, 0 /* 0xce */, 0 /* 0xcf */,
|
|
|
|
0 /* 0xd0 */, 0 /* 0xd1 */, 0 /* 0xd2 */, 0 /* 0xd3 */,
|
|
|
|
0 /* 0xd4 */, 0 /* 0xd5 */, 0 /* 0xd6 */, 0 /* 0xd7 */,
|
|
|
|
0 /* 0xd8 */, 0 /* 0xd9 */, 0 /* 0xda */, 0 /* 0xdb */,
|
|
|
|
0 /* 0xdc */, 0 /* 0xdd */, 0 /* 0xde */, 0 /* 0xdf */,
|
|
|
|
0 /* 0xe0 */, 0 /* 0xe1 */, 0 /* 0xe2 */, 0 /* 0xe3 */,
|
|
|
|
0 /* 0xe4 */, 0 /* 0xe5 */, 0 /* 0xe6 */, 0 /* 0xe7 */,
|
|
|
|
0 /* 0xe8 */, 0 /* 0xe9 */, 0 /* 0xea */, 0 /* 0xeb */,
|
|
|
|
0 /* 0xec */, 0 /* 0xed */, 0 /* 0xee */, 0 /* 0xef */,
|
|
|
|
0 /* 0xf0 */, 0 /* 0xf1 */, 0 /* 0xf2 */, 0 /* 0xf3 */,
|
|
|
|
0 /* 0xf4 */, 0 /* 0xf5 */, 0 /* 0xf6 */, 0 /* 0xf7 */,
|
|
|
|
0 /* 0xf8 */, 0 /* 0xf9 */, 0 /* 0xfa */, 0 /* 0xfb */,
|
|
|
|
0 /* 0xfc */, 0 /* 0xfd */, 0 /* 0xfe */, 0 /* 0xff */
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
int check_header_name(const uint8_t *name, size_t len, int nocase)
|
|
|
|
{
|
|
|
|
const uint8_t *last;
|
|
|
|
if(len == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if(*name == ':') {
|
|
|
|
if(len == 1) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
++name;
|
|
|
|
--len;
|
|
|
|
}
|
|
|
|
for(last = name + len; name != last; ++name) {
|
|
|
|
if(nocase && 'A' <= *name && *name <= 'Z') continue;
|
|
|
|
if(!VALID_HD_NAME_CHARS[*name]) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
int check_header_name(const uint8_t *name, size_t len)
|
|
|
|
{
|
|
|
|
return check_header_name(name, len, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int check_header_name_nocase(const uint8_t *name, size_t len)
|
|
|
|
{
|
|
|
|
return check_header_name(name, len, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
int VALID_HD_VALUE_CHARS[] = {
|
|
|
|
1 /* NUL */, 0 /* SOH */, 0 /* STX */, 0 /* ETX */,
|
|
|
|
0 /* EOT */, 0 /* ENQ */, 0 /* ACK */, 0 /* BEL */,
|
|
|
|
0 /* BS */, 1 /* HT */, 0 /* LF */, 0 /* VT */,
|
|
|
|
0 /* FF */, 0 /* CR */, 0 /* SO */, 0 /* SI */,
|
|
|
|
0 /* DLE */, 0 /* DC1 */, 0 /* DC2 */, 0 /* DC3 */,
|
|
|
|
0 /* DC4 */, 0 /* NAK */, 0 /* SYN */, 0 /* ETB */,
|
|
|
|
0 /* CAN */, 0 /* EM */, 0 /* SUB */, 0 /* ESC */,
|
|
|
|
0 /* FS */, 0 /* GS */, 0 /* RS */, 0 /* US */,
|
|
|
|
1 /* SPC */, 1 /* ! */, 1 /* " */, 1 /* # */,
|
|
|
|
1 /* $ */, 1 /* % */, 1 /* & */, 1 /* ' */,
|
|
|
|
1 /* ( */, 1 /* ) */, 1 /* * */, 1 /* + */,
|
|
|
|
1 /* , */, 1 /* - */, 1 /* . */, 1 /* / */,
|
|
|
|
1 /* 0 */, 1 /* 1 */, 1 /* 2 */, 1 /* 3 */,
|
|
|
|
1 /* 4 */, 1 /* 5 */, 1 /* 6 */, 1 /* 7 */,
|
|
|
|
1 /* 8 */, 1 /* 9 */, 1 /* : */, 1 /* ; */,
|
|
|
|
1 /* < */, 1 /* = */, 1 /* > */, 1 /* ? */,
|
|
|
|
1 /* @ */, 1 /* A */, 1 /* B */, 1 /* C */,
|
|
|
|
1 /* D */, 1 /* E */, 1 /* F */, 1 /* G */,
|
|
|
|
1 /* H */, 1 /* I */, 1 /* J */, 1 /* K */,
|
|
|
|
1 /* L */, 1 /* M */, 1 /* N */, 1 /* O */,
|
|
|
|
1 /* P */, 1 /* Q */, 1 /* R */, 1 /* S */,
|
|
|
|
1 /* T */, 1 /* U */, 1 /* V */, 1 /* W */,
|
|
|
|
1 /* X */, 1 /* Y */, 1 /* Z */, 1 /* [ */,
|
|
|
|
1 /* \ */, 1 /* ] */, 1 /* ^ */, 1 /* _ */,
|
|
|
|
1 /* ` */, 1 /* a */, 1 /* b */, 1 /* c */,
|
|
|
|
1 /* d */, 1 /* e */, 1 /* f */, 1 /* g */,
|
|
|
|
1 /* h */, 1 /* i */, 1 /* j */, 1 /* k */,
|
|
|
|
1 /* l */, 1 /* m */, 1 /* n */, 1 /* o */,
|
|
|
|
1 /* p */, 1 /* q */, 1 /* r */, 1 /* s */,
|
|
|
|
1 /* t */, 1 /* u */, 1 /* v */, 1 /* w */,
|
|
|
|
1 /* x */, 1 /* y */, 1 /* z */, 1 /* { */,
|
|
|
|
1 /* | */, 1 /* } */, 1 /* ~ */, 0 /* DEL */,
|
|
|
|
1 /* 0x80 */, 1 /* 0x81 */, 1 /* 0x82 */, 1 /* 0x83 */,
|
|
|
|
1 /* 0x84 */, 1 /* 0x85 */, 1 /* 0x86 */, 1 /* 0x87 */,
|
|
|
|
1 /* 0x88 */, 1 /* 0x89 */, 1 /* 0x8a */, 1 /* 0x8b */,
|
|
|
|
1 /* 0x8c */, 1 /* 0x8d */, 1 /* 0x8e */, 1 /* 0x8f */,
|
|
|
|
1 /* 0x90 */, 1 /* 0x91 */, 1 /* 0x92 */, 1 /* 0x93 */,
|
|
|
|
1 /* 0x94 */, 1 /* 0x95 */, 1 /* 0x96 */, 1 /* 0x97 */,
|
|
|
|
1 /* 0x98 */, 1 /* 0x99 */, 1 /* 0x9a */, 1 /* 0x9b */,
|
|
|
|
1 /* 0x9c */, 1 /* 0x9d */, 1 /* 0x9e */, 1 /* 0x9f */,
|
|
|
|
1 /* 0xa0 */, 1 /* 0xa1 */, 1 /* 0xa2 */, 1 /* 0xa3 */,
|
|
|
|
1 /* 0xa4 */, 1 /* 0xa5 */, 1 /* 0xa6 */, 1 /* 0xa7 */,
|
|
|
|
1 /* 0xa8 */, 1 /* 0xa9 */, 1 /* 0xaa */, 1 /* 0xab */,
|
|
|
|
1 /* 0xac */, 1 /* 0xad */, 1 /* 0xae */, 1 /* 0xaf */,
|
|
|
|
1 /* 0xb0 */, 1 /* 0xb1 */, 1 /* 0xb2 */, 1 /* 0xb3 */,
|
|
|
|
1 /* 0xb4 */, 1 /* 0xb5 */, 1 /* 0xb6 */, 1 /* 0xb7 */,
|
|
|
|
1 /* 0xb8 */, 1 /* 0xb9 */, 1 /* 0xba */, 1 /* 0xbb */,
|
|
|
|
1 /* 0xbc */, 1 /* 0xbd */, 1 /* 0xbe */, 1 /* 0xbf */,
|
|
|
|
1 /* 0xc0 */, 1 /* 0xc1 */, 1 /* 0xc2 */, 1 /* 0xc3 */,
|
|
|
|
1 /* 0xc4 */, 1 /* 0xc5 */, 1 /* 0xc6 */, 1 /* 0xc7 */,
|
|
|
|
1 /* 0xc8 */, 1 /* 0xc9 */, 1 /* 0xca */, 1 /* 0xcb */,
|
|
|
|
1 /* 0xcc */, 1 /* 0xcd */, 1 /* 0xce */, 1 /* 0xcf */,
|
|
|
|
1 /* 0xd0 */, 1 /* 0xd1 */, 1 /* 0xd2 */, 1 /* 0xd3 */,
|
|
|
|
1 /* 0xd4 */, 1 /* 0xd5 */, 1 /* 0xd6 */, 1 /* 0xd7 */,
|
|
|
|
1 /* 0xd8 */, 1 /* 0xd9 */, 1 /* 0xda */, 1 /* 0xdb */,
|
|
|
|
1 /* 0xdc */, 1 /* 0xdd */, 1 /* 0xde */, 1 /* 0xdf */,
|
|
|
|
1 /* 0xe0 */, 1 /* 0xe1 */, 1 /* 0xe2 */, 1 /* 0xe3 */,
|
|
|
|
1 /* 0xe4 */, 1 /* 0xe5 */, 1 /* 0xe6 */, 1 /* 0xe7 */,
|
|
|
|
1 /* 0xe8 */, 1 /* 0xe9 */, 1 /* 0xea */, 1 /* 0xeb */,
|
|
|
|
1 /* 0xec */, 1 /* 0xed */, 1 /* 0xee */, 1 /* 0xef */,
|
|
|
|
1 /* 0xf0 */, 1 /* 0xf1 */, 1 /* 0xf2 */, 1 /* 0xf3 */,
|
|
|
|
1 /* 0xf4 */, 1 /* 0xf5 */, 1 /* 0xf6 */, 1 /* 0xf7 */,
|
|
|
|
1 /* 0xf8 */, 1 /* 0xf9 */, 1 /* 0xfa */, 1 /* 0xfb */,
|
|
|
|
1 /* 0xfc */, 1 /* 0xfd */, 1 /* 0xfe */, 1 /* 0xff */
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
int check_header_value(const uint8_t* value, size_t len)
|
|
|
|
{
|
|
|
|
const uint8_t *last;
|
|
|
|
for(last = value + len; value != last; ++value) {
|
|
|
|
if(!VALID_HD_VALUE_CHARS[*value]) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int check_nv(const uint8_t *name, size_t namelen,
|
|
|
|
const uint8_t *value, size_t valuelen)
|
|
|
|
{
|
|
|
|
if(!check_header_name(name, namelen)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if(!check_header_value(value, valuelen)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-08-27 19:47:22 +02:00
|
|
|
} // namespace http2
|
|
|
|
|
|
|
|
} // namespace nghttp2
|