Function expectEqualSlices [src]

This function is intended to be used only in tests. When the two slices are not equal, prints diagnostics to stderr to show exactly how they are not equal (with the differences highlighted in red), then returns a test failure error. The colorized output is optional and controlled by the return of std.io.tty.detectConfig(). If your inputs are UTF-8 encoded strings, consider calling expectEqualStrings instead.

Prototype

pub fn expectEqualSlices(comptime T: type, expected: []const T, actual: []const T) !void

Parameters

T: typeexpected: []const Tactual: []const T

Source

pub fn expectEqualSlices(comptime T: type, expected: []const T, actual: []const T) !void { if (expected.ptr == actual.ptr and expected.len == actual.len) { return; } const diff_index: usize = diff_index: { const shortest = @min(expected.len, actual.len); var index: usize = 0; while (index < shortest) : (index += 1) { if (!std.meta.eql(actual[index], expected[index])) break :diff_index index; } break :diff_index if (expected.len == actual.len) return else shortest; }; if (!backend_can_print) { return error.TestExpectedEqual; } print("slices differ. first difference occurs at index {d} (0x{X})\n", .{ diff_index, diff_index }); // TODO: Should this be configurable by the caller? const max_lines: usize = 16; const max_window_size: usize = if (T == u8) max_lines * 16 else max_lines; // Print a maximum of max_window_size items of each input, starting just before the // first difference to give a bit of context. var window_start: usize = 0; if (@max(actual.len, expected.len) > max_window_size) { const alignment = if (T == u8) 16 else 2; window_start = std.mem.alignBackward(usize, diff_index - @min(diff_index, alignment), alignment); } const expected_window = expected[window_start..@min(expected.len, window_start + max_window_size)]; const expected_truncated = window_start + expected_window.len < expected.len; const actual_window = actual[window_start..@min(actual.len, window_start + max_window_size)]; const actual_truncated = window_start + actual_window.len < actual.len; const stderr = std.io.getStdErr(); const ttyconf = std.io.tty.detectConfig(stderr); var differ = if (T == u8) BytesDiffer{ .expected = expected_window, .actual = actual_window, .ttyconf = ttyconf, } else SliceDiffer(T){ .start_index = window_start, .expected = expected_window, .actual = actual_window, .ttyconf = ttyconf, }; // Print indexes as hex for slices of u8 since it's more likely to be binary data where // that is usually useful. const index_fmt = if (T == u8) "0x{X}" else "{}"; print("\n============ expected this output: ============= len: {} (0x{X})\n\n", .{ expected.len, expected.len }); if (window_start > 0) { if (T == u8) { print("... truncated, start index: " ++ index_fmt ++ " ...\n", .{window_start}); } else { print("... truncated ...\n", .{}); } } differ.write(stderr.writer()) catch {}; if (expected_truncated) { const end_offset = window_start + expected_window.len; const num_missing_items = expected.len - (window_start + expected_window.len); if (T == u8) { print("... truncated, indexes [" ++ index_fmt ++ "..] not shown, remaining bytes: " ++ index_fmt ++ " ...\n", .{ end_offset, num_missing_items }); } else { print("... truncated, remaining items: " ++ index_fmt ++ " ...\n", .{num_missing_items}); } } // now reverse expected/actual and print again differ.expected = actual_window; differ.actual = expected_window; print("\n============= instead found this: ============== len: {} (0x{X})\n\n", .{ actual.len, actual.len }); if (window_start > 0) { if (T == u8) { print("... truncated, start index: " ++ index_fmt ++ " ...\n", .{window_start}); } else { print("... truncated ...\n", .{}); } } differ.write(stderr.writer()) catch {}; if (actual_truncated) { const end_offset = window_start + actual_window.len; const num_missing_items = actual.len - (window_start + actual_window.len); if (T == u8) { print("... truncated, indexes [" ++ index_fmt ++ "..] not shown, remaining bytes: " ++ index_fmt ++ " ...\n", .{ end_offset, num_missing_items }); } else { print("... truncated, remaining items: " ++ index_fmt ++ " ...\n", .{num_missing_items}); } } print("\n================================================\n\n", .{}); return error.TestExpectedEqual; }