Commit 87f32bec26

Andrew Kelley <andrew@ziglang.org>
2025-07-01 17:28:38
std.io.Reader: finish implementing the unit tests
1 parent 49093a6
Changed files (2)
lib/std/io/Reader.zig
@@ -445,7 +445,7 @@ pub fn toss(r: *Reader, n: usize) void {
 }
 
 /// Equivalent to `toss(r.bufferedLen())`.
-pub fn tossAll(r: *Reader) void {
+pub fn tossBuffered(r: *Reader) void {
     r.seek = 0;
     r.end = 0;
 }
@@ -557,7 +557,7 @@ pub fn discardShort(r: *Reader, n: usize) ShortError!usize {
 /// See also:
 /// * `peek`
 /// * `readSliceShort`
-pub fn readSlice(r: *Reader, buffer: []u8) Error!void {
+pub fn readSliceAll(r: *Reader, buffer: []u8) Error!void {
     const n = try readSliceShort(r, buffer);
     if (n != buffer.len) return error.EndOfStream;
 }
@@ -571,7 +571,7 @@ pub fn readSlice(r: *Reader, buffer: []u8) Error!void {
 /// only if the stream reached the end.
 ///
 /// See also:
-/// * `readSlice`
+/// * `readSliceAll`
 pub fn readSliceShort(r: *Reader, buffer: []u8) ShortError!usize {
     const in_buffer = r.buffer[r.seek..r.end];
     const copy_len = @min(buffer.len, in_buffer.len);
@@ -630,7 +630,7 @@ pub fn readSliceShort(r: *Reader, buffer: []u8) ShortError!usize {
 /// comptime-known and matches host endianness.
 ///
 /// See also:
-/// * `readSlice`
+/// * `readSliceAll`
 /// * `readSliceEndianAlloc`
 pub inline fn readSliceEndian(
     r: *Reader,
@@ -638,7 +638,7 @@ pub inline fn readSliceEndian(
     buffer: []Elem,
     endian: std.builtin.Endian,
 ) Error!void {
-    try readSlice(r, @ptrCast(buffer));
+    try readSliceAll(r, @ptrCast(buffer));
     if (native_endian != endian) for (buffer) |*elem| std.mem.byteSwapAllFields(Elem, elem);
 }
 
@@ -655,15 +655,16 @@ pub inline fn readSliceEndianAlloc(
 ) ReadAllocError![]Elem {
     const dest = try allocator.alloc(Elem, len);
     errdefer allocator.free(dest);
-    try readSlice(r, @ptrCast(dest));
+    try readSliceAll(r, @ptrCast(dest));
     if (native_endian != endian) for (dest) |*elem| std.mem.byteSwapAllFields(Elem, elem);
     return dest;
 }
 
-pub fn readSliceAlloc(r: *Reader, allocator: Allocator, len: usize) ReadAllocError![]u8 {
+/// Shortcut for calling `readSliceAll` with a buffer provided by `allocator`.
+pub fn readAlloc(r: *Reader, allocator: Allocator, len: usize) ReadAllocError![]u8 {
     const dest = try allocator.alloc(u8, len);
     errdefer allocator.free(dest);
-    try readSlice(r, dest);
+    try readSliceAll(r, dest);
     return dest;
 }
 
@@ -1092,6 +1093,7 @@ pub fn takeVarInt(r: *Reader, comptime Int: type, endian: std.builtin.Endian, n:
 ///
 /// See also:
 /// * `peekStruct`
+/// * `takeStructEndian`
 pub fn takeStruct(r: *Reader, comptime T: type) Error!*align(1) T {
     // Only extern and packed structs have defined in-memory layout.
     comptime assert(@typeInfo(T).@"struct".layout != .auto);
@@ -1104,6 +1106,7 @@ pub fn takeStruct(r: *Reader, comptime T: type) Error!*align(1) T {
 ///
 /// See also:
 /// * `takeStruct`
+/// * `peekStructEndian`
 pub fn peekStruct(r: *Reader, comptime T: type) Error!*align(1) T {
     // Only extern and packed structs have defined in-memory layout.
     comptime assert(@typeInfo(T).@"struct".layout != .auto);
@@ -1114,6 +1117,10 @@ pub fn peekStruct(r: *Reader, comptime T: type) Error!*align(1) T {
 ///
 /// This function is inline to avoid referencing `std.mem.byteSwapAllFields`
 /// when `endian` is comptime-known and matches the host endianness.
+///
+/// See also:
+/// * `takeStruct`
+/// * `peekStructEndian`
 pub inline fn takeStructEndian(r: *Reader, comptime T: type, endian: std.builtin.Endian) Error!T {
     var res = (try r.takeStruct(T)).*;
     if (native_endian != endian) std.mem.byteSwapAllFields(T, &res);
@@ -1124,6 +1131,10 @@ pub inline fn takeStructEndian(r: *Reader, comptime T: type, endian: std.builtin
 ///
 /// This function is inline to avoid referencing `std.mem.byteSwapAllFields`
 /// when `endian` is comptime-known and matches the host endianness.
+///
+/// See also:
+/// * `takeStructEndian`
+/// * `peekStruct`
 pub inline fn peekStructEndian(r: *Reader, comptime T: type, endian: std.builtin.Endian) Error!T {
     var res = (try r.peekStruct(T)).*;
     if (native_endian != endian) std.mem.byteSwapAllFields(T, &res);
@@ -1489,25 +1500,47 @@ test takeInt {
 
 test takeVarInt {
     var r: Reader = .fixed(&.{ 0x12, 0x34, 0x56 });
-    std.debug.print("{x}", .{r.buffer});
     try testing.expectEqual(0x123456, try r.takeVarInt(u64, .big, 3));
     try testing.expectError(error.EndOfStream, r.takeVarInt(u16, .little, 1));
 }
 
 test takeStruct {
-    return error.Unimplemented;
+    var r: Reader = .fixed(&.{ 0x12, 0x00, 0x34, 0x56 });
+    const S = extern struct { a: u8, b: u16 };
+    switch (native_endian) {
+        .little => try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x5634 }), (try r.takeStruct(S)).*),
+        .big => try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x3456 }), (try r.takeStruct(S)).*),
+    }
+    try testing.expectError(error.EndOfStream, r.takeStruct(S));
 }
 
 test peekStruct {
-    return error.Unimplemented;
+    var r: Reader = .fixed(&.{ 0x12, 0x00, 0x34, 0x56 });
+    const S = extern struct { a: u8, b: u16 };
+    switch (native_endian) {
+        .little => {
+            try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x5634 }), (try r.peekStruct(S)).*);
+            try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x5634 }), (try r.peekStruct(S)).*);
+        },
+        .big => {
+            try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x3456 }), (try r.peekStruct(S)).*);
+            try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x3456 }), (try r.peekStruct(S)).*);
+        },
+    }
 }
 
 test takeStructEndian {
-    return error.Unimplemented;
+    var r: Reader = .fixed(&.{ 0x12, 0x00, 0x34, 0x56 });
+    const S = extern struct { a: u8, b: u16 };
+    try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x3456 }), try r.takeStructEndian(S, .big));
+    try testing.expectError(error.EndOfStream, r.takeStructEndian(S, .little));
 }
 
 test peekStructEndian {
-    return error.Unimplemented;
+    var r: Reader = .fixed(&.{ 0x12, 0x00, 0x34, 0x56 });
+    const S = extern struct { a: u8, b: u16 };
+    try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x3456 }), try r.peekStructEndian(S, .big));
+    try testing.expectEqual(@as(S, .{ .a = 0x12, .b = 0x5634 }), try r.peekStructEndian(S, .little));
 }
 
 test takeEnum {
@@ -1519,15 +1552,46 @@ test takeEnum {
 }
 
 test takeLeb128 {
-    return error.Unimplemented;
+    var r: Reader = .fixed("\xc7\x9f\x7f\x80");
+    try testing.expectEqual(-12345, try r.takeLeb128(i64));
+    try testing.expectEqual(0x80, try r.peekByte());
+    try testing.expectError(error.EndOfStream, r.takeLeb128(i64));
 }
 
 test readSliceShort {
-    return error.Unimplemented;
+    var r: Reader = .fixed("HelloFren");
+    var buf: [5]u8 = undefined;
+    try testing.expectEqual(5, try r.readSliceShort(&buf));
+    try testing.expectEqualStrings("Hello", buf[0..5]);
+    try testing.expectEqual(4, try r.readSliceShort(&buf));
+    try testing.expectEqualStrings("Fren", buf[0..4]);
+    try testing.expectEqual(0, try r.readSliceShort(&buf));
 }
 
 test readVec {
-    return error.Unimplemented;
+    var r: Reader = .fixed(std.ascii.letters);
+    var flat_buffer: [52]u8 = undefined;
+    var bufs: [2][]u8 = .{
+        flat_buffer[0..26],
+        flat_buffer[26..],
+    };
+    // Short reads are possible with this function but not with fixed.
+    try testing.expectEqual(26 * 2, try r.readVec(&bufs));
+    try testing.expectEqualStrings(std.ascii.letters[0..26], bufs[0]);
+    try testing.expectEqualStrings(std.ascii.letters[26..], bufs[1]);
+}
+
+test readVecLimit {
+    var r: Reader = .fixed(std.ascii.letters);
+    var flat_buffer: [52]u8 = undefined;
+    var bufs: [2][]u8 = .{
+        flat_buffer[0..26],
+        flat_buffer[26..],
+    };
+    // Short reads are possible with this function but not with fixed.
+    try testing.expectEqual(50, try r.readVecLimit(&bufs, .limited(50)));
+    try testing.expectEqualStrings(std.ascii.letters[0..26], bufs[0]);
+    try testing.expectEqualStrings(std.ascii.letters[26..50], bufs[1][0..24]);
 }
 
 test "expected error.EndOfStream" {
lib/std/ascii.zig
@@ -10,6 +10,10 @@
 
 const std = @import("std");
 
+pub const lowercase = "abcdefghijklmnopqrstuvwxyz";
+pub const uppercase = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
+pub const letters = lowercase ++ uppercase;
+
 /// The C0 control codes of the ASCII encoding.
 ///
 /// See also: https://en.wikipedia.org/wiki/C0_and_C1_control_codes and `isControl`