Сервер NGHTTP2 не возвращает код состояния HTTP к клиентуC++

Программы на C++. Форум разработчиков
Ответить Пред. темаСлед. тема
Anonymous
 Сервер NGHTTP2 не возвращает код состояния HTTP к клиенту

Сообщение Anonymous »

Я построил сервер HTTP/2 NGHTTP2 и адаптировал официальный пример для запуска его в Windows: https://nghttp2.org/documentation/tutorial-server.html https://127.0.0.1:9090 затем консоль сервера отображает факт соединения. Следующим из кода сервер должен ответить кодом состояния 200. Однако Curl не отображает код состояния ответа 200, несмотря на использование флага -v в Curl. Оказывается, что данные, отправленные с сервера, не достигают клиента. < /P>
Что может быть проблемой и как его решить?#include

#pragma warning (disable: 4996)
#pragma comment(lib, "path\\to\\nghttp2.lib")
#pragma comment(lib, "ws2_32.lib")
#pragma comment(lib, "Crypt32.lib")

#include
#include
#include

#include
#include
#include
#include

#include
#include

#define NGHTTP2_NO_SSIZE_T
#include "nghttp2/nghttp2.h"

#define OUTPUT_WOULDBLOCK_THRESHOLD (1 = 0x30000000L
if (SSL_CTX_set1_curves_list(ssl_ctx, "P-256") != 1) {
errx(1, "SSL_CTX_set1_curves_list failed: %s",
ERR_error_string(ERR_get_error(), NULL));
}
#else /* !(OPENSSL_VERSION_NUMBER >= 0x30000000L) */
{
EC_KEY* ecdh;
ecdh = EC_KEY_new_by_curve_name(NID_X9_62_prime256v1);
if (!ecdh) {
//errx(1, "EC_KEY_new_by_curv_name failed: %s",
// ERR_error_string(ERR_get_error(), NULL));
}
SSL_CTX_set_tmp_ecdh(ssl_ctx, ecdh);
EC_KEY_free(ecdh);
}
#endif /* !(OPENSSL_VERSION_NUMBER >= 0x30000000L) */

if (SSL_CTX_use_PrivateKey_file(ssl_ctx, key_file, SSL_FILETYPE_PEM) != 1) {
std::cout root.next;
session_data->root.next = stream_data;
stream_data->prev = &session_data->root;
if (stream_data->next) {
stream_data->next->prev = stream_data;
}
}

static void remove_stream(http2_session_data* session_data,
http2_stream_data* stream_data) {
(void)session_data;

stream_data->prev->next = stream_data->next;
if (stream_data->next) {
stream_data->next->prev = stream_data->prev;
}
}

static http2_stream_data*
create_http2_stream_data(http2_session_data* session_data, int32_t stream_id) {
http2_stream_data* stream_data;
stream_data = (http2_stream_data*)malloc(sizeof(http2_stream_data));
memset(stream_data, 0, sizeof(http2_stream_data));
stream_data->stream_id = stream_id;
stream_data->fd = -1;

add_stream(session_data, stream_data);
return stream_data;
}

static void delete_http2_stream_data(http2_stream_data* stream_data) {
if (stream_data->fd != -1) {
close(stream_data->fd);
}
free(stream_data->request_path);
free(stream_data);
}

static http2_session_data* create_http2_session_data(app_context* app_ctx,
int fd,
struct sockaddr* addr,
int addrlen) {
int rv;
http2_session_data* session_data;
SSL* ssl;
char host[NI_MAXHOST];
int val = 1;

ssl = create_ssl(app_ctx->ssl_ctx);
session_data = (http2_session_data*)malloc(sizeof(http2_session_data));
memset(session_data, 0, sizeof(http2_session_data));
session_data->app_ctx = app_ctx;
setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, (char*)&val, sizeof(val));
session_data->bev = bufferevent_openssl_socket_new(
app_ctx->evbase, fd, ssl, BUFFEREVENT_SSL_ACCEPTING,
BEV_OPT_CLOSE_ON_FREE | BEV_OPT_DEFER_CALLBACKS);
bufferevent_enable(session_data->bev, EV_READ | EV_WRITE);
rv = getnameinfo(addr, (socklen_t)addrlen, host, sizeof(host), NULL, 0,
NI_NUMERICHOST);
if (rv != 0) {
session_data->client_addr = strdup("(unknown)");
}
else {
session_data->client_addr = strdup(host);
}

return session_data;
}

static void delete_http2_session_data(http2_session_data* session_data) {
http2_stream_data* stream_data;
SSL* ssl = bufferevent_openssl_get_ssl(session_data->bev);
fprintf(stderr, "%s disconnected\n", session_data->client_addr);
if (ssl) {
SSL_shutdown(ssl);
}
bufferevent_free(session_data->bev);
nghttp2_session_del(session_data->session);
for (stream_data = session_data->root.next; stream_data;) {
http2_stream_data* next = stream_data->next;
delete_http2_stream_data(stream_data);
stream_data = next;
}
free(session_data->client_addr);
free(session_data);
}

