mirror of
https://codeberg.org/ziglang/zig.git
synced 2025-12-06 13:54:21 +00:00
std.http: do -> wait, fix redirects
This commit is contained in:
parent
13101295b9
commit
7285eedcd2
3 changed files with 64 additions and 30 deletions
|
|
@ -365,7 +365,7 @@ pub const Response = struct {
|
|||
CompressionNotSupported,
|
||||
};
|
||||
|
||||
pub fn parse(res: *Response, bytes: []const u8) ParseError!void {
|
||||
pub fn parse(res: *Response, bytes: []const u8, trailing: bool) ParseError!void {
|
||||
var it = mem.tokenize(u8, bytes[0 .. bytes.len - 4], "\r\n");
|
||||
|
||||
const first_line = it.next() orelse return error.HttpHeadersInvalid;
|
||||
|
|
@ -398,6 +398,8 @@ pub const Response = struct {
|
|||
|
||||
try res.headers.append(header_name, header_value);
|
||||
|
||||
if (trailing) continue;
|
||||
|
||||
if (std.ascii.eqlIgnoreCase(header_name, "content-length")) {
|
||||
if (res.content_length != null) return error.HttpHeadersInvalid;
|
||||
res.content_length = std.fmt.parseInt(u64, header_value, 10) catch return error.InvalidContentLength;
|
||||
|
|
@ -480,7 +482,7 @@ pub const Response = struct {
|
|||
|
||||
/// A HTTP request that has been sent.
|
||||
///
|
||||
/// Order of operations: request[ -> write -> finish] -> do -> read
|
||||
/// Order of operations: request -> start[ -> write -> finish] -> wait -> read
|
||||
pub const Request = struct {
|
||||
uri: Uri,
|
||||
client: *Client,
|
||||
|
|
@ -508,8 +510,9 @@ pub const Request = struct {
|
|||
.zstd => |*zstd| zstd.deinit(),
|
||||
}
|
||||
|
||||
req.response.headers.deinit();
|
||||
|
||||
if (req.response.parser.header_bytes_owned) {
|
||||
req.response.headers.deinit();
|
||||
req.response.parser.header_bytes.deinit(req.client.allocator);
|
||||
}
|
||||
|
||||
|
|
@ -524,6 +527,44 @@ pub const Request = struct {
|
|||
req.* = undefined;
|
||||
}
|
||||
|
||||
// This function must deallocate all resources associated with the request, or keep those which will be used
|
||||
// This needs to be kept in sync with deinit and request
|
||||
fn redirect(req: *Request, uri: Uri) !void {
|
||||
assert(req.response.parser.done);
|
||||
|
||||
switch (req.response.compression) {
|
||||
.none => {},
|
||||
.deflate => |*deflate| deflate.deinit(),
|
||||
.gzip => |*gzip| gzip.deinit(),
|
||||
.zstd => |*zstd| zstd.deinit(),
|
||||
}
|
||||
|
||||
req.client.connection_pool.release(req.client, req.connection);
|
||||
|
||||
const protocol = protocol_map.get(uri.scheme) orelse return error.UnsupportedUrlScheme;
|
||||
|
||||
const port: u16 = uri.port orelse switch (protocol) {
|
||||
.plain => 80,
|
||||
.tls => 443,
|
||||
};
|
||||
|
||||
const host = uri.host orelse return error.UriMissingHost;
|
||||
|
||||
req.uri = uri;
|
||||
req.connection = try req.client.connect(host, port, protocol);
|
||||
req.redirects_left -= 1;
|
||||
req.response.headers.clearRetainingCapacity();
|
||||
req.response.parser.reset();
|
||||
|
||||
req.response = .{
|
||||
.status = undefined,
|
||||
.reason = undefined,
|
||||
.version = undefined,
|
||||
.headers = req.response.headers,
|
||||
.parser = req.response.parser,
|
||||
};
|
||||
}
|
||||
|
||||
pub const StartError = BufferedConnection.WriteError || error{ InvalidContentLength, UnsupportedTransferEncoding };
|
||||
|
||||
/// Send the request to the server.
|
||||
|
|
@ -627,14 +668,14 @@ pub const Request = struct {
|
|||
return index;
|
||||
}
|
||||
|
||||
pub const DoError = RequestError || TransferReadError || proto.HeadersParser.CheckCompleteHeadError || Response.ParseError || Uri.ParseError || error{ TooManyHttpRedirects, HttpRedirectMissingLocation, CompressionInitializationFailed, CompressionNotSupported };
|
||||
pub const WaitError = RequestError || StartError || TransferReadError || proto.HeadersParser.CheckCompleteHeadError || Response.ParseError || Uri.ParseError || error{ TooManyHttpRedirects, CannotRedirect, HttpRedirectMissingLocation, CompressionInitializationFailed, CompressionNotSupported };
|
||||
|
||||
/// Waits for a response from the server and parses any headers that are sent.
|
||||
/// This function will block until the final response is received.
|
||||
///
|
||||
/// If `handle_redirects` is true, then this function will automatically follow
|
||||
/// redirects.
|
||||
pub fn do(req: *Request) DoError!void {
|
||||
/// If `handle_redirects` is true and the request has no payload, then this function will automatically follow
|
||||
/// redirects. If a request payload is present, then this function will error with error.CannotRedirect.
|
||||
pub fn wait(req: *Request) WaitError!void {
|
||||
while (true) { // handle redirects
|
||||
while (true) { // read headers
|
||||
try req.connection.data.buffered.fill();
|
||||
|
|
@ -645,7 +686,7 @@ pub const Request = struct {
|
|||
if (req.response.parser.state.isContent()) break;
|
||||
}
|
||||
|
||||
try req.response.parse(req.response.parser.header_bytes.items);
|
||||
try req.response.parse(req.response.parser.header_bytes.items, false);
|
||||
|
||||
if (req.response.status == .switching_protocols) {
|
||||
req.connection.data.closing = false;
|
||||
|
|
@ -684,7 +725,7 @@ pub const Request = struct {
|
|||
req.response.parser.done = true;
|
||||
}
|
||||
|
||||
if (req.response.status.class() == .redirect and req.handle_redirects) {
|
||||
if (req.transfer_encoding == .none and req.response.status.class() == .redirect and req.handle_redirects) {
|
||||
req.response.skip = true;
|
||||
|
||||
const empty = @as([*]u8, undefined)[0..0];
|
||||
|
|
@ -694,26 +735,17 @@ pub const Request = struct {
|
|||
|
||||
const location = req.response.headers.getFirstValue("location") orelse
|
||||
return error.HttpRedirectMissingLocation;
|
||||
const new_url = Uri.parse(location) catch try Uri.parseWithoutScheme(location);
|
||||
|
||||
var new_arena = std.heap.ArenaAllocator.init(req.client.allocator);
|
||||
const resolved_url = try req.uri.resolve(new_url, false, new_arena.allocator());
|
||||
errdefer new_arena.deinit();
|
||||
const arena = req.arena.allocator();
|
||||
|
||||
req.arena.deinit();
|
||||
req.arena = new_arena;
|
||||
const location_duped = try arena.dupe(u8, location);
|
||||
|
||||
const new_req = try req.client.request(req.method, resolved_url, req.headers, .{
|
||||
.version = req.version,
|
||||
.max_redirects = req.redirects_left - 1,
|
||||
.header_strategy = if (req.response.parser.header_bytes_owned) .{
|
||||
.dynamic = req.response.parser.max_header_bytes,
|
||||
} else .{
|
||||
.static = req.response.parser.header_bytes.items.ptr[0..req.response.parser.max_header_bytes],
|
||||
},
|
||||
});
|
||||
req.deinit();
|
||||
req.* = new_req;
|
||||
const new_url = Uri.parse(location_duped) catch try Uri.parseWithoutScheme(location_duped);
|
||||
const resolved_url = try req.uri.resolve(new_url, false, arena);
|
||||
|
||||
try req.redirect(resolved_url);
|
||||
|
||||
try req.start();
|
||||
} else {
|
||||
req.response.skip = false;
|
||||
if (!req.response.parser.done) {
|
||||
|
|
@ -731,6 +763,9 @@ pub const Request = struct {
|
|||
};
|
||||
}
|
||||
|
||||
if (req.response.status.class() == .redirect and req.handle_redirects and req.transfer_encoding != .none)
|
||||
return error.CannotRedirect; // The request body has already been sent. The request is still in a valid state, but the redirect must be handled manually.
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
@ -768,7 +803,7 @@ pub const Request = struct {
|
|||
|
||||
// The response headers before the trailers are already guaranteed to be valid, so they will always be parsed again and cannot return an error.
|
||||
// This will *only* fail for a malformed trailer.
|
||||
req.response.parse(req.response.parser.header_bytes.items) catch return error.InvalidTrailers;
|
||||
req.response.parse(req.response.parser.header_bytes.items, true) catch return error.InvalidTrailers;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -62,7 +62,7 @@ test "client requests server" {
|
|||
try client_req.writeAll("Hello, ");
|
||||
try client_req.writeAll("World!\n");
|
||||
try client_req.finish();
|
||||
try client_req.do(); // this waits for a response
|
||||
try client_req.wait(); // this waits for a response
|
||||
|
||||
const body = try client_req.reader().readAllAlloc(allocator, 8192 * 1024);
|
||||
defer allocator.free(body);
|
||||
|
|
|
|||
|
|
@ -486,8 +486,7 @@ fn fetchAndUnpack(
|
|||
defer req.deinit();
|
||||
|
||||
try req.start();
|
||||
|
||||
try req.do();
|
||||
try req.wait();
|
||||
|
||||
if (mem.endsWith(u8, uri.path, ".tar.gz")) {
|
||||
// I observed the gzip stream to read 1 byte at a time, so I am using a
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue