const builtin = @import("builtin"); const std = @import("std"); const testing = std.testing; test "trailers" { if (builtin.single_threaded) return error.SkipZigTest; if (builtin.os.tag == .wasi) return error.SkipZigTest; const gpa = testing.allocator; const address = try std.net.Address.parseIp("127.0.0.1", 0); var http_server = try address.listen(.{ .reuse_address = true, }); const port = http_server.listen_address.in.getPort(); const server_thread = try std.Thread.spawn(.{}, serverThread, .{&http_server}); defer server_thread.join(); var client: std.http.Client = .{ .allocator = gpa }; defer client.deinit(); const location = try std.fmt.allocPrint(gpa, "http://127.0.0.1:{d}/trailer", .{port}); defer gpa.free(location); const uri = try std.Uri.parse(location); { var server_header_buffer: [1024]u8 = undefined; var req = try client.open(.GET, uri, .{ .server_header_buffer = &server_header_buffer, }); defer req.deinit(); try req.send(.{}); try req.wait(); const body = try req.reader().readAllAlloc(gpa, 8192); defer gpa.free(body); try testing.expectEqualStrings("Hello, World!\n", body); var it = req.response.iterateHeaders(); { const header = it.next().?; try testing.expect(!it.is_trailer); try testing.expectEqualStrings("connection", header.name); try testing.expectEqualStrings("keep-alive", header.value); } { const header = it.next().?; try testing.expect(!it.is_trailer); try testing.expectEqualStrings("transfer-encoding", header.name); try testing.expectEqualStrings("chunked", header.value); } { const header = it.next().?; try testing.expect(it.is_trailer); try testing.expectEqualStrings("X-Checksum", header.name); try testing.expectEqualStrings("aaaa", header.value); } try testing.expectEqual(null, it.next()); } // connection has been kept alive try testing.expect(client.connection_pool.free_len == 1); } fn serverThread(http_server: *std.net.Server) anyerror!void { var header_buffer: [1024]u8 = undefined; var remaining: usize = 1; accept: while (remaining != 0) : (remaining -= 1) { const conn = try http_server.accept(); defer conn.stream.close(); var res = std.http.Server.init(conn, .{ .client_header_buffer = &header_buffer }); res.wait() catch |err| switch (err) { error.HttpHeadersInvalid => continue :accept, error.EndOfStream => continue, else => return err, }; try serve(&res); try testing.expectEqual(.reset, res.reset()); } } fn serve(res: *std.http.Server) !void { try testing.expectEqualStrings(res.request.target, "/trailer"); res.transfer_encoding = .chunked; try res.send(); try res.writeAll("Hello, "); try res.writeAll("World!\n"); try res.connection.writeAll("0\r\nX-Checksum: aaaa\r\n\r\n"); } test "HTTP server handles a chunked transfer coding request" { // This test requires spawning threads. if (builtin.single_threaded) { return error.SkipZigTest; } const native_endian = comptime builtin.cpu.arch.endian(); if (builtin.zig_backend == .stage2_llvm and native_endian == .big) { // https://github.com/ziglang/zig/issues/13782 return error.SkipZigTest; } if (builtin.os.tag == .wasi) return error.SkipZigTest; const allocator = std.testing.allocator; const expect = std.testing.expect; const max_header_size = 8192; const address = try std.net.Address.parseIp("127.0.0.1", 0); var server = try address.listen(.{ .reuse_address = true }); defer server.deinit(); const server_port = server.listen_address.in.getPort(); const server_thread = try std.Thread.spawn(.{}, (struct { fn apply(s: *std.net.Server) !void { var header_buffer: [max_header_size]u8 = undefined; const conn = try s.accept(); defer conn.stream.close(); var res = std.http.Server.init(conn, .{ .client_header_buffer = &header_buffer }); try res.wait(); try expect(res.request.transfer_encoding == .chunked); const server_body: []const u8 = "message from server!\n"; res.transfer_encoding = .{ .content_length = server_body.len }; res.extra_headers = &.{ .{ .name = "content-type", .value = "text/plain" }, }; res.keep_alive = false; try res.send(); var buf: [128]u8 = undefined; const n = try res.readAll(&buf); try expect(std.mem.eql(u8, buf[0..n], "ABCD")); _ = try res.writer().writeAll(server_body); try res.finish(); } }).apply, .{&server}); const request_bytes = "POST / HTTP/1.1\r\n" ++ "Content-Type: text/plain\r\n" ++ "Transfer-Encoding: chunked\r\n" ++ "\r\n" ++ "1\r\n" ++ "A\r\n" ++ "1\r\n" ++ "B\r\n" ++ "2\r\n" ++ "CD\r\n" ++ "0\r\n" ++ "\r\n"; const stream = try std.net.tcpConnectToHost(allocator, "127.0.0.1", server_port); defer stream.close(); _ = try stream.writeAll(request_bytes[0..]); server_thread.join(); }