/* Serialize the frame and send (or buffer) the data to
bufferevent. */
static int session_send(http2_session_data* session_data) {
int rv;
rv = nghttp2_session_send(session_data->session);
if (rv != 0) {
//warnx("Fatal error: %s", nghttp2_strerror(rv));
return -1;
}
return 0;
}

/* Read the data in the bufferevent and feed them into nghttp2 library
function. Invocation of nghttp2_session_mem_recv2() may make
additional pending frames, so call session_send() at the end of the
function. */
static int session_recv(http2_session_data* session_data) {
nghttp2_ssize readlen;
struct evbuffer* input = bufferevent_get_input(session_data->bev);
size_t datalen = evbuffer_get_length(input);
unsigned char* data = evbuffer_pullup(input, -1);

readlen = nghttp2_session_mem_recv2(session_data->session, data, datalen);
if (readlen < 0) {
//warnx("Fatal error: %s", nghttp2_strerror((int)readlen));
return -1;
}
if (evbuffer_drain(input, (size_t)readlen) != 0) {
//warnx("Fatal error: evbuffer_drain failed");
return -1;
}
if (session_send(session_data) != 0) {
return -1;
}
return 0;
}

static nghttp2_ssize send_callback(nghttp2_session* session,
const uint8_t* data, size_t length,
int flags, void* user_data) {
http2_session_data* session_data = (http2_session_data*)user_data;
struct bufferevent* bev = session_data->bev;
(void)session;
(void)flags;

/* Avoid excessive buffering in server side. */
if (evbuffer_get_length(bufferevent_get_output(session_data->bev)) >=
OUTPUT_WOULDBLOCK_THRESHOLD) {
return NGHTTP2_ERR_WOULDBLOCK;
}

bufferevent_write(bev, data, length);
return (nghttp2_ssize)length;
}

/* Returns nonzero if the string |s| ends with the substring |sub| */
static int ends_with(const char* s, const char* sub) {
size_t slen = strlen(s);
size_t sublen = strlen(sub);
if (slen < sublen) {
return 0;
}
return memcmp(s + slen - sublen, sub, sublen) == 0;
}

/* Returns int value of hex string character |c| */
static uint8_t hex_to_uint(uint8_t c) {
if ('0' stream_id, NGHTTP2_INTERNAL_ERROR);
if (rv != 0) {
//warnx("Fatal error: %s", nghttp2_strerror(rv));
return -1;
}
return 0;
// }

writelen = write(pipefd[1], ERROR_HTML, sizeof(ERROR_HTML) - 1);
close(pipefd[1]);

if (writelen != sizeof(ERROR_HTML) - 1) {
// close(pipefd[0]);
return -1;
}

stream_data->fd = pipefd[0];

if (send_response(session, stream_data->stream_id, hdrs, ARRLEN(hdrs),
pipefd[0]) != 0) {
//close(pipefd[0]);
return -1;
}
return 0;
}

/* nghttp2_on_header_callback: Called when nghttp2 library emits
single header name/value pair. */
static int on_header_callback(nghttp2_session* session,
const nghttp2_frame* frame, const uint8_t* name,
size_t namelen, const uint8_t* value,
size_t valuelen, uint8_t flags, void* user_data) {
http2_stream_data* stream_data;
const char PATH[] = ":path";
(void)flags;
(void)user_data;

switch (frame->hd.type) {
case NGHTTP2_HEADERS:
if (frame->headers.cat != NGHTTP2_HCAT_REQUEST) {
break;
}
stream_data = (http2_stream_data*)
nghttp2_session_get_stream_user_data(session, frame->hd.stream_id);
if (!stream_data || stream_data->request_path) {
break;
}
if (namelen == sizeof(PATH) - 1 && memcmp(PATH, name, namelen) == 0) {
size_t j;
for (j = 0; j < valuelen && value[j] != '?'; ++j)
;
stream_data->request_path = percent_decode(value, j);
}
break;
}
return 0;
}

static int on_begin_headers_callback(nghttp2_session* session,
const nghttp2_frame* frame,
void* user_data) {
http2_session_data* session_data = (http2_session_data*)user_data;
http2_stream_data* stream_data;

if (frame->hd.type != NGHTTP2_HEADERS ||
frame->headers.cat != NGHTTP2_HCAT_REQUEST) {
return 0;
}
stream_data = create_http2_stream_data(session_data, frame->hd.stream_id);
nghttp2_session_set_stream_user_data(session, frame->hd.stream_id,
stream_data);
return 0;
}

/* Minimum check for directory traversal. Returns nonzero if it is
safe. */
static int check_path(const char* path) {
/* We don't like '\' in url. */
return path[0] && path[0] == '/' && strchr(path, '\\') == NULL &&
strstr(path, "/../") == NULL && strstr(path, "/./") == NULL &&
!ends_with(path, "/..") && !ends_with(path, "/.");
}

static int on_request_recv(nghttp2_session* session,
http2_session_data* session_data,
http2_stream_data* stream_data) {
int fd;
nghttp2_nv hdrs[] = { MAKE_NV(":status", "200") };
char* rel_path;

if (!stream_data->request_path) {
if (error_reply(session, stream_data) != 0) {
return NGHTTP2_ERR_CALLBACK_FAILURE;
}
return 0;
}
fprintf(stderr, "%s GET %s\n", session_data->client_addr,
stream_data->request_path);
if (!check_path(stream_data->request_path)) {
if (error_reply(session, stream_data) != 0) {
return NGHTTP2_ERR_CALLBACK_FAILURE;
}
return 0;
}
for (rel_path = stream_data->request_path; *rel_path == '/'; ++rel_path)
;
fd = _open(rel_path, _O_RDONLY);
if (fd == -1) {
if (error_reply(session, stream_data) != 0) {
return NGHTTP2_ERR_CALLBACK_FAILURE;
}
return 0;
}
stream_data->fd = fd;

if (send_response(session, stream_data->stream_id, hdrs, ARRLEN(hdrs), fd) !=
0) {
close(fd);
return NGHTTP2_ERR_CALLBACK_FAILURE;
}
return 0;
}

static int on_frame_recv_callback(nghttp2_session* session,
const nghttp2_frame* frame, void* user_data) {
http2_session_data* session_data = (http2_session_data*)user_data;
http2_stream_data* stream_data;
switch (frame->hd.type) {
case NGHTTP2_DATA:
case NGHTTP2_HEADERS:
/* Check that the client request has finished */
if (frame->hd.flags & NGHTTP2_FLAG_END_STREAM) {
stream_data = (http2_stream_data*)
nghttp2_session_get_stream_user_data(session, frame->hd.stream_id);
/* For DATA and HEADERS frame, this callback may be called after
on_stream_close_callback. Check that stream still alive. */
if (!stream_data) {
return 0;
}
return on_request_recv(session, session_data, stream_data);
}
break;
default:
break;
}
return 0;
}

static int on_stream_close_callback(nghttp2_session* session, int32_t stream_id,
uint32_t error_code, void* user_data) {
http2_session_data* session_data = (http2_session_data*)user_data;
http2_stream_data* stream_data;
(void)error_code;

stream_data = (http2_stream_data*)nghttp2_session_get_stream_user_data(session, stream_id);
if (!stream_data) {
return 0;
}
remove_stream(session_data, stream_data);
delete_http2_stream_data(stream_data);
return 0;
}

static void initialize_nghttp2_session(http2_session_data* session_data) {
nghttp2_session_callbacks* callbacks;

nghttp2_session_callbacks_new(&callbacks);

nghttp2_session_callbacks_set_send_callback2(callbacks, send_callback);

nghttp2_session_callbacks_set_on_frame_recv_callback(callbacks,
on_frame_recv_callback);

nghttp2_session_callbacks_set_on_stream_close_callback(
callbacks, on_stream_close_callback);

nghttp2_session_callbacks_set_on_header_callback(callbacks,
on_header_callback);

nghttp2_session_callbacks_set_on_begin_headers_callback(
callbacks, on_begin_headers_callback);

nghttp2_session_server_new(&session_data->session, callbacks, session_data);

nghttp2_session_callbacks_del(callbacks);
}

/* Send HTTP/2 client connection header, which includes 24 bytes
magic octets and SETTINGS frame */
static int send_server_connection_header(http2_session_data* session_data) {
nghttp2_settings_entry iv[1] = {
{NGHTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, 100} };
int rv;

rv = nghttp2_submit_settings(session_data->session, NGHTTP2_FLAG_NONE, iv,
ARRLEN(iv));
if (rv != 0) {
//warnx("Fatal error: %s", nghttp2_strerror(rv));
return -1;
}
return 0;
}

/* readcb for bufferevent after client connection header was
checked. */
static void readcb(struct bufferevent* bev, void* ptr) {
http2_session_data* session_data = (http2_session_data*)ptr;
(void)bev;

if (session_recv(session_data) != 0) {
delete_http2_session_data(session_data);
return;
}
}

