From 96a62af7f5d2ec4f774b51bf2183d9154ace6820 Mon Sep 17 00:00:00 2001 From: Casper Kuethe Date: Sun, 18 Feb 2024 22:44:59 +0100 Subject: [PATCH 1/3] add configuration options --- vlib/io/buffered_reader.v | 9 ++ .../tests/config/incomplete_request_test.v | 86 ++++++++++ .../vweb/tests/config/max_header_size_test.v | 59 +++++++ .../tests/config/max_request_body_len_test.v | 56 +++++++ vlib/x/vweb/tests/large_payload_test.v | 12 +- vlib/x/vweb/vweb.v | 149 ++++++++++++++---- 6 files changed, 339 insertions(+), 32 deletions(-) create mode 100644 vlib/x/vweb/tests/config/incomplete_request_test.v create mode 100644 vlib/x/vweb/tests/config/max_header_size_test.v create mode 100644 vlib/x/vweb/tests/config/max_request_body_len_test.v diff --git a/vlib/io/buffered_reader.v b/vlib/io/buffered_reader.v index 5cd9c1e83de1a2..ba9bae8d78192c 100644 --- a/vlib/io/buffered_reader.v +++ b/vlib/io/buffered_reader.v @@ -156,3 +156,12 @@ pub fn (mut r BufferedReader) read_line(config BufferedReadLineConfig) !string { } return Eof{} } + +// get_read_data returns the data that the buffered reader has read from `start` until `end` +pub fn (r &BufferedReader) get_read_data(start int, end int) ![]u8 { + if end > r.buf.len { + return Eof{} + } + + return r.buf[start..end] +} diff --git a/vlib/x/vweb/tests/config/incomplete_request_test.v b/vlib/x/vweb/tests/config/incomplete_request_test.v new file mode 100644 index 00000000000000..789d8a986b79cb --- /dev/null +++ b/vlib/x/vweb/tests/config/incomplete_request_test.v @@ -0,0 +1,86 @@ +import time +import x.vweb +import net +import net.http + +const port = 13020 +const exit_after = time.second * 20 + +pub struct Context { + vweb.Context +} + +pub struct App { +mut: + started chan bool +} + +pub fn (mut app App) before_accept_loop() { + app.started <- true +} + +pub fn (mut app App) index(mut ctx Context) vweb.Result { + return ctx.text('Hello V!') +} + +fn testsuite_begin() { + mut app := &App{} + spawn vweb.run_at[App, Context](mut app, + port: port + timeout_in_seconds: 15 + config: vweb.Config{ + max_nr_of_incomplete_requests: 1 + } + ) + _ := <-app.started + + spawn fn () { + time.sleep(exit_after) + assert false, 'timeout reached' + exit(1) + }() +} + +fn test_incomplete_request() { + // simulate a slow network where the request headers are received in parts + // we set the maximum number of incomplete requests on a connection to 1. + // This means that when we send another incomplete request we expect vweb + // to close the connection. + + mut conn := net.dial_tcp('127.0.0.1:${port}')! + defer { + conn.close() or {} + } + + conn.write_string('GET / HTTP/1.1\r +User-Agent: V\r +Acc')! + + // simulate network delay + time.sleep(time.second * 2) + conn.write_string('ept: */*\r\n\r\n')! + + // simulate network delay + time.sleep(time.second * 2) + mut buf := []u8{len: 86} + conn.read(mut buf)! + resp := http.parse_response(buf.bytestr())! + assert resp.status() == .ok + assert resp.body == 'Hello V!' + + // do second request + + // simulate network delay + time.sleep(time.second * 2) + conn.write_string('GET / HTTP/1.1\r +User-Agent: V\r +Acc')! + + // simulate network delay + time.sleep(time.second * 2) + + buf = []u8{len: 103} + conn.read(mut buf)! + bad_resp := http.parse_response(buf.bytestr())! + assert bad_resp.status() == .bad_request +} diff --git a/vlib/x/vweb/tests/config/max_header_size_test.v b/vlib/x/vweb/tests/config/max_header_size_test.v new file mode 100644 index 00000000000000..f6c9f0117eb31f --- /dev/null +++ b/vlib/x/vweb/tests/config/max_header_size_test.v @@ -0,0 +1,59 @@ +import net.http +import time +import x.vweb + +const port = 13021 +const exit_after = time.second * 10 +const header_size = 1000 +const localserver = 'http://localhost:${port}' + +pub struct Context { + vweb.Context +} + +pub struct App { +mut: + started chan bool +} + +pub fn (mut app App) before_accept_loop() { + app.started <- true +} + +pub fn (mut app App) index(mut ctx Context) vweb.Result { + return ctx.text('Hello V!') +} + +fn testsuite_begin() { + mut app := &App{} + spawn vweb.run_at[App, Context](mut app, + port: port + timeout_in_seconds: 2 + config: vweb.Config{ + max_header_len: header_size + } + ) + _ := <-app.started + + spawn fn () { + time.sleep(exit_after) + assert false, 'timeout reached' + exit(1) + }() +} + +fn test_larger_header_size() { + mut buf := []u8{len: header_size * 2, init: `a`} + str := buf.bytestr() + + mut header := http.new_custom_header_from_map({ + 'X-Large-Header': str + })! + + mut x := http.fetch(http.FetchConfig{ + url: localserver + header: header + })! + + assert x.status() == .request_entity_too_large +} diff --git a/vlib/x/vweb/tests/config/max_request_body_len_test.v b/vlib/x/vweb/tests/config/max_request_body_len_test.v new file mode 100644 index 00000000000000..bef2bfcc8bc9d2 --- /dev/null +++ b/vlib/x/vweb/tests/config/max_request_body_len_test.v @@ -0,0 +1,56 @@ +import time +import x.vweb +import net +import net.http + +const port = 13022 +const exit_after = time.second * 10 +const body_len = 1000 +const localserver = 'http://localhost:${port}' + +pub struct Context { + vweb.Context +} + +pub struct App { +mut: + started chan bool +} + +pub fn (mut app App) before_accept_loop() { + app.started <- true +} + +pub fn (mut app App) index(mut ctx Context) vweb.Result { + return ctx.text('Hello V!') +} + +fn testsuite_begin() { + mut app := &App{} + spawn vweb.run_at[App, Context](mut app, + port: port + timeout_in_seconds: 15 + config: vweb.Config{ + max_request_body_len: body_len + } + ) + _ := <-app.started + + spawn fn () { + time.sleep(exit_after) + assert false, 'timeout reached' + exit(1) + }() +} + +fn test_larger_body_len() { + mut buf := []u8{len: body_len * 2, init: `a`} + str := buf.bytestr() + + mut x := http.fetch(http.FetchConfig{ + url: localserver + data: str + })! + + assert x.status() == .request_entity_too_large +} diff --git a/vlib/x/vweb/tests/large_payload_test.v b/vlib/x/vweb/tests/large_payload_test.v index 8989b3cadab014..6b9e912b41fe4a 100644 --- a/vlib/x/vweb/tests/large_payload_test.v +++ b/vlib/x/vweb/tests/large_payload_test.v @@ -10,6 +10,7 @@ const port = 13002 const localserver = 'http://127.0.0.1:${port}' const exit_after = time.second * 10 +const header_size = vweb.max_read * 2 const tmp_file = os.join_path(os.vtmp_dir(), 'vweb_large_payload.txt') @@ -47,7 +48,14 @@ fn testsuite_begin() { }() mut app := &App{} - spawn vweb.run_at[App, Context](mut app, port: port, timeout_in_seconds: 2, family: .ip) + spawn vweb.run_at[App, Context](mut app, + port: port + timeout_in_seconds: 2 + family: .ip + config: vweb.Config{ + max_header_len: header_size + } + ) // app startup time _ := <-app.started } @@ -68,7 +76,7 @@ fn test_large_request_body() { fn test_large_request_header() { // same test as test_large_request_body, but then with a large header, // which is parsed seperately - mut buf := []u8{len: vweb.max_read * 2, init: `a`} + mut buf := []u8{len: header_size, init: `a`} str := buf.bytestr() // make 1 header longer than vwebs max read limit diff --git a/vlib/x/vweb/vweb.v b/vlib/x/vweb/vweb.v index cd684afae2e251..51b6d3a4e31c6a 100644 --- a/vlib/x/vweb/vweb.v +++ b/vlib/x/vweb/vweb.v @@ -216,6 +216,39 @@ pub struct RunParams { port int = 8080 show_startup_message bool = true timeout_in_seconds int = 30 + config Config +} + +// Config contains options that can change the internal behaviour of vweb +pub struct Config { +pub mut: + // maximum size of HTTP headers in bytes, default is 96KB + max_header_len int = 96_000 + // maximum number of incomplete HTTP requests that can be send over a connection + // before vweb closes the connection + max_nr_of_incomplete_requests int = 3 + // the maximum size of a request body in bytes, default is 1MB + max_request_body_len int = 1_000_000 +} + +struct IncomingRequest { +pub mut: + req http.Request + buf []u8 + nr_incomplete_requests int +} + +// done frees the current incomplete request header buffer and reset the HTTP request. +// use this function when the connection is kept open, but the request is handled +@[manualfree] +pub fn (mut ir IncomingRequest) done() { + ir.buf.clear() + ir.req = http.Request{} +} + +// reset set the incomplete request counter to 0 +pub fn (mut ir IncomingRequest) reset() { + ir.nr_incomplete_requests = 0 } struct FileResponse { @@ -259,21 +292,22 @@ struct RequestParams { controllers []&ControllerPath routes &map[string]Route timeout_in_seconds int + config Config mut: // request body buffer buf &u8 = unsafe { nil } // idx keeps track of how much of the request body has been read // for each incomplete request, see `handle_conn` - idx []int - incomplete_requests []http.Request - file_responses []FileResponse - string_responses []StringResponse + idx []int + incoming_requests []IncomingRequest + file_responses []FileResponse + string_responses []StringResponse } // reset request parameters for `fd`: // reset content-length index and the http request pub fn (mut params RequestParams) request_done(fd int) { - params.incomplete_requests[fd] = http.Request{} + params.incoming_requests[fd].done() params.idx[fd] = 0 } @@ -304,6 +338,7 @@ pub fn run_at[A, X](mut global_app A, params RunParams) ! { controllers: controllers_sorted routes: &routes timeout_in_seconds: params.timeout_in_seconds + config: params.config } pico_context.idx = []int{len: picoev.max_fds} @@ -312,7 +347,7 @@ pub fn run_at[A, X](mut global_app A, params RunParams) ! { defer { unsafe { free(pico_context.buf) } } - pico_context.incomplete_requests = []http.Request{len: picoev.max_fds} + pico_context.incoming_requests = []IncomingRequest{len: picoev.max_fds} pico_context.file_responses = []FileResponse{len: picoev.max_fds} pico_context.string_responses = []StringResponse{len: picoev.max_fds} @@ -359,6 +394,7 @@ fn ev_callback[A, X](mut pv picoev.Picoev, fd int, events int) { $if debug_ev_callback ? { eprintln('[vweb] error: write event on connection should be closed') } + params.incoming_requests[fd].reset() pv.close_conn(fd) } } else if events == picoev.picoev_read { @@ -380,6 +416,7 @@ fn handle_timeout(mut pv picoev.Picoev, mut params RequestParams, fd int) { } fast_send_resp(mut conn, vweb.http_408) or {} + params.incoming_requests[fd].reset() pv.close_conn(fd) params.request_done(fd) @@ -411,11 +448,13 @@ fn handle_write_file(mut pv picoev.Picoev, mut params RequestParams, fd int) { params.file_responses[fd].file.read_into_ptr(data, bytes_to_write) or { params.file_responses[fd].done() + params.incoming_requests[fd].reset() pv.close_conn(fd) return } actual_written := send_string_ptr(mut conn, data, bytes_to_write) or { params.file_responses[fd].done() + params.incoming_requests[fd].reset() pv.close_conn(fd) return } @@ -425,7 +464,8 @@ fn handle_write_file(mut pv picoev.Picoev, mut params RequestParams, fd int) { if params.file_responses[fd].pos == params.file_responses[fd].total { // file is done writing params.file_responses[fd].done() - handle_complete_request(params.file_responses[fd].should_close_conn, mut pv, fd) + handle_complete_request(params.file_responses[fd].should_close_conn, mut params, mut + pv, fd) return } } @@ -449,6 +489,7 @@ fn handle_write_string(mut pv picoev.Picoev, mut params RequestParams, fd int) { data := unsafe { params.string_responses[fd].str.str + params.string_responses[fd].pos } actual_written := send_string_ptr(mut conn, data, bytes_to_write) or { params.string_responses[fd].done() + params.incoming_requests[fd].reset() pv.close_conn(fd) return } @@ -456,9 +497,8 @@ fn handle_write_string(mut pv picoev.Picoev, mut params RequestParams, fd int) { if params.string_responses[fd].pos == params.string_responses[fd].str.len { // done writing params.string_responses[fd].done() - pv.close_conn(fd) - handle_complete_request(params.string_responses[fd].should_close_conn, mut pv, - fd) + handle_complete_request(params.string_responses[fd].should_close_conn, mut params, mut + pv, fd) return } } @@ -475,8 +515,28 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { is_blocking: false } - // cap the max_read to 8KB - mut reader := io.new_buffered_reader(reader: conn, cap: vweb.max_read) + mut incoming_req := ¶ms.incoming_requests[fd] + is_first_data_of_request := incoming_req.buf.len == 0 + + mut reader := if !is_first_data_of_request { + // the previous request was incomplete, append incoming data to the buffer + // so the request headers can be parsed again in their entirety + max_size := incoming_req.buf.len + vweb.max_read + mut new_buf := []u8{len: max_size, cap: max_size} + // copy the previous request buffer to the new buffered reader + copy(mut new_buf, incoming_req.buf) + &io.BufferedReader{ + reader: conn + buf: new_buf + len: incoming_req.buf.len + offset: 0 + mfails: 2 + } + } else { + // cap the max_read to 8KB + io.new_buffered_reader(reader: conn, cap: vweb.max_read) + } + defer { unsafe { reader.free() @@ -484,7 +544,7 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { } // take the previous incomplete request - mut req := params.incomplete_requests[fd] + mut req := incoming_req.req // check if there is an incomplete request for this file desriptor if params.idx[fd] == 0 { @@ -495,24 +555,39 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { // first time that this connection is being read from, so we parse the // request header first req = http.parse_request_head(mut reader) or { - // Prevents errors from being thrown when BufferedReader is empty + // Prevents errors from being thrown when BufferedReader is empty. + // If the buffered reader was empty it means that the client probably + // closed the connection. if err !is io.Eof { - eprintln('[vweb] error parsing request: ${err}') + // a request could not be parsed + if is_first_data_of_request + && incoming_req.nr_incomplete_requests >= params.config.max_nr_of_incomplete_requests { + fast_send_resp(mut conn, vweb.http_400) or {} + } else if is_first_data_of_request { + // too many incomplete requests send by the client. Closing connection... + incoming_req.nr_incomplete_requests++ + } + + // store the currently read data and wait for the client to sent more data + if total_read := reader.get_read_data(incoming_req.buf.len, reader.total_read) { + incoming_req.buf << total_read + return + } } - // the buffered reader was empty meaning that the client probably - // closed the connection. + + params.request_done(fd) pv.close_conn(fd) - params.incomplete_requests[fd] = http.Request{} return } - if reader.total_read >= vweb.max_read { + if reader.total_read >= params.config.max_header_len { // throw an error when the request header is larger than 8KB // same limit that apache handles - eprintln('[vweb] error parsing request: too large') + eprintln('[vweb] error parsing request: request headers are too large') fast_send_resp(mut conn, vweb.http_413) or {} + params.request_done(fd) + params.incoming_requests[fd].reset() pv.close_conn(fd) - params.incomplete_requests[fd] = http.Request{} return } } @@ -536,16 +611,26 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { n := reader.read(mut buf) or { eprintln('[vweb] error parsing request: ${err}') + params.request_done(fd) + params.incoming_requests[fd].reset() pv.close_conn(fd) - params.incomplete_requests[fd] = http.Request{} - params.idx[fd] = 0 return } + // body size has exceeded the set limits + if params.idx[fd] + n > params.config.max_request_body_len { + eprintln('[vweb] error parsing request: request body is too large') + fast_send_resp(mut conn, vweb.http_413) or {} + + params.request_done(fd) + params.incoming_requests[fd].reset() + pv.close_conn(fd) + return + } // there is no more data to be sent, but it is less than the Content-Length header // so it is a mismatch of body length and content length. // Or if there is more data received then the Content-Length header specified - if (n == 0 && params.idx[fd] != 0) || params.idx[fd] + n > content_length.int() { + else if (n == 0 && params.idx[fd] != 0) || params.idx[fd] + n > content_length.int() { fast_send_resp(mut conn, http.new_response( status: .bad_request body: 'Mismatch of body length and Content-Length header' @@ -555,16 +640,16 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { ).join(vweb.headers_close) )) or {} + params.request_done(fd) + params.incoming_requests[fd].reset() pv.close_conn(fd) - params.incomplete_requests[fd] = http.Request{} - params.idx[fd] = 0 return } else if n < bytes_to_read || params.idx[fd] + n < content_length.int() { // request is incomplete wait until the socket becomes ready to read again params.idx[fd] += n // TODO: change this to a memcpy function? req.data += buf[0..n].bytestr() - params.incomplete_requests[fd] = req + params.incoming_requests[fd].req = req return } else { // request is complete: n = bytes_to_read @@ -600,7 +685,7 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { if completed_context.res.body.len < vweb.max_read { fast_send_resp(mut conn, completed_context.res) or {} handle_complete_request(completed_context.client_wants_to_close, mut - pv, fd) + params, mut pv, fd) } else { params.string_responses[fd].open = true params.string_responses[fd].str = completed_context.res.body @@ -612,7 +697,7 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { params.string_responses[fd].done() fast_send_resp(mut conn, vweb.http_500) or {} handle_complete_request(completed_context.client_wants_to_close, mut - pv, fd) + params, mut pv, fd) return } // no errors we can send the HTTP headers @@ -630,6 +715,7 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { // Context checks if the file is valid, so this should never happen fast_send_resp(mut conn, vweb.http_500) or {} params.file_responses[fd].done() + params.incoming_requests[fd].reset() pv.close_conn(fd) return } @@ -641,6 +727,7 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { // should not happen fast_send_resp(mut conn, vweb.http_500) or {} params.file_responses[fd].done() + params.incoming_requests[fd].reset() pv.close_conn(fd) return } @@ -650,14 +737,16 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { } } else { // invalid request headers/data + params.incoming_requests[fd].reset() pv.close_conn(fd) } } // close the connection when `should_close` is true. @[inline] -fn handle_complete_request(should_close bool, mut pv picoev.Picoev, fd int) { +fn handle_complete_request(should_close bool, mut params RequestParams, mut pv picoev.Picoev, fd int) { if should_close { + params.incoming_requests[fd].reset() pv.close_conn(fd) } } From a71b427c3242baefa04e18ead24c8ff7a3d3e893 Mon Sep 17 00:00:00 2001 From: Casper Kuethe Date: Sun, 18 Feb 2024 22:46:31 +0100 Subject: [PATCH 2/3] mark incomplete request test as flaky --- vlib/x/vweb/tests/config/incomplete_request_test.v | 2 ++ 1 file changed, 2 insertions(+) diff --git a/vlib/x/vweb/tests/config/incomplete_request_test.v b/vlib/x/vweb/tests/config/incomplete_request_test.v index 789d8a986b79cb..7de36e3811f319 100644 --- a/vlib/x/vweb/tests/config/incomplete_request_test.v +++ b/vlib/x/vweb/tests/config/incomplete_request_test.v @@ -1,3 +1,5 @@ +// vtest flaky: true +// vtest retry: 3 import time import x.vweb import net From a3307f41c3343a439873209da4f8689f7807527e Mon Sep 17 00:00:00 2001 From: Casper Kuethe Date: Sun, 18 Feb 2024 23:01:53 +0100 Subject: [PATCH 3/3] reuse buffer --- vlib/x/vweb/vweb.v | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/vlib/x/vweb/vweb.v b/vlib/x/vweb/vweb.v index 51b6d3a4e31c6a..09825bda499b1b 100644 --- a/vlib/x/vweb/vweb.v +++ b/vlib/x/vweb/vweb.v @@ -521,13 +521,9 @@ fn handle_read[A, X](mut pv picoev.Picoev, mut params RequestParams, fd int) { mut reader := if !is_first_data_of_request { // the previous request was incomplete, append incoming data to the buffer // so the request headers can be parsed again in their entirety - max_size := incoming_req.buf.len + vweb.max_read - mut new_buf := []u8{len: max_size, cap: max_size} - // copy the previous request buffer to the new buffered reader - copy(mut new_buf, incoming_req.buf) &io.BufferedReader{ reader: conn - buf: new_buf + buf: incoming_req.buf len: incoming_req.buf.len offset: 0 mfails: 2