2012-01-29 16:34:10 +01:00
|
|
|
/*
|
2014-03-30 12:09:21 +02:00
|
|
|
* nghttp2 - HTTP/2 C Library
|
2012-01-29 16:34:10 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
#ifndef UTIL_H
|
|
|
|
#define UTIL_H
|
|
|
|
|
2013-07-12 17:19:03 +02:00
|
|
|
#include "nghttp2_config.h"
|
2012-07-27 15:11:13 +02:00
|
|
|
|
2015-05-13 15:30:35 +02:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2014-01-08 17:27:56 +01:00
|
|
|
#include <unistd.h>
|
2015-05-13 15:30:35 +02:00
|
|
|
#endif // HAVE_UNISTD_H
|
2014-01-08 17:27:56 +01:00
|
|
|
#include <getopt.h>
|
2015-05-13 15:30:35 +02:00
|
|
|
#ifdef HAVE_NETDB_H
|
2015-02-24 15:50:18 +01:00
|
|
|
#include <netdb.h>
|
2015-05-13 15:30:35 +02:00
|
|
|
#endif // HAVE_NETDB_H
|
2014-01-08 17:27:56 +01:00
|
|
|
|
2015-01-31 10:13:07 +01:00
|
|
|
#include <cmath>
|
2012-11-23 13:14:39 +01:00
|
|
|
#include <cstring>
|
2014-11-14 15:14:39 +01:00
|
|
|
#include <cassert>
|
2012-01-29 16:34:10 +01:00
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <algorithm>
|
2012-02-07 17:50:58 +01:00
|
|
|
#include <sstream>
|
2013-07-21 16:49:12 +02:00
|
|
|
#include <memory>
|
2014-11-15 14:06:16 +01:00
|
|
|
#include <chrono>
|
2015-10-28 16:21:36 +01:00
|
|
|
#include <map>
|
2016-01-15 15:04:58 +01:00
|
|
|
#include <random>
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2014-02-27 13:47:04 +01:00
|
|
|
#include "http-parser/http_parser.h"
|
|
|
|
|
2016-02-13 15:21:32 +01:00
|
|
|
#include "template.h"
|
2016-02-21 07:28:11 +01:00
|
|
|
#include "network.h"
|
2016-03-12 10:00:50 +01:00
|
|
|
#include "allocator.h"
|
2016-02-13 15:21:32 +01:00
|
|
|
|
2013-07-12 17:19:03 +02:00
|
|
|
namespace nghttp2 {
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2016-03-24 17:41:06 +01:00
|
|
|
constexpr auto NGHTTP2_H2_ALPN = StringRef::from_lit("\x2h2");
|
|
|
|
constexpr auto NGHTTP2_H2 = StringRef::from_lit("h2");
|
|
|
|
|
2015-02-06 13:45:14 +01:00
|
|
|
// The additional HTTP/2 protocol ALPN protocol identifier we also
|
2015-04-05 16:00:59 +02:00
|
|
|
// supports for our applications to make smooth migration into final
|
|
|
|
// h2 ALPN ID.
|
2016-03-24 17:41:06 +01:00
|
|
|
constexpr auto NGHTTP2_H2_16_ALPN = StringRef::from_lit("\x5h2-16");
|
|
|
|
constexpr auto NGHTTP2_H2_16 = StringRef::from_lit("h2-16");
|
2015-02-20 15:50:17 +01:00
|
|
|
|
2016-03-24 17:41:06 +01:00
|
|
|
constexpr auto NGHTTP2_H2_14_ALPN = StringRef::from_lit("\x5h2-14");
|
|
|
|
constexpr auto NGHTTP2_H2_14 = StringRef::from_lit("h2-14");
|
2014-12-15 14:51:34 +01:00
|
|
|
|
2016-03-24 17:41:06 +01:00
|
|
|
constexpr auto NGHTTP2_H1_1_ALPN = StringRef::from_lit("\x8http/1.1");
|
|
|
|
constexpr auto NGHTTP2_H1_1 = StringRef::from_lit("http/1.1");
|
2015-09-14 15:33:48 +02:00
|
|
|
|
2012-01-29 16:34:10 +01:00
|
|
|
namespace util {
|
|
|
|
|
2015-11-27 16:41:39 +01:00
|
|
|
inline bool is_alpha(const char c) {
|
2015-09-07 16:23:07 +02:00
|
|
|
return ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z');
|
|
|
|
}
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-11-27 16:41:39 +01:00
|
|
|
inline bool is_digit(const char c) { return '0' <= c && c <= '9'; }
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-11-27 16:41:39 +01:00
|
|
|
inline bool is_hex_digit(const char c) {
|
|
|
|
return is_digit(c) || ('A' <= c && c <= 'F') || ('a' <= c && c <= 'f');
|
2015-09-07 16:23:07 +02:00
|
|
|
}
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-11-27 16:14:11 +01:00
|
|
|
bool in_rfc3986_unreserved_chars(const char c);
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-03-05 15:22:21 +01:00
|
|
|
bool in_rfc3986_sub_delims(const char c);
|
|
|
|
|
2014-04-03 06:20:50 +02:00
|
|
|
// Returns true if |c| is in token (HTTP-p1, Section 3.2.6)
|
|
|
|
bool in_token(char c);
|
|
|
|
|
2015-02-07 08:09:49 +01:00
|
|
|
bool in_attr_char(char c);
|
|
|
|
|
2015-02-10 14:35:10 +01:00
|
|
|
// Returns integer corresponding to hex notation |c|. It is undefined
|
2015-11-27 16:41:39 +01:00
|
|
|
// if is_hex_digit(c) is false.
|
2015-02-10 14:35:10 +01:00
|
|
|
uint32_t hex_to_uint(char c);
|
|
|
|
|
2015-11-27 16:14:11 +01:00
|
|
|
std::string percent_encode(const unsigned char *target, size_t len);
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-11-27 16:14:11 +01:00
|
|
|
std::string percent_encode(const std::string &target);
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2015-03-05 15:22:21 +01:00
|
|
|
// percent-encode path component of URI |s|.
|
|
|
|
std::string percent_encode_path(const std::string &s);
|
|
|
|
|
2015-03-04 13:31:27 +01:00
|
|
|
template <typename InputIt>
|
2015-11-27 16:14:11 +01:00
|
|
|
std::string percent_decode(InputIt first, InputIt last) {
|
2015-03-04 13:31:27 +01:00
|
|
|
std::string result;
|
2016-01-26 07:23:42 +01:00
|
|
|
result.resize(last - first);
|
|
|
|
auto p = std::begin(result);
|
2015-03-04 13:31:27 +01:00
|
|
|
for (; first != last; ++first) {
|
2016-01-26 07:23:42 +01:00
|
|
|
if (*first != '%') {
|
|
|
|
*p++ = *first;
|
2015-03-04 13:31:27 +01:00
|
|
|
continue;
|
|
|
|
}
|
2016-01-26 07:23:42 +01:00
|
|
|
|
|
|
|
if (first + 1 != last && first + 2 != last && is_hex_digit(*(first + 1)) &&
|
|
|
|
is_hex_digit(*(first + 2))) {
|
|
|
|
*p++ = (hex_to_uint(*(first + 1)) << 4) + hex_to_uint(*(first + 2));
|
|
|
|
first += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p++ = *first;
|
2015-03-04 13:31:27 +01:00
|
|
|
}
|
2016-01-26 07:23:42 +01:00
|
|
|
result.resize(p - std::begin(result));
|
2015-03-04 13:31:27 +01:00
|
|
|
return result;
|
|
|
|
}
|
2012-01-29 16:34:10 +01:00
|
|
|
|
2016-03-19 10:33:36 +01:00
|
|
|
StringRef percent_decode(BlockAllocator &balloc, const StringRef &src);
|
|
|
|
|
2014-04-03 06:20:50 +02:00
|
|
|
// Percent encode |target| if character is not in token or '%'.
|
2014-11-27 15:39:04 +01:00
|
|
|
std::string percent_encode_token(const std::string &target);
|
2014-04-03 06:20:50 +02:00
|
|
|
|
2014-10-27 16:17:32 +01:00
|
|
|
// Returns quotedString version of |target|. Currently, this function
|
|
|
|
// just replace '"' with '\"'.
|
2014-11-27 15:39:04 +01:00
|
|
|
std::string quote_string(const std::string &target);
|
2014-10-27 16:17:32 +01:00
|
|
|
|
2013-07-15 17:15:04 +02:00
|
|
|
std::string format_hex(const unsigned char *s, size_t len);
|
|
|
|
|
2015-11-12 16:53:29 +01:00
|
|
|
template <size_t N> std::string format_hex(const unsigned char(&s)[N]) {
|
2015-07-17 18:49:20 +02:00
|
|
|
return format_hex(s, N);
|
|
|
|
}
|
|
|
|
|
2015-07-26 19:12:07 +02:00
|
|
|
template <size_t N> std::string format_hex(const std::array<uint8_t, N> &s) {
|
|
|
|
return format_hex(s.data(), s.size());
|
|
|
|
}
|
|
|
|
|
2012-02-07 17:50:58 +01:00
|
|
|
std::string http_date(time_t t);
|
|
|
|
|
2014-11-23 08:40:38 +01:00
|
|
|
// Returns given time |t| from epoch in Common Log format (e.g.,
|
|
|
|
// 03/Jul/2014:00:19:38 +0900)
|
|
|
|
std::string common_log_date(time_t t);
|
|
|
|
|
|
|
|
// Returns given millisecond |ms| from epoch in ISO 8601 format (e.g.,
|
|
|
|
// 2014-11-15T12:58:24.741Z)
|
|
|
|
std::string iso8601_date(int64_t ms);
|
|
|
|
|
2016-03-12 09:31:48 +01:00
|
|
|
time_t parse_http_date(const StringRef &s);
|
2012-02-07 17:50:58 +01:00
|
|
|
|
2015-02-10 14:35:10 +01:00
|
|
|
char upcase(char c);
|
|
|
|
|
|
|
|
inline char lowcase(char c) {
|
|
|
|
static unsigned char tbl[] = {
|
|
|
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
|
|
|
|
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29,
|
|
|
|
30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44,
|
|
|
|
45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59,
|
|
|
|
60, 61, 62, 63, 64, 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j',
|
|
|
|
'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y',
|
|
|
|
'z', 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104,
|
|
|
|
105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119,
|
|
|
|
120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130, 131, 132, 133, 134,
|
|
|
|
135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 145, 146, 147, 148, 149,
|
|
|
|
150, 151, 152, 153, 154, 155, 156, 157, 158, 159, 160, 161, 162, 163, 164,
|
|
|
|
165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176, 177, 178, 179,
|
|
|
|
180, 181, 182, 183, 184, 185, 186, 187, 188, 189, 190, 191, 192, 193, 194,
|
|
|
|
195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207, 208, 209,
|
|
|
|
210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223, 224,
|
|
|
|
225, 226, 227, 228, 229, 230, 231, 232, 233, 234, 235, 236, 237, 238, 239,
|
|
|
|
240, 241, 242, 243, 244, 245, 246, 247, 248, 249, 250, 251, 252, 253, 254,
|
|
|
|
255,
|
|
|
|
};
|
|
|
|
return tbl[static_cast<unsigned char>(c)];
|
|
|
|
}
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename InputIterator1, typename InputIterator2>
|
2015-11-27 16:31:42 +01:00
|
|
|
bool starts_with(InputIterator1 first1, InputIterator1 last1,
|
|
|
|
InputIterator2 first2, InputIterator2 last2) {
|
2014-11-27 15:39:04 +01:00
|
|
|
if (last1 - first1 < last2 - first2) {
|
2012-05-19 10:05:12 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return std::equal(first2, last2, first1);
|
|
|
|
}
|
|
|
|
|
2016-03-25 15:20:48 +01:00
|
|
|
template <typename S, typename T> bool starts_with(const S &a, const T &b) {
|
|
|
|
return starts_with(a.begin(), a.end(), b.begin(), b.end());
|
2015-02-22 09:01:19 +01:00
|
|
|
}
|
|
|
|
|
2012-05-19 10:05:12 +02:00
|
|
|
struct CaseCmp {
|
2014-11-27 15:39:04 +01:00
|
|
|
bool operator()(char lhs, char rhs) const {
|
2015-02-10 14:35:10 +01:00
|
|
|
return lowcase(lhs) == lowcase(rhs);
|
2012-05-19 10:05:12 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename InputIterator1, typename InputIterator2>
|
2015-11-27 16:31:42 +01:00
|
|
|
bool istarts_with(InputIterator1 first1, InputIterator1 last1,
|
|
|
|
InputIterator2 first2, InputIterator2 last2) {
|
2014-11-27 15:39:04 +01:00
|
|
|
if (last1 - first1 < last2 - first2) {
|
2012-05-19 10:05:12 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return std::equal(first2, last2, first1, CaseCmp());
|
|
|
|
}
|
|
|
|
|
2016-03-25 15:18:31 +01:00
|
|
|
template <typename S, typename T> bool istarts_with(const S &a, const T &b) {
|
|
|
|
return istarts_with(a.begin(), a.end(), b.begin(), b.end());
|
2015-02-10 14:35:10 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 15:18:31 +01:00
|
|
|
template <typename T, typename CharT, size_t N>
|
|
|
|
bool istarts_with_l(const T &a, const CharT(&b)[N]) {
|
|
|
|
return istarts_with(a.begin(), a.end(), b, b + N - 1);
|
2016-02-28 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename InputIterator1, typename InputIterator2>
|
2015-11-27 16:36:43 +01:00
|
|
|
bool ends_with(InputIterator1 first1, InputIterator1 last1,
|
|
|
|
InputIterator2 first2, InputIterator2 last2) {
|
2014-11-27 15:39:04 +01:00
|
|
|
if (last1 - first1 < last2 - first2) {
|
2012-02-07 17:50:58 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-11-27 15:39:04 +01:00
|
|
|
return std::equal(first2, last2, last1 - (last2 - first2));
|
2012-02-07 17:50:58 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 15:04:44 +01:00
|
|
|
template <typename T, typename S> bool ends_with(const T &a, const S &b) {
|
|
|
|
return ends_with(a.begin(), a.end(), b.begin(), b.end());
|
2015-02-10 14:35:10 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 15:04:44 +01:00
|
|
|
template <typename T, typename CharT, size_t N>
|
|
|
|
bool ends_with_l(const T &a, const CharT(&b)[N]) {
|
|
|
|
return ends_with(a.begin(), a.end(), b, b + N - 1);
|
2016-03-19 10:33:36 +01:00
|
|
|
}
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename InputIterator1, typename InputIterator2>
|
2015-11-27 16:36:43 +01:00
|
|
|
bool iends_with(InputIterator1 first1, InputIterator1 last1,
|
|
|
|
InputIterator2 first2, InputIterator2 last2) {
|
2014-11-27 15:39:04 +01:00
|
|
|
if (last1 - first1 < last2 - first2) {
|
2012-11-22 13:46:15 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-11-27 15:39:04 +01:00
|
|
|
return std::equal(first2, last2, last1 - (last2 - first2), CaseCmp());
|
2012-11-22 13:46:15 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 15:00:34 +01:00
|
|
|
template <typename T, typename S> bool iends_with(const T &a, const S &b) {
|
|
|
|
return iends_with(a.begin(), a.end(), b.begin(), b.end());
|
2015-02-10 14:35:10 +01:00
|
|
|
}
|
2012-02-07 17:50:58 +01:00
|
|
|
|
2016-03-25 15:00:34 +01:00
|
|
|
template <typename T, typename CharT, size_t N>
|
|
|
|
bool iends_with_l(const T &a, const CharT(&b)[N]) {
|
|
|
|
return iends_with(a.begin(), a.end(), b, b + N - 1);
|
2016-03-09 13:15:32 +01:00
|
|
|
}
|
|
|
|
|
2016-01-16 15:54:21 +01:00
|
|
|
template <typename InputIt1, typename InputIt2>
|
|
|
|
bool strieq(InputIt1 first1, InputIt1 last1, InputIt2 first2, InputIt2 last2) {
|
|
|
|
if (std::distance(first1, last1) != std::distance(first2, last2)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::equal(first1, last1, first2, CaseCmp());
|
|
|
|
}
|
|
|
|
|
2016-03-25 14:45:33 +01:00
|
|
|
template <typename T, typename S> bool strieq(const T &a, const S &b) {
|
|
|
|
return strieq(a.begin(), a.end(), b.begin(), b.end());
|
2015-07-20 15:37:26 +02:00
|
|
|
}
|
|
|
|
|
2016-02-07 10:16:47 +01:00
|
|
|
template <typename CharT, typename InputIt, size_t N>
|
|
|
|
bool strieq_l(const CharT(&a)[N], InputIt b, size_t blen) {
|
2016-03-24 15:16:20 +01:00
|
|
|
return strieq(a, a + (N - 1), b, b + blen);
|
2015-02-22 07:32:48 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 14:51:20 +01:00
|
|
|
template <typename CharT, size_t N, typename T>
|
|
|
|
bool strieq_l(const CharT(&a)[N], const T &b) {
|
|
|
|
return strieq(a, a + (N - 1), b.begin(), b.end());
|
2016-03-09 13:15:32 +01:00
|
|
|
}
|
|
|
|
|
2016-03-24 15:16:20 +01:00
|
|
|
template <typename InputIt1, typename InputIt2>
|
|
|
|
bool streq(InputIt1 first1, InputIt1 last1, InputIt2 first2, InputIt2 last2) {
|
|
|
|
if (std::distance(first1, last1) != std::distance(first2, last2)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return std::equal(first1, last1, first2);
|
|
|
|
}
|
|
|
|
|
2016-03-25 14:48:13 +01:00
|
|
|
template <typename T, typename S> bool streq(const T &a, const S &b) {
|
|
|
|
return streq(a.begin(), a.end(), b.begin(), b.end());
|
2016-03-24 15:16:20 +01:00
|
|
|
}
|
|
|
|
|
2016-02-07 10:16:47 +01:00
|
|
|
template <typename CharT, typename InputIt, size_t N>
|
|
|
|
bool streq_l(const CharT(&a)[N], InputIt b, size_t blen) {
|
2016-03-24 16:02:07 +01:00
|
|
|
return streq(a, a + (N - 1), b, b + blen);
|
2015-02-20 15:50:17 +01:00
|
|
|
}
|
|
|
|
|
2016-03-25 14:53:31 +01:00
|
|
|
template <typename CharT, size_t N, typename T>
|
|
|
|
bool streq_l(const CharT(&a)[N], const T &b) {
|
|
|
|
return streq(a, a + (N - 1), b.begin(), b.end());
|
2016-03-19 10:33:36 +01:00
|
|
|
}
|
|
|
|
|
2016-03-24 16:27:59 +01:00
|
|
|
// Returns true if |a| contains |b|. If both |a| and |b| are empty,
|
|
|
|
// this function returns false.
|
2016-03-25 14:56:23 +01:00
|
|
|
template <typename S, typename T> bool strifind(const S &a, const T &b) {
|
|
|
|
return std::search(a.begin(), a.end(), b.begin(), b.end(), CaseCmp()) !=
|
|
|
|
a.end();
|
|
|
|
}
|
2012-06-04 16:48:31 +02:00
|
|
|
|
2015-07-09 19:52:11 +02:00
|
|
|
template <typename InputIt> void inp_strlower(InputIt first, InputIt last) {
|
|
|
|
std::transform(first, last, first, lowcase);
|
|
|
|
}
|
|
|
|
|
2013-08-27 17:09:46 +02:00
|
|
|
// Lowercase |s| in place.
|
2015-02-10 14:35:10 +01:00
|
|
|
inline void inp_strlower(std::string &s) {
|
2015-07-09 19:52:11 +02:00
|
|
|
inp_strlower(std::begin(s), std::end(s));
|
2015-02-10 14:35:10 +01:00
|
|
|
}
|
2013-08-27 17:09:46 +02:00
|
|
|
|
2015-01-31 10:13:07 +01:00
|
|
|
// Returns string representation of |n| with 2 fractional digits.
|
|
|
|
std::string dtos(double n);
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename T> std::string utos(T n) {
|
2013-01-16 14:43:11 +01:00
|
|
|
std::string res;
|
2014-11-27 15:39:04 +01:00
|
|
|
if (n == 0) {
|
2013-01-16 14:43:11 +01:00
|
|
|
res = "0";
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
int i = 0;
|
|
|
|
T t = n;
|
2014-11-27 15:39:04 +01:00
|
|
|
for (; t; t /= 10, ++i)
|
|
|
|
;
|
2013-01-16 14:43:11 +01:00
|
|
|
res.resize(i);
|
|
|
|
--i;
|
2014-11-27 15:39:04 +01:00
|
|
|
for (; n; --i, n /= 10) {
|
|
|
|
res[i] = (n % 10) + '0';
|
2013-01-16 14:43:11 +01:00
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2016-03-10 14:42:07 +01:00
|
|
|
template <typename T, typename OutputIt> OutputIt utos(OutputIt dst, T n) {
|
|
|
|
if (n == 0) {
|
|
|
|
*dst++ = '0';
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
int i = 0;
|
|
|
|
T t = n;
|
|
|
|
for (; t; t /= 10, ++i)
|
|
|
|
;
|
|
|
|
--i;
|
|
|
|
auto p = dst + i;
|
2016-03-10 15:50:04 +01:00
|
|
|
auto res = p + 1;
|
2016-03-10 14:42:07 +01:00
|
|
|
for (; n; --i, n /= 10) {
|
|
|
|
*p-- = (n % 10) + '0';
|
|
|
|
}
|
2016-03-10 15:50:04 +01:00
|
|
|
return res;
|
2016-03-10 14:42:07 +01:00
|
|
|
}
|
|
|
|
|
2016-03-12 10:00:50 +01:00
|
|
|
template <typename T>
|
|
|
|
StringRef make_string_ref_uint(BlockAllocator &balloc, T n) {
|
|
|
|
auto iov = make_byte_ref(balloc, str_size("18446744073709551615") + 1);
|
|
|
|
auto p = iov.base;
|
|
|
|
p = util::utos(p, n);
|
|
|
|
*p = '\0';
|
|
|
|
return StringRef{iov.base, p};
|
|
|
|
}
|
|
|
|
|
2016-01-11 10:12:31 +01:00
|
|
|
template <typename T> std::string utos_unit(T n) {
|
2015-01-13 13:54:53 +01:00
|
|
|
char u = 0;
|
|
|
|
if (n >= (1 << 30)) {
|
|
|
|
u = 'G';
|
|
|
|
n /= (1 << 30);
|
|
|
|
} else if (n >= (1 << 20)) {
|
|
|
|
u = 'M';
|
|
|
|
n /= (1 << 20);
|
|
|
|
} else if (n >= (1 << 10)) {
|
|
|
|
u = 'K';
|
|
|
|
n /= (1 << 10);
|
|
|
|
}
|
|
|
|
if (u == 0) {
|
|
|
|
return utos(n);
|
|
|
|
}
|
|
|
|
return utos(n) + u;
|
|
|
|
}
|
|
|
|
|
2016-01-11 10:12:31 +01:00
|
|
|
// Like utos_unit(), but 2 digits fraction part is followed.
|
|
|
|
template <typename T> std::string utos_funit(T n) {
|
2015-01-31 10:13:07 +01:00
|
|
|
char u = 0;
|
|
|
|
int b = 0;
|
|
|
|
if (n >= (1 << 30)) {
|
|
|
|
u = 'G';
|
|
|
|
b = 30;
|
|
|
|
} else if (n >= (1 << 20)) {
|
|
|
|
u = 'M';
|
|
|
|
b = 20;
|
|
|
|
} else if (n >= (1 << 10)) {
|
|
|
|
u = 'K';
|
|
|
|
b = 10;
|
|
|
|
}
|
|
|
|
if (b == 0) {
|
|
|
|
return utos(n);
|
|
|
|
}
|
|
|
|
return dtos(static_cast<double>(n) / (1 << b)) + u;
|
|
|
|
}
|
|
|
|
|
2014-05-14 15:39:28 +02:00
|
|
|
extern const char UPPER_XDIGITS[];
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename T> std::string utox(T n) {
|
2014-05-14 15:39:28 +02:00
|
|
|
std::string res;
|
2014-11-27 15:39:04 +01:00
|
|
|
if (n == 0) {
|
2014-05-14 15:39:28 +02:00
|
|
|
res = "0";
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
int i = 0;
|
|
|
|
T t = n;
|
2014-11-27 15:39:04 +01:00
|
|
|
for (; t; t /= 16, ++i)
|
|
|
|
;
|
2014-05-14 15:39:28 +02:00
|
|
|
res.resize(i);
|
|
|
|
--i;
|
2014-11-27 15:39:04 +01:00
|
|
|
for (; n; --i, n /= 16) {
|
2014-05-14 15:39:28 +02:00
|
|
|
res[i] = UPPER_XDIGITS[(n & 0x0f)];
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
void to_token68(std::string &base64str);
|
|
|
|
void to_base64(std::string &token68str);
|
2013-08-03 11:51:01 +02:00
|
|
|
|
2014-01-08 17:27:56 +01:00
|
|
|
void show_candidates(const char *unkopt, option *options);
|
|
|
|
|
2014-02-27 13:47:04 +01:00
|
|
|
bool has_uri_field(const http_parser_url &u, http_parser_url_fields field);
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
bool fieldeq(const char *uri1, const http_parser_url &u1, const char *uri2,
|
|
|
|
const http_parser_url &u2, http_parser_url_fields field);
|
2014-02-27 13:47:04 +01:00
|
|
|
|
|
|
|
bool fieldeq(const char *uri, const http_parser_url &u,
|
2014-11-27 15:39:04 +01:00
|
|
|
http_parser_url_fields field, const char *t);
|
2014-02-27 13:47:04 +01:00
|
|
|
|
2016-03-08 13:38:15 +01:00
|
|
|
StringRef get_uri_field(const char *uri, const http_parser_url &u,
|
|
|
|
http_parser_url_fields field);
|
2014-02-27 13:47:04 +01:00
|
|
|
|
|
|
|
uint16_t get_default_port(const char *uri, const http_parser_url &u);
|
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
bool porteq(const char *uri1, const http_parser_url &u1, const char *uri2,
|
|
|
|
const http_parser_url &u2);
|
2014-02-27 13:47:04 +01:00
|
|
|
|
2014-11-27 15:39:04 +01:00
|
|
|
void write_uri_field(std::ostream &o, const char *uri, const http_parser_url &u,
|
2014-02-27 13:47:04 +01:00
|
|
|
http_parser_url_fields field);
|
|
|
|
|
2014-03-15 07:32:38 +01:00
|
|
|
bool numeric_host(const char *hostname);
|
|
|
|
|
2015-09-06 11:39:32 +02:00
|
|
|
bool numeric_host(const char *hostname, int family);
|
|
|
|
|
2015-02-26 15:59:53 +01:00
|
|
|
// Returns numeric address string of |addr|. If getnameinfo() is
|
|
|
|
// failed, "unknown" is returned.
|
2015-02-26 14:59:25 +01:00
|
|
|
std::string numeric_name(const struct sockaddr *sa, socklen_t salen);
|
2015-02-24 15:50:18 +01:00
|
|
|
|
2016-02-21 07:28:11 +01:00
|
|
|
// Returns string representation of numeric address and port of
|
|
|
|
// |addr|. If address family is AF_UNIX, this return path to UNIX
|
|
|
|
// domain socket. Otherwise, the format is like <HOST>:<PORT>. For
|
|
|
|
// IPv6 address, address is enclosed by square brackets ([]).
|
|
|
|
std::string to_numeric_addr(const Address *addr);
|
2016-02-11 09:07:48 +01:00
|
|
|
|
2015-08-13 11:01:37 +02:00
|
|
|
// Makes internal copy of stderr (and possibly stdout in the future),
|
|
|
|
// which is then used as pointer to /dev/stderr or /proc/self/fd/2
|
|
|
|
void store_original_fds();
|
|
|
|
|
|
|
|
// Restores the original stderr that was stored with copy_original_fds
|
|
|
|
// Used just before execv
|
|
|
|
void restore_original_fds();
|
|
|
|
|
2015-08-12 20:05:10 +02:00
|
|
|
// Closes |fd| which was returned by open_log_file (see below)
|
|
|
|
// and sets it to -1. In the case that |fd| points to stdout or
|
|
|
|
// stderr, or is -1, the descriptor is not closed (but still set to -1).
|
|
|
|
void close_log_file(int &fd);
|
|
|
|
|
2014-07-05 11:22:40 +02:00
|
|
|
// Opens |path| with O_APPEND enabled. If file does not exist, it is
|
|
|
|
// created first. This function returns file descriptor referring the
|
|
|
|
// opened file if it succeeds, or -1.
|
2015-08-12 20:05:10 +02:00
|
|
|
int open_log_file(const char *path);
|
2014-07-05 11:22:40 +02:00
|
|
|
|
2014-07-12 16:30:13 +02:00
|
|
|
// Returns ASCII dump of |data| of length |len|. Only ASCII printable
|
|
|
|
// characters are preserved. Other characters are replaced with ".".
|
|
|
|
std::string ascii_dump(const uint8_t *data, size_t len);
|
|
|
|
|
2014-08-12 15:22:02 +02:00
|
|
|
// Returns absolute path of executable path. If argc == 0 or |cwd| is
|
|
|
|
// nullptr, this function returns nullptr. If argv[0] starts with
|
|
|
|
// '/', this function returns argv[0]. Oterwise return cwd + "/" +
|
|
|
|
// argv[0]. If non-null is returned, it is NULL-terminated string and
|
|
|
|
// dynamically allocated by malloc. The caller is responsible to free
|
|
|
|
// it.
|
2014-11-27 15:39:04 +01:00
|
|
|
char *get_exec_path(int argc, char **const argv, const char *cwd);
|
2014-08-12 15:22:02 +02:00
|
|
|
|
2014-09-24 16:05:13 +02:00
|
|
|
// Validates path so that it does not contain directory traversal
|
|
|
|
// vector. Returns true if path is safe. The |path| must start with
|
|
|
|
// "/" otherwise returns false. This function should be called after
|
|
|
|
// percent-decode was performed.
|
2014-11-27 15:39:04 +01:00
|
|
|
bool check_path(const std::string &path);
|
2014-09-24 16:05:13 +02:00
|
|
|
|
2014-11-05 16:56:07 +01:00
|
|
|
// Returns the |tv| value as 64 bit integer using a microsecond as an
|
|
|
|
// unit.
|
2014-11-27 15:39:04 +01:00
|
|
|
int64_t to_time64(const timeval &tv);
|
2014-11-05 16:56:07 +01:00
|
|
|
|
2016-03-24 16:02:07 +01:00
|
|
|
// Returns true if ALPN ID |proto| is supported HTTP/2 protocol
|
|
|
|
// identifier.
|
|
|
|
bool check_h2_is_selected(const StringRef &proto);
|
2014-11-14 15:14:39 +01:00
|
|
|
|
2014-12-15 14:51:34 +01:00
|
|
|
// Selects h2 protocol ALPN ID if one of supported h2 versions are
|
|
|
|
// present in |in| of length inlen. Returns true if h2 version is
|
|
|
|
// selected.
|
|
|
|
bool select_h2(const unsigned char **out, unsigned char *outlen,
|
|
|
|
const unsigned char *in, unsigned int inlen);
|
|
|
|
|
2015-09-14 15:33:48 +02:00
|
|
|
// Selects protocol ALPN ID if one of identifiers contained in |protolist| is
|
|
|
|
// present in |in| of length inlen. Returns true if identifier is
|
|
|
|
// selected.
|
|
|
|
bool select_protocol(const unsigned char **out, unsigned char *outlen,
|
2015-10-15 14:42:11 +02:00
|
|
|
const unsigned char *in, unsigned int inlen,
|
|
|
|
std::vector<std::string> proto_list);
|
2015-09-14 15:33:48 +02:00
|
|
|
|
2014-11-14 15:14:39 +01:00
|
|
|
// Returns default ALPN protocol list, which only contains supported
|
|
|
|
// HTTP/2 protocol identifier.
|
|
|
|
std::vector<unsigned char> get_default_alpn();
|
|
|
|
|
2015-09-14 15:33:48 +02:00
|
|
|
// Parses delimited strings in |s| and returns the array of substring,
|
|
|
|
// delimited by |delim|. The any white spaces around substring are
|
|
|
|
// treated as a part of substring.
|
2016-03-24 13:05:02 +01:00
|
|
|
std::vector<std::string> parse_config_str_list(const StringRef &s,
|
|
|
|
char delim = ',');
|
2015-09-14 15:33:48 +02:00
|
|
|
|
2016-02-28 13:35:26 +01:00
|
|
|
// Parses delimited strings in |s| and returns Substrings in |s|
|
|
|
|
// delimited by |delim|. The any white spaces around substring are
|
|
|
|
// treated as a part of substring.
|
|
|
|
std::vector<StringRef> split_str(const StringRef &s, char delim);
|
|
|
|
|
2014-11-19 17:53:30 +01:00
|
|
|
// Returns given time |tp| in Common Log format (e.g.,
|
|
|
|
// 03/Jul/2014:00:19:38 +0900)
|
|
|
|
// Expected type of |tp| is std::chrono::timepoint
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename T> std::string format_common_log(const T &tp) {
|
|
|
|
auto t =
|
|
|
|
std::chrono::duration_cast<std::chrono::seconds>(tp.time_since_epoch());
|
2014-11-23 08:40:38 +01:00
|
|
|
return common_log_date(t.count());
|
2014-11-19 17:53:30 +01:00
|
|
|
}
|
2014-11-23 08:40:38 +01:00
|
|
|
|
2014-11-15 14:06:16 +01:00
|
|
|
// Returns given time |tp| in ISO 8601 format (e.g.,
|
|
|
|
// 2014-11-15T12:58:24.741Z)
|
2014-11-19 17:53:30 +01:00
|
|
|
// Expected type of |tp| is std::chrono::timepoint
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename T> std::string format_iso8601(const T &tp) {
|
|
|
|
auto t = std::chrono::duration_cast<std::chrono::milliseconds>(
|
|
|
|
tp.time_since_epoch());
|
2014-11-23 08:40:38 +01:00
|
|
|
return iso8601_date(t.count());
|
2014-11-19 17:53:30 +01:00
|
|
|
}
|
|
|
|
|
2015-09-07 16:11:23 +02:00
|
|
|
// Returns given time |tp| in HTTP date format.
|
|
|
|
template <typename T> std::string format_http_date(const T &tp) {
|
|
|
|
auto t =
|
|
|
|
std::chrono::duration_cast<std::chrono::seconds>(tp.time_since_epoch());
|
|
|
|
return http_date(t.count());
|
|
|
|
}
|
|
|
|
|
2014-11-19 17:53:30 +01:00
|
|
|
// Return the system precision of the template parameter |Clock| as
|
|
|
|
// a nanosecond value of type |Rep|
|
2014-11-27 15:39:04 +01:00
|
|
|
template <typename Clock, typename Rep> Rep clock_precision() {
|
2014-11-19 17:53:30 +01:00
|
|
|
std::chrono::duration<Rep, std::nano> duration = typename Clock::duration(1);
|
|
|
|
|
|
|
|
return duration.count();
|
|
|
|
}
|
|
|
|
|
2014-12-27 18:59:06 +01:00
|
|
|
int make_socket_closeonexec(int fd);
|
|
|
|
int make_socket_nonblocking(int fd);
|
|
|
|
int make_socket_nodelay(int fd);
|
|
|
|
|
|
|
|
int create_nonblock_socket(int family);
|
|
|
|
|
|
|
|
bool check_socket_connected(int fd);
|
|
|
|
|
2015-01-10 13:33:53 +01:00
|
|
|
// Returns true if |host| is IPv6 numeric address (e.g., ::1)
|
|
|
|
bool ipv6_numeric_addr(const char *host);
|
|
|
|
|
2015-01-13 13:54:53 +01:00
|
|
|
// Parses NULL terminated string |s| as unsigned integer and returns
|
|
|
|
// the parsed integer. Additionally, if |s| ends with 'k', 'm', 'g'
|
|
|
|
// and its upper case characters, multiply the integer by 1024, 1024 *
|
|
|
|
// 1024 and 1024 * 1024 respectively. If there is an error, returns
|
|
|
|
// -1.
|
|
|
|
int64_t parse_uint_with_unit(const char *s);
|
2016-03-24 13:05:02 +01:00
|
|
|
// The following overload does not require |s| is NULL terminated.
|
|
|
|
int64_t parse_uint_with_unit(const uint8_t *s, size_t len);
|
|
|
|
int64_t parse_uint_with_unit(const StringRef &s);
|
2015-01-13 13:54:53 +01:00
|
|
|
|
2015-01-15 15:07:25 +01:00
|
|
|
// Parses NULL terminated string |s| as unsigned integer and returns
|
|
|
|
// the parsed integer. If there is an error, returns -1.
|
|
|
|
int64_t parse_uint(const char *s);
|
2016-03-24 13:05:02 +01:00
|
|
|
// The following overload does not require |s| is NULL terminated.
|
2015-01-15 15:07:25 +01:00
|
|
|
int64_t parse_uint(const uint8_t *s, size_t len);
|
2015-01-17 11:33:30 +01:00
|
|
|
int64_t parse_uint(const std::string &s);
|
2016-03-09 13:15:32 +01:00
|
|
|
int64_t parse_uint(const StringRef &s);
|
2015-01-15 15:07:25 +01:00
|
|
|
|
2015-01-27 16:36:44 +01:00
|
|
|
// Parses NULL terminated string |s| as unsigned integer and returns
|
|
|
|
// the parsed integer casted to double. If |s| ends with "s", the
|
|
|
|
// parsed value's unit is a second. If |s| ends with "ms", the unit
|
2015-10-10 04:36:13 +02:00
|
|
|
// is millisecond. Similarly, it also supports 'm' and 'h' for
|
|
|
|
// minutes and hours respectively. If none of them are given, the
|
|
|
|
// unit is second. This function returns
|
|
|
|
// std::numeric_limits<double>::infinity() if error occurs.
|
2015-01-29 15:28:47 +01:00
|
|
|
double parse_duration_with_unit(const char *s);
|
2016-03-24 13:05:02 +01:00
|
|
|
// The following overload does not require |s| is NULL terminated.
|
|
|
|
double parse_duration_with_unit(const uint8_t *s, size_t len);
|
|
|
|
double parse_duration_with_unit(const StringRef &s);
|
2015-01-27 16:36:44 +01:00
|
|
|
|
|
|
|
// Returns string representation of time duration |t|. If t has
|
|
|
|
// fractional part (at least more than or equal to 1e-3), |t| is
|
|
|
|
// multiplied by 1000 and the unit "ms" is appended. Otherwise, |t|
|
|
|
|
// is left as is and "s" is appended.
|
|
|
|
std::string duration_str(double t);
|
|
|
|
|
2015-01-31 10:13:07 +01:00
|
|
|
// Returns string representation of time duration |t|. It appends
|
|
|
|
// unit after the formatting. The available units are s, ms and us.
|
|
|
|
// The unit which is equal to or less than |t| is used and 2
|
|
|
|
// fractional digits follow.
|
|
|
|
std::string format_duration(const std::chrono::microseconds &u);
|
|
|
|
|
2015-12-01 15:49:28 +01:00
|
|
|
// Just like above, but this takes |t| as seconds.
|
|
|
|
std::string format_duration(double t);
|
|
|
|
|
2015-02-22 04:27:51 +01:00
|
|
|
// Creates "host:port" string using given |host| and |port|. If
|
|
|
|
// |host| is numeric IPv6 address (e.g., ::1), it is enclosed by "["
|
2015-03-13 15:01:55 +01:00
|
|
|
// and "]". If |port| is 80 or 443, port part is omitted.
|
2016-02-13 15:21:32 +01:00
|
|
|
std::string make_http_hostport(const StringRef &host, uint16_t port);
|
|
|
|
|
|
|
|
// Just like make_http_hostport(), but doesn't treat 80 and 443
|
|
|
|
// specially.
|
|
|
|
std::string make_hostport(const StringRef &host, uint16_t port);
|
2015-02-22 04:27:51 +01:00
|
|
|
|
2015-03-23 18:30:51 +01:00
|
|
|
// Dumps |src| of length |len| in the format similar to `hexdump -C`.
|
|
|
|
void hexdump(FILE *out, const uint8_t *src, size_t len);
|
|
|
|
|
2015-07-25 15:22:17 +02:00
|
|
|
// Copies 2 byte unsigned integer |n| in host byte order to |buf| in
|
|
|
|
// network byte order.
|
|
|
|
void put_uint16be(uint8_t *buf, uint16_t n);
|
|
|
|
|
|
|
|
// Copies 4 byte unsigned integer |n| in host byte order to |buf| in
|
|
|
|
// network byte order.
|
|
|
|
void put_uint32be(uint8_t *buf, uint32_t n);
|
|
|
|
|
|
|
|
// Retrieves 2 byte unsigned integer stored in |data| in network byte
|
|
|
|
// order and returns it in host byte order.
|
|
|
|
uint16_t get_uint16(const uint8_t *data);
|
|
|
|
|
|
|
|
// Retrieves 4 byte unsigned integer stored in |data| in network byte
|
|
|
|
// order and returns it in host byte order.
|
|
|
|
uint32_t get_uint32(const uint8_t *data);
|
|
|
|
|
|
|
|
// Retrieves 8 byte unsigned integer stored in |data| in network byte
|
|
|
|
// order and returns it in host byte order.
|
|
|
|
uint64_t get_uint64(const uint8_t *data);
|
|
|
|
|
2015-10-29 15:08:52 +01:00
|
|
|
// Reads mime types file (see /etc/mime.types), and stores extension
|
|
|
|
// -> MIME type map in |res|. This function returns 0 if it succeeds,
|
|
|
|
// or -1.
|
|
|
|
int read_mime_types(std::map<std::string, std::string> &res,
|
|
|
|
const char *filename);
|
2015-10-28 16:21:36 +01:00
|
|
|
|
2016-01-15 15:04:58 +01:00
|
|
|
template <typename Generator>
|
|
|
|
std::string random_alpha_digit(Generator &gen, size_t len) {
|
|
|
|
std::string res;
|
|
|
|
res.reserve(len);
|
|
|
|
std::uniform_int_distribution<> dis(0, 26 * 2 + 10 - 1);
|
|
|
|
for (; len > 0; --len) {
|
|
|
|
res += "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"[dis(
|
|
|
|
gen)];
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2016-03-10 14:42:07 +01:00
|
|
|
template <typename OutputIterator, typename CharT, size_t N>
|
|
|
|
OutputIterator copy_lit(OutputIterator it, CharT(&s)[N]) {
|
|
|
|
return std::copy_n(s, N - 1, it);
|
|
|
|
}
|
|
|
|
|
2016-06-27 15:42:28 +02:00
|
|
|
// Returns x**y
|
|
|
|
double int_pow(double x, size_t y);
|
|
|
|
|
2016-07-06 15:31:28 +02:00
|
|
|
uint32_t hash32(const StringRef &s);
|
|
|
|
|
|
|
|
// Computes SHA-256 of |s|, and stores it in |buf|. This function
|
|
|
|
// returns 0 if it succeeds, or -1.
|
|
|
|
int sha256(uint8_t *buf, const StringRef &s);
|
|
|
|
|
2012-01-29 16:34:10 +01:00
|
|
|
} // namespace util
|
|
|
|
|
2013-07-12 17:19:03 +02:00
|
|
|
} // namespace nghttp2
|
2012-01-29 16:34:10 +01:00
|
|
|
|
|
|
|
#endif // UTIL_H
|