mirror of
https://gitea.invidious.io/iv-org/litespeed-quic.git
synced 2024-08-15 00:53:43 +00:00
4429f8ea1e
- [FEATURE] QUIC and HTTP/3 Internet Draft 31 support. Drop ID-30 and ID-31 support. - [BUGFIX] Divide-by-zero in newly enabled conn stats code when no packets were sent. - [BUGFIX] Memory leak in gQUIC client when server hello cannot be parsed. - [BUGFIX] Server Initial packet size calculation. - Log user-agent and CONN_CLOSE reason when peer reports error. - Example programs: Specify ALPN for echo and md5 clients and servers (issue #184). - Example programs: Don't add "QUIC_" prefix to lines in keylog file (issue #185). - http_server: Fix fd leak in preadv mode; fix preadv() usage when reading from disk.
525 lines
14 KiB
C
525 lines
14 KiB
C
/* Copyright (c) 2017 - 2020 LiteSpeed Technologies Inc. See LICENSE. */
|
|
/*
|
|
* md5_client.c -- This client sends one or more files to MD5 QUIC server
|
|
* for MD5 sum calculation.
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
|
|
#ifndef WIN32
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#else
|
|
#include "vc_compat.h"
|
|
#include "getopt.h"
|
|
#endif
|
|
|
|
#include <event2/event.h>
|
|
#include <openssl/md5.h>
|
|
|
|
#include "lsquic.h"
|
|
#include "test_common.h"
|
|
#include "prog.h"
|
|
|
|
#include "../src/liblsquic/lsquic_logger.h"
|
|
#include "../src/liblsquic/lsquic_int_types.h"
|
|
#include "../src/liblsquic/lsquic_varint.h"
|
|
#include "../src/liblsquic/lsquic_hq.h"
|
|
#include "../src/liblsquic/lsquic_sfcw.h"
|
|
#include "../src/liblsquic/lsquic_hash.h"
|
|
#include "../src/liblsquic/lsquic_stream.h"
|
|
|
|
/* Set to non-zero value to test out what happens when reset is sent */
|
|
#define RESET_AFTER_N_WRITES 0
|
|
|
|
static int g_write_file = 1;
|
|
|
|
#define LOCAL_BUF_SIZE 0x100
|
|
|
|
static struct {
|
|
unsigned stream_id; /* If set, reset this stream ID */
|
|
off_t offset; /* Reset it after writing this many bytes */
|
|
} g_reset_stream;
|
|
|
|
struct file {
|
|
LIST_ENTRY(file) next_file;
|
|
const char *filename;
|
|
struct lsquic_reader reader;
|
|
int fd;
|
|
unsigned priority;
|
|
enum {
|
|
FILE_RESET = (1 << 0),
|
|
} file_flags;
|
|
size_t md5_off;
|
|
char md5str[MD5_DIGEST_LENGTH * 2];
|
|
};
|
|
|
|
struct lsquic_conn_ctx;
|
|
|
|
struct client_ctx {
|
|
struct lsquic_conn_ctx *conn_h;
|
|
LIST_HEAD(, file) files;
|
|
unsigned n_files;
|
|
struct file *cur_file;
|
|
lsquic_engine_t *engine;
|
|
struct service_port *sport;
|
|
struct prog *prog;
|
|
};
|
|
|
|
struct lsquic_conn_ctx {
|
|
lsquic_conn_t *conn;
|
|
struct client_ctx *client_ctx;
|
|
};
|
|
|
|
|
|
static lsquic_conn_ctx_t *
|
|
client_on_new_conn (void *stream_if_ctx, lsquic_conn_t *conn)
|
|
{
|
|
struct client_ctx *client_ctx = stream_if_ctx;
|
|
lsquic_conn_ctx_t *conn_h = malloc(sizeof(*conn_h));
|
|
conn_h->conn = conn;
|
|
conn_h->client_ctx = client_ctx;
|
|
client_ctx->conn_h = conn_h;
|
|
assert(client_ctx->n_files > 0);
|
|
unsigned n = client_ctx->n_files;
|
|
while (n--)
|
|
lsquic_conn_make_stream(conn);
|
|
print_conn_info(conn);
|
|
return conn_h;
|
|
}
|
|
|
|
|
|
static void
|
|
client_on_goaway_received (lsquic_conn_t *conn)
|
|
{
|
|
LSQ_NOTICE("GOAWAY received");
|
|
}
|
|
|
|
|
|
static void
|
|
client_on_conn_closed (lsquic_conn_t *conn)
|
|
{
|
|
lsquic_conn_ctx_t *conn_h = lsquic_conn_get_ctx(conn);
|
|
LSQ_NOTICE("Connection closed");
|
|
prog_stop(conn_h->client_ctx->prog);
|
|
free(conn_h);
|
|
}
|
|
|
|
|
|
struct lsquic_stream_ctx {
|
|
lsquic_stream_t *stream;
|
|
struct client_ctx *client_ctx;
|
|
struct file *file;
|
|
struct event *read_stdin_ev;
|
|
struct {
|
|
int initialized;
|
|
size_t size,
|
|
off;
|
|
} small;
|
|
};
|
|
|
|
|
|
static lsquic_stream_ctx_t *
|
|
client_on_new_stream (void *stream_if_ctx, lsquic_stream_t *stream)
|
|
{
|
|
struct client_ctx *const client_ctx = stream_if_ctx;
|
|
if (!stream)
|
|
{
|
|
assert(client_ctx->n_files > 0);
|
|
LSQ_NOTICE("%s: got null stream: no more streams possible; # files: %u",
|
|
__func__, client_ctx->n_files);
|
|
--client_ctx->n_files;
|
|
if (0 == client_ctx->n_files)
|
|
{
|
|
LSQ_DEBUG("closing connection");
|
|
lsquic_conn_close(client_ctx->conn_h->conn);
|
|
}
|
|
return NULL;
|
|
}
|
|
lsquic_stream_ctx_t *st_h = calloc(1, sizeof(*st_h));
|
|
st_h->stream = stream;
|
|
st_h->client_ctx = stream_if_ctx;
|
|
if (LIST_EMPTY(&st_h->client_ctx->files))
|
|
{
|
|
/* XXX: perhaps we should not be able to write immediately: there may
|
|
* be internal memory constraints...
|
|
*/
|
|
lsquic_stream_write(stream, "client request", 14);
|
|
(void) lsquic_stream_flush(stream);
|
|
lsquic_stream_wantwrite(stream, 0);
|
|
lsquic_stream_wantread(stream, 1);
|
|
}
|
|
else
|
|
{
|
|
st_h->file = LIST_FIRST(&st_h->client_ctx->files);
|
|
if (g_write_file)
|
|
{
|
|
st_h->file->fd = -1;
|
|
st_h->file->reader.lsqr_read = test_reader_read;
|
|
st_h->file->reader.lsqr_size = test_reader_size;
|
|
st_h->file->reader.lsqr_ctx = create_lsquic_reader_ctx(st_h->file->filename);
|
|
if (!st_h->file->reader.lsqr_ctx)
|
|
exit(1);
|
|
}
|
|
else
|
|
{
|
|
st_h->file->fd = open(st_h->file->filename, O_RDONLY);
|
|
if (st_h->file->fd < 0)
|
|
{
|
|
LSQ_ERROR("could not open %s for reading: %s",
|
|
st_h->file->filename, strerror(errno));
|
|
exit(1);
|
|
}
|
|
}
|
|
LIST_REMOVE(st_h->file, next_file);
|
|
lsquic_stream_set_priority(stream, st_h->file->priority);
|
|
lsquic_stream_wantwrite(stream, 1);
|
|
}
|
|
return st_h;
|
|
}
|
|
|
|
|
|
static size_t
|
|
buf_reader_size (void *reader_ctx)
|
|
{
|
|
lsquic_stream_ctx_t *const st_h = reader_ctx;
|
|
struct stat st;
|
|
off_t off;
|
|
|
|
if (st_h->small.initialized)
|
|
goto initialized;
|
|
|
|
if (0 != fstat(st_h->file->fd, &st))
|
|
{
|
|
LSQ_ERROR("fstat failed: %s", strerror(errno));
|
|
goto err;
|
|
}
|
|
|
|
off = lseek(st_h->file->fd, 0, SEEK_CUR);
|
|
if (off == (off_t) -1)
|
|
{
|
|
LSQ_ERROR("lseek failed: %s", strerror(errno));
|
|
goto err;
|
|
}
|
|
|
|
if (st.st_size < off)
|
|
{
|
|
LSQ_ERROR("size mismatch");
|
|
goto err;
|
|
}
|
|
|
|
st_h->small.initialized = 1;
|
|
st_h->small.off = off;
|
|
st_h->small.size = st.st_size;
|
|
|
|
initialized:
|
|
if (st_h->small.size - st_h->small.off > LOCAL_BUF_SIZE)
|
|
return LOCAL_BUF_SIZE;
|
|
else
|
|
return st_h->small.size - st_h->small.off;
|
|
|
|
err:
|
|
close(st_h->file->fd);
|
|
st_h->file->fd = 0;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static size_t
|
|
buf_reader_read (void *reader_ctx, void *buf, size_t count)
|
|
{
|
|
lsquic_stream_ctx_t *const st_h = reader_ctx;
|
|
ssize_t nr;
|
|
unsigned char local_buf[LOCAL_BUF_SIZE];
|
|
|
|
assert(st_h->small.initialized);
|
|
|
|
if (count > sizeof(local_buf))
|
|
count = sizeof(local_buf);
|
|
|
|
nr = read(st_h->file->fd, local_buf, count);
|
|
if (nr < 0)
|
|
{
|
|
LSQ_ERROR("read: %s", strerror(errno));
|
|
close(st_h->file->fd);
|
|
st_h->file->fd = 0;
|
|
return 0;
|
|
}
|
|
|
|
memcpy(buf, local_buf, nr);
|
|
st_h->small.off += nr;
|
|
return nr;
|
|
}
|
|
|
|
|
|
static void
|
|
client_file_on_write_buf (lsquic_stream_ctx_t *st_h)
|
|
{
|
|
ssize_t nw;
|
|
struct lsquic_reader reader = {
|
|
.lsqr_read = buf_reader_read,
|
|
.lsqr_size = buf_reader_size,
|
|
.lsqr_ctx = st_h,
|
|
};
|
|
|
|
if (g_reset_stream.stream_id == lsquic_stream_id(st_h->stream) &&
|
|
lseek(st_h->file->fd, 0, SEEK_CUR) >= g_reset_stream.offset)
|
|
{
|
|
lsquic_stream_reset(st_h->stream, 0x01 /* QUIC_INTERNAL_ERROR */);
|
|
g_reset_stream.stream_id = 0; /* Reset only once */
|
|
}
|
|
|
|
nw = lsquic_stream_writef(st_h->stream, &reader);
|
|
if (-1 == nw)
|
|
{
|
|
if (ECONNRESET == errno)
|
|
st_h->file->file_flags |= FILE_RESET;
|
|
LSQ_WARN("lsquic_stream_read: %s", strerror(errno));
|
|
lsquic_stream_close(st_h->stream);
|
|
return;
|
|
}
|
|
|
|
#if RESET_AFTER_N_WRITES
|
|
static int write_count = 0;
|
|
if (write_count++ > RESET_AFTER_N_WRITES)
|
|
lsquic_stream_reset(st_h->stream, 0);
|
|
#endif
|
|
|
|
if (0 == nw)
|
|
{
|
|
(void) close(st_h->file->fd);
|
|
if (0 == lsquic_stream_shutdown(st_h->stream, 1))
|
|
lsquic_stream_wantread(st_h->stream, 1);
|
|
else
|
|
{
|
|
if (ECONNRESET == errno)
|
|
st_h->file->file_flags |= FILE_RESET;
|
|
LSQ_WARN("lsquic_stream_shutdown: %s", strerror(errno));
|
|
lsquic_stream_close(st_h->stream);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
client_file_on_write_efficient (lsquic_stream_t *stream,
|
|
lsquic_stream_ctx_t *st_h)
|
|
{
|
|
ssize_t nw;
|
|
|
|
nw = lsquic_stream_writef(stream, &st_h->file->reader);
|
|
if (nw < 0)
|
|
{
|
|
LSQ_ERROR("write error: %s", strerror(errno));
|
|
exit(1);
|
|
}
|
|
if (nw == 0)
|
|
{
|
|
destroy_lsquic_reader_ctx(st_h->file->reader.lsqr_ctx);
|
|
st_h->file->reader.lsqr_ctx = NULL;
|
|
if (0 == lsquic_stream_shutdown(st_h->stream, 1))
|
|
lsquic_stream_wantread(st_h->stream, 1);
|
|
else
|
|
{
|
|
if (ECONNRESET == errno)
|
|
st_h->file->file_flags |= FILE_RESET;
|
|
LSQ_WARN("lsquic_stream_shutdown: %s", strerror(errno));
|
|
lsquic_stream_close(st_h->stream);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
client_file_on_write (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h)
|
|
{
|
|
if (g_write_file)
|
|
client_file_on_write_efficient(stream, st_h);
|
|
else
|
|
client_file_on_write_buf(st_h);
|
|
}
|
|
|
|
|
|
static void
|
|
client_file_on_read (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h)
|
|
{
|
|
char buf;
|
|
/* We expect to read in 32-character MD5 string */
|
|
size_t ntoread = sizeof(st_h->file->md5str) - st_h->file->md5_off;
|
|
if (0 == ntoread)
|
|
{
|
|
lsquic_stream_wantread(stream, 0);
|
|
/* XXX What about an error (due to RST_STREAM) here: how are we to
|
|
* handle it?
|
|
*/
|
|
/* Expect a FIN */
|
|
if (0 == lsquic_stream_read(stream, &buf, sizeof(buf)))
|
|
{
|
|
LSQ_NOTICE("%.*s %s", (int) sizeof(st_h->file->md5str),
|
|
st_h->file->md5str,
|
|
st_h->file->filename);
|
|
fflush(stdout);
|
|
LSQ_DEBUG("# of files: %d", st_h->client_ctx->n_files);
|
|
lsquic_stream_shutdown(stream, 0);
|
|
}
|
|
else
|
|
LSQ_ERROR("expected FIN from stream!");
|
|
}
|
|
else
|
|
{
|
|
ssize_t nr = lsquic_stream_read(stream,
|
|
st_h->file->md5str + st_h->file->md5_off, ntoread);
|
|
if (-1 == nr)
|
|
{
|
|
if (ECONNRESET == errno)
|
|
st_h->file->file_flags |= FILE_RESET;
|
|
LSQ_WARN("lsquic_stream_read: %s", strerror(errno));
|
|
lsquic_stream_close(stream);
|
|
return;
|
|
}
|
|
else
|
|
st_h->file->md5_off += nr;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
client_file_on_close (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h)
|
|
{
|
|
--st_h->client_ctx->n_files;
|
|
LSQ_NOTICE("%s called for stream %"PRIu64", # files: %u", __func__,
|
|
lsquic_stream_id(stream), st_h->client_ctx->n_files);
|
|
if (0 == st_h->client_ctx->n_files)
|
|
lsquic_conn_close(st_h->client_ctx->conn_h->conn);
|
|
if (!(st_h->file->file_flags & FILE_RESET) && 0 == RESET_AFTER_N_WRITES)
|
|
assert(st_h->file->md5_off == sizeof(st_h->file->md5str));
|
|
if (st_h->file->reader.lsqr_ctx)
|
|
{
|
|
destroy_lsquic_reader_ctx(st_h->file->reader.lsqr_ctx);
|
|
st_h->file->reader.lsqr_ctx = NULL;
|
|
}
|
|
if (st_h->file->fd >= 0)
|
|
(void) close(st_h->file->fd);
|
|
free(st_h->file);
|
|
free(st_h);
|
|
}
|
|
|
|
|
|
const struct lsquic_stream_if client_file_stream_if = {
|
|
.on_new_conn = client_on_new_conn,
|
|
.on_goaway_received = client_on_goaway_received,
|
|
.on_conn_closed = client_on_conn_closed,
|
|
.on_new_stream = client_on_new_stream,
|
|
.on_read = client_file_on_read,
|
|
.on_write = client_file_on_write,
|
|
.on_close = client_file_on_close,
|
|
};
|
|
|
|
|
|
static void
|
|
usage (const char *prog)
|
|
{
|
|
const char *const slash = strrchr(prog, '/');
|
|
if (slash)
|
|
prog = slash + 1;
|
|
printf(
|
|
"Usage: %s [opts]\n"
|
|
"\n"
|
|
"Options:\n"
|
|
" -f FILE File to send to the server -- must be specified at least\n"
|
|
" once.\n"
|
|
" -b Use buffering API for sending files over rather than\n"
|
|
" the efficient version.\n"
|
|
" -p PRIORITY Applicatble to previous file specified with -f\n"
|
|
" -r STREAM_ID:OFFSET\n"
|
|
" Reset stream STREAM_ID after sending more that OFFSET bytes.\n"
|
|
, prog);
|
|
}
|
|
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
int opt, s;
|
|
struct sport_head sports;
|
|
struct prog prog;
|
|
struct client_ctx client_ctx;
|
|
struct file *file;
|
|
|
|
file = NULL;
|
|
memset(&client_ctx, 0, sizeof(client_ctx));
|
|
client_ctx.prog = &prog;
|
|
|
|
TAILQ_INIT(&sports);
|
|
prog_init(&prog, 0, &sports, &client_file_stream_if, &client_ctx);
|
|
prog.prog_api.ea_alpn = "md5";
|
|
|
|
while (-1 != (opt = getopt(argc, argv, PROG_OPTS "bhr:f:p:")))
|
|
{
|
|
switch (opt) {
|
|
case 'p':
|
|
if (file)
|
|
file->priority = atoi(optarg);
|
|
else
|
|
{
|
|
fprintf(stderr, "No file to apply priority to\n");
|
|
exit(1);
|
|
}
|
|
break;
|
|
case 'b':
|
|
g_write_file = 0;
|
|
break;
|
|
case 'f':
|
|
file = calloc(1, sizeof(*file));
|
|
LIST_INSERT_HEAD(&client_ctx.files, file, next_file);
|
|
++client_ctx.n_files;
|
|
file->filename = optarg;
|
|
break;
|
|
case 'r':
|
|
g_reset_stream.stream_id = atoi(optarg);
|
|
g_reset_stream.offset = atoi(strchr(optarg, ':') + 1);
|
|
break;
|
|
case 'h':
|
|
usage(argv[0]);
|
|
prog_print_common_options(&prog, stdout);
|
|
exit(0);
|
|
default:
|
|
if (0 != prog_set_opt(&prog, opt, optarg))
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
if (LIST_EMPTY(&client_ctx.files))
|
|
{
|
|
fprintf(stderr, "please specify one of more files using -f\n");
|
|
exit(1);
|
|
}
|
|
|
|
if (0 != prog_prep(&prog))
|
|
{
|
|
LSQ_ERROR("could not prep");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
client_ctx.sport = TAILQ_FIRST(&sports);
|
|
|
|
if (0 != prog_connect(&prog, NULL, 0))
|
|
{
|
|
LSQ_ERROR("could not connect");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
LSQ_DEBUG("entering event loop");
|
|
|
|
s = prog_run(&prog);
|
|
prog_cleanup(&prog);
|
|
|
|
exit(0 == s ? EXIT_SUCCESS : EXIT_FAILURE);
|
|
}
|