Skip to content

Commit 460dea1

Browse files
authored
fix: build sha256 hash in a incremental way (#33)
1 parent 8eacae4 commit 460dea1

File tree

5 files changed

+54
-30
lines changed

5 files changed

+54
-30
lines changed

book-src/02-01-sha-digest.md

+20-13
Original file line numberDiff line numberDiff line change
@@ -5,32 +5,39 @@ There are many crypto algorithm implementations in std, `sha256`, `md5` are supp
55
```zig
66
const std = @import("std");
77
const fs = std.fs;
8-
const print = std.debug.print;
98
const Sha256 = std.crypto.hash.sha2.Sha256;
109
10+
// In real world, this may set to page_size, usually it's 4096.
11+
const BUF_SIZE = 16;
12+
1113
fn sha256_digest(
12-
allocator: std.mem.Allocator,
1314
file: fs.File,
1415
) ![Sha256.digest_length]u8 {
15-
const md = try file.metadata();
16-
const size = md.size();
17-
const bytes = try file.reader().readAllAlloc(allocator, size);
18-
defer allocator.free(bytes);
19-
20-
var out: [Sha256.digest_length]u8 = undefined;
21-
Sha256.hash(bytes, &out, .{});
22-
return out;
16+
var sha256 = Sha256.init(.{});
17+
const rdr = file.reader();
18+
19+
var buf: [BUF_SIZE]u8 = undefined;
20+
var n = try rdr.read(&buf);
21+
while (n != 0) {
22+
sha256.update(buf[0..n]);
23+
n = try rdr.read(&buf);
24+
}
25+
26+
return sha256.finalResult();
2327
}
2428
2529
pub fn main() !void {
2630
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
2731
defer _ = gpa.deinit();
2832
const allocator = gpa.allocator();
2933
30-
const file = try fs.cwd().openFile("build.zig", .{});
34+
const file = try fs.cwd().openFile("tests/zig-zen.txt", .{});
3135
defer file.close();
3236
33-
const digest = try sha256_digest(allocator, file);
34-
print("SHA-256 digest is {s}", .{std.fmt.fmtSliceHexLower(&digest)});
37+
const digest = try sha256_digest(file);
38+
const hex_digest = try std.fmt.allocPrint(allocator, "{s}", .{std.fmt.fmtSliceHexLower(&digest)});
39+
defer allocator.free(hex_digest);
40+
41+
try std.testing.expectEqualStrings("2210e9263ece534df0beff39ec06850d127dc60aa17bbc7769c5dc2ea5f3e342", hex_digest);
3542
}
3643
```

src/01-01.zig

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ pub fn main() !void {
77
defer _ = gpa.deinit();
88
const allocator = gpa.allocator();
99

10-
const file = try fs.cwd().openFile("tests/hello-world.txt", .{});
10+
const file = try fs.cwd().openFile("tests/zig-zen.txt", .{});
1111
defer file.close();
1212

1313
const rdr = file.reader();

src/02-01.zig

+20-13
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,37 @@
11
const std = @import("std");
22
const fs = std.fs;
3-
const print = std.debug.print;
43
const Sha256 = std.crypto.hash.sha2.Sha256;
54

5+
// In real world, this may set to page_size, usually it's 4096.
6+
const BUF_SIZE = 16;
7+
68
fn sha256_digest(
7-
allocator: std.mem.Allocator,
89
file: fs.File,
910
) ![Sha256.digest_length]u8 {
10-
const md = try file.metadata();
11-
const size = md.size();
12-
const bytes = try file.reader().readAllAlloc(allocator, size);
13-
defer allocator.free(bytes);
14-
15-
var out: [Sha256.digest_length]u8 = undefined;
16-
Sha256.hash(bytes, &out, .{});
17-
return out;
11+
var sha256 = Sha256.init(.{});
12+
const rdr = file.reader();
13+
14+
var buf: [BUF_SIZE]u8 = undefined;
15+
var n = try rdr.read(&buf);
16+
while (n != 0) {
17+
sha256.update(buf[0..n]);
18+
n = try rdr.read(&buf);
19+
}
20+
21+
return sha256.finalResult();
1822
}
1923

2024
pub fn main() !void {
2125
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
2226
defer _ = gpa.deinit();
2327
const allocator = gpa.allocator();
2428

25-
const file = try fs.cwd().openFile("tests/hello-world.txt", .{});
29+
const file = try fs.cwd().openFile("tests/zig-zen.txt", .{});
2630
defer file.close();
2731

28-
const digest = try sha256_digest(allocator, file);
29-
print("SHA-256 digest is {s}\n", .{std.fmt.fmtSliceHexLower(&digest)});
32+
const digest = try sha256_digest(file);
33+
const hex_digest = try std.fmt.allocPrint(allocator, "{s}", .{std.fmt.fmtSliceHexLower(&digest)});
34+
defer allocator.free(hex_digest);
35+
36+
try std.testing.expectEqualStrings("2210e9263ece534df0beff39ec06850d127dc60aa17bbc7769c5dc2ea5f3e342", hex_digest);
3037
}

tests/hello-world.txt

-3
This file was deleted.

tests/zig-zen.txt

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
* Communicate intent precisely.
2+
* Edge cases matter.
3+
* Favor reading code over writing code.
4+
* Only one obvious way to do things.
5+
* Runtime crashes are better than bugs.
6+
* Compile errors are better than runtime crashes.
7+
* Incremental improvements.
8+
* Avoid local maximums.
9+
* Reduce the amount one must remember.
10+
* Focus on code rather than style.
11+
* Resource allocation may fail; resource deallocation must succeed.
12+
* Memory is a resource.
13+
* Together we serve the users.

0 commit comments

Comments
 (0)