Function findEndRecord [src]

Find and return the end record for the given seekable zip stream. Note that seekable_stream must be an instance of std.io.SeekableStream and its context must also have a .reader() method that returns an instance of std.io.Reader.

Prototype

pub fn findEndRecord(seekable_stream: anytype, stream_len: u64) !EndRecord

Parameters

stream_len: u64

Source

pub fn findEndRecord(seekable_stream: anytype, stream_len: u64) !EndRecord { var buf: [@sizeOf(EndRecord) + std.math.maxInt(u16)]u8 = undefined; const record_len_max = @min(stream_len, buf.len); var loaded_len: u32 = 0; var comment_len: u16 = 0; while (true) { const record_len: u32 = @as(u32, comment_len) + @sizeOf(EndRecord); if (record_len > record_len_max) return error.ZipNoEndRecord; if (record_len > loaded_len) { const new_loaded_len = @min(loaded_len + 300, record_len_max); const read_len = new_loaded_len - loaded_len; try seekable_stream.seekTo(stream_len - @as(u64, new_loaded_len)); const read_buf: []u8 = buf[buf.len - new_loaded_len ..][0..read_len]; const len = try seekable_stream.context.reader().readAll(read_buf); if (len != read_len) return error.ZipTruncated; loaded_len = new_loaded_len; } const record_bytes = buf[buf.len - record_len ..][0..@sizeOf(EndRecord)]; if (std.mem.eql(u8, record_bytes[0..4], &end_record_sig) and std.mem.readInt(u16, record_bytes[20..22], .little) == comment_len) { const record: *align(1) EndRecord = @ptrCast(record_bytes.ptr); if (builtin.target.cpu.arch.endian() != .little) { std.mem.byteSwapAllFields(@TypeOf(record.*), record); } return record.*; } if (comment_len == std.math.maxInt(u16)) return error.ZipNoEndRecord; comment_len += 1; } }