/* writecb for bufferevent. To greaceful shutdown after sending or
receiving GOAWAY, we check the some conditions on the nghttp2
library and output buffer of bufferevent. If it indicates we have
no business to this session, tear down the connection. If the
connection is not going to shutdown, we call session_send() to
process pending data in the output buffer. This is necessary
because we have a threshold on the buffer size to avoid too much
buffering. See send_callback(). */
static void writecb(struct bufferevent* bev, void* ptr)
{
http2_session_data* session_data = (http2_session_data*)ptr;
if (evbuffer_get_length(bufferevent_get_output(bev)) > 0) {
return;
}
if (nghttp2_session_want_read(session_data->session) == 0 &&
nghttp2_session_want_write(session_data->session) == 0) {
delete_http2_session_data(session_data);
return;
}

if (session_send(session_data) != 0) {
delete_http2_session_data(session_data);
return;
}
}

/* eventcb for bufferevent */
static void eventcb(struct bufferevent* bev, short events, void* ptr) {
http2_session_data* session_data = (http2_session_data*)ptr;
if (events & BEV_EVENT_CONNECTED) {
const unsigned char* alpn = NULL;
unsigned int alpnlen = 0;
SSL* ssl;
(void)bev;

fprintf(stderr, "%s connected\n", session_data->client_addr);

ssl = bufferevent_openssl_get_ssl(session_data->bev);

SSL_get0_alpn_selected(ssl, &alpn, &alpnlen);

if (alpn == NULL || alpnlen != 2 || memcmp("h2", alpn, 2) != 0) {
fprintf(stderr, "%s h2 is not negotiated\n", session_data->client_addr);
delete_http2_session_data(session_data);
return;
}

initialize_nghttp2_session(session_data);

if (send_server_connection_header(session_data) != 0 ||
session_send(session_data) != 0) {
delete_http2_session_data(session_data);
return;
}

return;
}
if (events & BEV_EVENT_EOF) {
fprintf(stderr, "%s EOF\n", session_data->client_addr);
}
else if (events & BEV_EVENT_ERROR) {
fprintf(stderr, "%s network error\n", session_data->client_addr);
}
else if (events & BEV_EVENT_TIMEOUT) {
fprintf(stderr, "%s timeout\n", session_data->client_addr);
}
delete_http2_session_data(session_data);
}

/* callback for evconnlistener */
static void acceptcb(struct evconnlistener* listener, evutil_socket_t fd,
struct sockaddr* addr, int addrlen, void* arg) {
std::cout ai_addrlen);
if (listener) {
freeaddrinfo(res);

return;
}
}
//errx(1, "Could not start listener");
}

static void initialize_app_context(app_context* app_ctx, SSL_CTX* ssl_ctx,
struct event_base* evbase) {
memset(app_ctx, 0, sizeof(app_context));
app_ctx->ssl_ctx = ssl_ctx;
app_ctx->evbase = evbase;
}

static void run(const char* service, const char* key_file,
const char* cert_file) {
SSL_CTX* ssl_ctx;
app_context app_ctx;
struct event_base* evbase;

ssl_ctx = create_ssl_ctx(key_file, cert_file);
evbase = event_base_new();
initialize_app_context(&app_ctx, ssl_ctx, evbase);
start_listen(evbase, service, &app_ctx);
std::cout

Подробнее здесь: https://stackoverflow.com/questions/796 ... -to-client
Реклама
Ответить Пред. темаСлед. тема

Быстрый ответ

Изменение регистра текста: 
Смайлики
:) :( :oops: :roll: :wink: :muza: :clever: :sorry: :angel: :read: *x)
Ещё смайлики…
   
К этому ответу прикреплено по крайней мере одно вложение.

Если вы не хотите добавлять вложения, оставьте поля пустыми.

Максимально разрешённый размер вложения: 15 МБ.

  • Похожие темы
    Ответы
    Просмотры
    Последнее сообщение
  • HTTP 2 Server с использованием библиотеки NGHTTP2 в Windows
    Anonymous » » в форуме C++
    0 Ответы
    18 Просмотры
    Последнее сообщение Anonymous
  • Как построить NGHTTP2-ASIO в Windows? [закрыто]
    Anonymous » » в форуме C++
    0 Ответы
    14 Просмотры
    Последнее сообщение Anonymous
  • Не удается скомпилировать nghttp2 static с проектом QT C ++ с MSVC
    Anonymous » » в форуме C++
    0 Ответы
    3 Просмотры
    Последнее сообщение Anonymous
  • Как получить код состояния HTTP из HTTP-запроса
    Anonymous » » в форуме C#
    0 Ответы
    6 Просмотры
    Последнее сообщение Anonymous
  • Какой стек HTTP для использования для клиента, требующего HTTP/1.1, HTTP/2 и HTTP/3
    Anonymous » » в форуме Android
    0 Ответы
    66 Просмотры
    Последнее сообщение Anonymous

Вернуться в «C++»