2021-03-04 12:33:54 +01:00
|
|
|
import os
|
|
|
|
|
|
|
|
struct Point {
|
|
|
|
x f64
|
|
|
|
y f64
|
|
|
|
z f64
|
|
|
|
}
|
|
|
|
|
2021-03-06 19:44:53 +01:00
|
|
|
struct Extended_Point {
|
|
|
|
a f64
|
|
|
|
b f64
|
|
|
|
c f64
|
|
|
|
d f64
|
|
|
|
e f64
|
|
|
|
f f64
|
|
|
|
g f64
|
|
|
|
h f64
|
|
|
|
i f64
|
|
|
|
}
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
enum Color {
|
|
|
|
red
|
|
|
|
green
|
|
|
|
blue
|
|
|
|
}
|
2021-03-04 12:33:54 +01:00
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
[flag]
|
|
|
|
enum Permissions {
|
|
|
|
read
|
|
|
|
write
|
|
|
|
execute
|
|
|
|
}
|
2021-03-06 19:44:53 +01:00
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
const (
|
|
|
|
unit_point = Point{1.0, 1.0, 1.0}
|
|
|
|
another_point = Point{0.25, 2.25, 6.25}
|
|
|
|
extended_point = Extended_Point{1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0}
|
2022-04-15 13:58:56 +02:00
|
|
|
another_byte = u8(123)
|
2021-03-10 17:45:12 +01:00
|
|
|
another_color = Color.red
|
|
|
|
another_permission = Permissions.read | .write
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2021-11-22 13:40:55 +01:00
|
|
|
tfolder = os.join_path_single(os.temp_dir(), 'os_file_test')
|
|
|
|
tfile = os.join_path_single(tfolder, 'test_file')
|
2021-03-10 17:45:12 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
fn testsuite_begin() ? {
|
2021-03-06 20:04:51 +01:00
|
|
|
os.rmdir_all(tfolder) or {}
|
2021-03-04 12:33:54 +01:00
|
|
|
assert !os.is_dir(tfolder)
|
2021-03-10 17:45:12 +01:00
|
|
|
os.mkdir_all(tfolder) ?
|
2021-08-28 09:23:01 +02:00
|
|
|
os.chdir(tfolder) ?
|
2021-03-04 12:33:54 +01:00
|
|
|
assert os.is_dir(tfolder)
|
|
|
|
}
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
fn testsuite_end() ? {
|
2021-08-28 09:23:01 +02:00
|
|
|
os.chdir(os.wd_at_startup) ?
|
2021-03-10 17:45:12 +01:00
|
|
|
os.rmdir_all(tfolder) ?
|
2021-03-04 12:33:54 +01:00
|
|
|
assert !os.is_dir(tfolder)
|
|
|
|
}
|
|
|
|
|
2021-05-21 12:18:08 +02:00
|
|
|
// test_read_bytes_into_newline_text tests reading text from a file with newlines.
|
|
|
|
// This test simulates reading a larger text file step by step into a buffer and
|
|
|
|
// returning on each newline, even before the buffer is full, and reaching EOF before
|
|
|
|
// the buffer is completely filled.
|
|
|
|
fn test_read_bytes_into_newline_text() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_string('Hello World!\nGood\r morning.') ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
mut buf := []u8{len: 8}
|
2021-05-21 12:18:08 +02:00
|
|
|
|
|
|
|
n0 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n0 == 8
|
|
|
|
|
|
|
|
n1 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n1 == 5
|
|
|
|
|
|
|
|
n2 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n2 == 8
|
|
|
|
|
|
|
|
n3 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n3 == 6
|
|
|
|
|
|
|
|
f.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// test_read_bytes_into_newline_binary tests reading a binary file with NUL bytes.
|
|
|
|
// This test simulates the scenario when a byte stream is read and a newline byte
|
|
|
|
// appears in that stream and an EOF occurs before the buffer is full.
|
|
|
|
fn test_read_bytes_into_newline_binary() ? {
|
|
|
|
os.rm(tfile) or {} // FIXME This is a workaround for macos, because the file isn't truncated when open with 'w'
|
2022-04-15 14:35:35 +02:00
|
|
|
mut bw := []u8{len: 15}
|
2021-05-21 12:18:08 +02:00
|
|
|
bw[9] = 0xff
|
|
|
|
bw[12] = 10 // newline
|
|
|
|
|
|
|
|
n0_bytes := bw[0..10]
|
|
|
|
n1_bytes := bw[10..13]
|
|
|
|
n2_bytes := bw[13..]
|
|
|
|
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write(bw) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
mut buf := []u8{len: 10}
|
2021-05-21 12:18:08 +02:00
|
|
|
|
|
|
|
n0 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n0 == 10
|
|
|
|
assert buf[..n0] == n0_bytes
|
|
|
|
|
|
|
|
n1 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n1 == 3
|
|
|
|
assert buf[..n1] == n1_bytes
|
|
|
|
|
|
|
|
n2 := f.read_bytes_into_newline(mut buf) ?
|
|
|
|
assert n2 == 2
|
|
|
|
assert buf[..n2] == n2_bytes
|
|
|
|
f.close()
|
|
|
|
}
|
|
|
|
|
2021-05-13 08:48:55 +02:00
|
|
|
// test_read_eof_last_read_partial_buffer_fill tests that when reading a file
|
|
|
|
// the end-of-file is detected and results in a none error being returned. This
|
|
|
|
// test simulates file reading where the end-of-file is reached inside an fread
|
|
|
|
// containing data.
|
|
|
|
fn test_read_eof_last_read_partial_buffer_fill() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
bw := []u8{len: 199, init: 5}
|
2021-05-13 08:48:55 +02:00
|
|
|
f.write(bw) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
mut br := []u8{len: 100}
|
2021-05-13 08:48:55 +02:00
|
|
|
// Read first 100 bytes of 199 byte file, should fill buffer with no error.
|
|
|
|
n0 := f.read(mut br) ?
|
|
|
|
assert n0 == 100
|
|
|
|
// Read remaining 99 bytes of 199 byte file, should fill buffer with no
|
|
|
|
// error, even though end-of-file was reached.
|
|
|
|
n1 := f.read(mut br) ?
|
|
|
|
assert n1 == 99
|
|
|
|
// Read again, end-of-file was previously reached so should return none
|
|
|
|
// error.
|
|
|
|
if _ := f.read(mut br) {
|
|
|
|
// This is not intended behavior because the read function should
|
|
|
|
// not return a number of bytes read when end-of-file is reached.
|
|
|
|
assert false
|
|
|
|
} else {
|
|
|
|
// Expect none to have been returned when end-of-file.
|
|
|
|
assert err is none
|
|
|
|
}
|
|
|
|
f.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// test_read_eof_last_read_full_buffer_fill tests that when reading a file the
|
|
|
|
// end-of-file is detected and results in a none error being returned. This test
|
|
|
|
// simulates file reading where the end-of-file is reached at the beinning of an
|
|
|
|
// fread that returns no data.
|
|
|
|
fn test_read_eof_last_read_full_buffer_fill() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
bw := []u8{len: 200, init: 5}
|
2021-05-13 08:48:55 +02:00
|
|
|
f.write(bw) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2022-04-15 14:35:35 +02:00
|
|
|
mut br := []u8{len: 100}
|
2021-05-13 08:48:55 +02:00
|
|
|
// Read first 100 bytes of 200 byte file, should fill buffer with no error.
|
|
|
|
n0 := f.read(mut br) ?
|
|
|
|
assert n0 == 100
|
|
|
|
// Read remaining 100 bytes of 200 byte file, should fill buffer with no
|
|
|
|
// error. The end-of-file isn't reached yet, but there is no more data.
|
|
|
|
n1 := f.read(mut br) ?
|
|
|
|
assert n1 == 100
|
|
|
|
// Read again, end-of-file was previously reached so should return none
|
|
|
|
// error.
|
|
|
|
if _ := f.read(mut br) {
|
|
|
|
// This is not intended behavior because the read function should
|
|
|
|
// not return a number of bytes read when end-of-file is reached.
|
|
|
|
assert false
|
|
|
|
} else {
|
|
|
|
// Expect none to have been returned when end-of-file.
|
|
|
|
assert err is none
|
|
|
|
}
|
|
|
|
f.close()
|
|
|
|
}
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
fn test_write_struct() ? {
|
|
|
|
os.rm(tfile) or {} // FIXME This is a workaround for macos, because the file isn't truncated when open with 'w'
|
2021-03-04 12:33:54 +01:00
|
|
|
size_of_point := int(sizeof(Point))
|
2021-03-10 17:45:12 +01:00
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_struct(another_point) ?
|
2021-03-04 12:33:54 +01:00
|
|
|
f.close()
|
2021-03-10 17:45:12 +01:00
|
|
|
x := os.read_file(tfile) ?
|
2022-04-15 13:58:56 +02:00
|
|
|
pcopy := unsafe { &u8(memdup(&another_point, size_of_point)) }
|
2021-04-05 19:47:32 +02:00
|
|
|
y := unsafe { pcopy.vstring_with_len(size_of_point) }
|
2021-03-04 12:33:54 +01:00
|
|
|
assert x == y
|
|
|
|
$if debug {
|
|
|
|
eprintln(x.bytes())
|
|
|
|
eprintln(y.bytes())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
fn test_write_struct_at() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_struct(extended_point) ?
|
|
|
|
f.write_struct_at(another_point, 3) ?
|
|
|
|
f.close()
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
|
|
|
mut p := Point{}
|
|
|
|
f.read_struct_at(mut p, 3) ?
|
2021-03-04 12:33:54 +01:00
|
|
|
f.close()
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
assert p == another_point
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_read_struct() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_struct(another_point) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2021-03-04 12:33:54 +01:00
|
|
|
mut p := Point{}
|
2021-03-10 17:45:12 +01:00
|
|
|
f.read_struct(mut p) ?
|
2021-03-04 12:33:54 +01:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
assert p == another_point
|
|
|
|
}
|
2021-03-06 19:44:53 +01:00
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
fn test_read_struct_at() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
2022-04-15 13:58:56 +02:00
|
|
|
f.write([u8(1), 2, 3]) ?
|
2021-03-10 17:45:12 +01:00
|
|
|
f.write_struct(another_point) ?
|
2021-03-06 19:44:53 +01:00
|
|
|
f.close()
|
2021-03-10 17:45:12 +01:00
|
|
|
f = os.open_file(tfile, 'r') ?
|
2021-03-06 19:44:53 +01:00
|
|
|
mut p := Point{}
|
2021-03-10 17:45:12 +01:00
|
|
|
f.read_struct_at(mut p, 3) ?
|
2021-03-06 19:44:53 +01:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
assert p == another_point
|
|
|
|
}
|
|
|
|
|
2021-03-10 17:45:12 +01:00
|
|
|
fn test_write_raw() ? {
|
|
|
|
os.rm(tfile) or {} // FIXME This is a workaround for macos, because the file isn't truncated when open with 'w'
|
|
|
|
size_of_point := int(sizeof(Point))
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_raw(another_point) ?
|
|
|
|
f.close()
|
|
|
|
x := os.read_file(tfile) ?
|
2022-04-15 13:58:56 +02:00
|
|
|
pcopy := unsafe { &u8(memdup(&another_point, size_of_point)) }
|
2021-04-05 19:47:32 +02:00
|
|
|
y := unsafe { pcopy.vstring_with_len(size_of_point) }
|
2021-03-10 17:45:12 +01:00
|
|
|
assert x == y
|
|
|
|
$if debug {
|
|
|
|
eprintln(x.bytes())
|
|
|
|
eprintln(y.bytes())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_write_raw_at() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_raw(extended_point) ?
|
|
|
|
f.write_raw_at(another_point, 3) ?
|
2021-03-06 19:44:53 +01:00
|
|
|
f.close()
|
2021-03-10 17:45:12 +01:00
|
|
|
f = os.open_file(tfile, 'r') ?
|
2021-03-06 19:44:53 +01:00
|
|
|
mut p := Point{}
|
2021-03-10 17:45:12 +01:00
|
|
|
f.read_struct_at(mut p, 3) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
assert p == another_point
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_write_raw_at_negative_pos() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
if _ := f.write_raw_at(another_point, -1) {
|
|
|
|
assert false
|
|
|
|
}
|
2022-02-11 14:52:33 +01:00
|
|
|
f.write_raw_at(another_point, -234) or { assert err.msg() == 'Invalid argument' }
|
2021-03-10 17:45:12 +01:00
|
|
|
f.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_read_raw() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_raw(another_point) ?
|
|
|
|
f.write_raw(another_byte) ?
|
|
|
|
f.write_raw(another_color) ?
|
|
|
|
f.write_raw(another_permission) ?
|
|
|
|
f.close()
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
|
|
|
p := f.read_raw<Point>() ?
|
|
|
|
b := f.read_raw<byte>() ?
|
|
|
|
c := f.read_raw<Color>() ?
|
|
|
|
x := f.read_raw<Permissions>() ?
|
2021-03-06 19:44:53 +01:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
assert p == another_point
|
2021-03-10 17:45:12 +01:00
|
|
|
assert b == another_byte
|
|
|
|
assert c == another_color
|
|
|
|
assert x == another_permission
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_read_raw_at() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
2022-04-15 13:58:56 +02:00
|
|
|
f.write([u8(1), 2, 3]) ?
|
2021-03-10 17:45:12 +01:00
|
|
|
f.write_raw(another_point) ?
|
|
|
|
f.write_raw(another_byte) ?
|
|
|
|
f.write_raw(another_color) ?
|
|
|
|
f.write_raw(another_permission) ?
|
|
|
|
f.close()
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2021-07-05 09:14:00 +02:00
|
|
|
mut at := u64(3)
|
2021-03-10 17:45:12 +01:00
|
|
|
p := f.read_raw_at<Point>(at) ?
|
2021-07-05 09:14:00 +02:00
|
|
|
at += sizeof(Point)
|
2021-03-10 17:45:12 +01:00
|
|
|
b := f.read_raw_at<byte>(at) ?
|
2021-07-05 09:14:00 +02:00
|
|
|
at += sizeof(byte)
|
2021-03-10 17:45:12 +01:00
|
|
|
c := f.read_raw_at<Color>(at) ?
|
2021-07-05 09:14:00 +02:00
|
|
|
at += sizeof(Color)
|
2021-03-10 17:45:12 +01:00
|
|
|
x := f.read_raw_at<Permissions>(at) ?
|
2021-07-05 09:14:00 +02:00
|
|
|
at += sizeof(Permissions)
|
2021-03-10 17:45:12 +01:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
assert p == another_point
|
|
|
|
assert b == another_byte
|
|
|
|
assert c == another_color
|
|
|
|
assert x == another_permission
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_read_raw_at_negative_pos() ? {
|
|
|
|
mut f := os.open_file(tfile, 'r') ?
|
|
|
|
if _ := f.read_raw_at<Point>(-1) {
|
|
|
|
assert false
|
|
|
|
}
|
2022-02-11 14:52:33 +01:00
|
|
|
f.read_raw_at<Point>(-234) or { assert err.msg() == 'Invalid argument' }
|
2021-03-10 17:45:12 +01:00
|
|
|
f.close()
|
2021-03-06 19:44:53 +01:00
|
|
|
}
|
2021-07-22 06:46:21 +02:00
|
|
|
|
|
|
|
fn test_seek() ? {
|
|
|
|
mut f := os.open_file(tfile, 'w') ?
|
|
|
|
f.write_raw(another_point) ?
|
|
|
|
f.write_raw(another_byte) ?
|
|
|
|
f.write_raw(another_color) ?
|
|
|
|
f.write_raw(another_permission) ?
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
// println('> ${sizeof(Point)} ${sizeof(byte)} ${sizeof(Color)} ${sizeof(Permissions)}')
|
|
|
|
f = os.open_file(tfile, 'r') ?
|
2021-07-22 15:25:23 +02:00
|
|
|
//
|
2021-07-22 06:46:21 +02:00
|
|
|
f.seek(i64(sizeof(Point)), .start) ?
|
|
|
|
assert f.tell() ? == sizeof(Point)
|
|
|
|
b := f.read_raw<byte>() ?
|
2021-07-22 15:25:23 +02:00
|
|
|
assert b == another_byte
|
2021-07-22 06:46:21 +02:00
|
|
|
|
|
|
|
f.seek(i64(sizeof(Color)), .current) ?
|
|
|
|
x := f.read_raw<Permissions>() ?
|
2021-07-22 15:25:23 +02:00
|
|
|
assert x == another_permission
|
|
|
|
//
|
|
|
|
f.close()
|
|
|
|
}
|
2021-07-22 06:46:21 +02:00
|
|
|
|
2021-07-22 15:25:23 +02:00
|
|
|
fn test_tell() ? {
|
|
|
|
for size in 10 .. 30 {
|
|
|
|
s := 'x'.repeat(size)
|
|
|
|
os.write_file(tfile, s) ?
|
|
|
|
fs := os.file_size(tfile)
|
2021-07-22 17:47:23 +02:00
|
|
|
assert int(fs) == size
|
2021-07-22 15:25:23 +02:00
|
|
|
//
|
|
|
|
mut f := os.open_file(tfile, 'r') ?
|
|
|
|
f.seek(-5, .end) ?
|
|
|
|
pos := f.tell() ?
|
|
|
|
f.close()
|
|
|
|
// dump(pos)
|
|
|
|
assert pos == size - 5
|
2021-07-22 12:28:00 +02:00
|
|
|
}
|
2021-07-22 06:46:21 +02:00
|
|
|
}
|