371 lines
13 KiB
JavaScript
371 lines
13 KiB
JavaScript
'use strict';
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('empty_blob', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write(new Blob([]));
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '');
|
|
assert_equals(await getFileSize(handle), 0);
|
|
}, 'write() with an empty blob to an empty file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('valid_blob', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write(new Blob(['1234567890']));
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() a blob to an empty file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('write_param_empty', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write({type: 'write', data: '1234567890'});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() with WriteParams without position to an empty file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('string_zero_offset', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write({type: 'write', position: 0, data: '1234567890'});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() a string to an empty file with zero offset');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('blob_zero_offset', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write(
|
|
{type: 'write', position: 0, data: new Blob(['1234567890'])});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() a blob to an empty file with zero offset');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('write_appends', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('12345');
|
|
await stream.write('67890');
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() called consecutively appends');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('write_appends_object_string', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('12345');
|
|
await stream.write({type: 'write', data: '67890'});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() WriteParams without position and string appends');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('write_appends_object_blob', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('12345');
|
|
await stream.write({type: 'write', data: new Blob(['67890'])});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234567890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() WriteParams without position and blob appends');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('string_with_offset', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('1234567890');
|
|
await stream.write({type: 'write', position: 4, data: 'abc'});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234abc890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() called with a string and a valid offset');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle =
|
|
await createEmptyFile('write_string_with_offset_after_seek', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('1234567890');
|
|
await stream.write({type: 'seek', position: 0});
|
|
await stream.write({type: 'write', position: 4, data: 'abc'});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234abc890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() called with a string and a valid offset after seek');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('blob_with_offset', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('1234567890');
|
|
await stream.write({type: 'write', position: 4, data: new Blob(['abc'])});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '1234abc890');
|
|
assert_equals(await getFileSize(handle), 10);
|
|
}, 'write() called with a blob and a valid offset');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('bad_offset', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write({type: 'write', position: 4, data: new Blob(['abc'])});
|
|
await stream.close();
|
|
|
|
assert_equals(await getFileContents(handle), '\0\0\0\0abc');
|
|
assert_equals(await getFileSize(handle), 7);
|
|
}, 'write() called with an offset beyond the end of the file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('empty_string', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('');
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), '');
|
|
assert_equals(await getFileSize(handle), 0);
|
|
}, 'write() with an empty string to an empty file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('valid_utf8_string', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('foo🤘');
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo🤘');
|
|
assert_equals(await getFileSize(handle), 7);
|
|
}, 'write() with a valid utf-8 string');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('string_with_unix_line_ending', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('foo\n');
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo\n');
|
|
assert_equals(await getFileSize(handle), 4);
|
|
}, 'write() with a string with unix line ending preserved');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('string_with_windows_line_ending', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await stream.write('foo\r\n');
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo\r\n');
|
|
assert_equals(await getFileSize(handle), 5);
|
|
}, 'write() with a string with windows line ending preserved');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('empty_array_buffer', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
const buf = new ArrayBuffer(0);
|
|
await stream.write(buf);
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), '');
|
|
assert_equals(await getFileSize(handle), 0);
|
|
}, 'write() with an empty array buffer to an empty file');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('valid_string_typed_byte_array', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
const buf = new ArrayBuffer(3);
|
|
const intView = new Uint8Array(buf);
|
|
intView[0] = 0x66;
|
|
intView[1] = 0x6f;
|
|
intView[2] = 0x6f;
|
|
await stream.write(buf);
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo');
|
|
assert_equals(await getFileSize(handle), 3);
|
|
}, 'write() with a valid typed array buffer');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('atomic_writes.txt', root);
|
|
const stream = await handle.createWritable();
|
|
await stream.write('foox');
|
|
|
|
const stream2 = await cleanup_writable(t, await handle.createWritable());
|
|
await stream2.write('bar');
|
|
|
|
assert_equals(await getFileSize(handle), 0);
|
|
|
|
await stream2.close();
|
|
assert_equals(await getFileContents(handle), 'bar');
|
|
assert_equals(await getFileSize(handle), 3);
|
|
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foox');
|
|
assert_equals(await getFileSize(handle), 4);
|
|
}, 'atomic writes: writable file streams make atomic changes on close');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('atomic_write_after_close.txt', root);
|
|
const stream = await handle.createWritable();
|
|
await stream.write('foo');
|
|
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo');
|
|
assert_equals(await getFileSize(handle), 3);
|
|
|
|
await promise_rejects_js(t, TypeError, stream.write('abc'));
|
|
}, 'atomic writes: write() after close() fails');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('atomic_truncate_after_close.txt', root);
|
|
const stream = await handle.createWritable();
|
|
await stream.write('foo');
|
|
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo');
|
|
assert_equals(await getFileSize(handle), 3);
|
|
|
|
await promise_rejects_js(t, TypeError, stream.truncate(0));
|
|
}, 'atomic writes: truncate() after close() fails');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('atomic_close_after_close.txt', root);
|
|
const stream = await handle.createWritable();
|
|
await stream.write('foo');
|
|
|
|
await stream.close();
|
|
assert_equals(await getFileContents(handle), 'foo');
|
|
assert_equals(await getFileSize(handle), 3);
|
|
|
|
await promise_rejects_js(t, TypeError, stream.close());
|
|
}, 'atomic writes: close() after close() fails');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('there_can_be_only_one.txt', root);
|
|
const stream = await handle.createWritable();
|
|
await stream.write('foo');
|
|
|
|
// This test might be flaky if there is a race condition allowing
|
|
// close() to be called multiple times.
|
|
const success_promises =
|
|
[...Array(100)].map(() => stream.close().then(() => 1).catch(() => 0));
|
|
const close_attempts = await Promise.all(success_promises);
|
|
const success_count = close_attempts.reduce((x, y) => x + y);
|
|
assert_equals(success_count, 1);
|
|
}, 'atomic writes: only one close() operation may succeed');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('writer_written', root);
|
|
const stream = await handle.createWritable();
|
|
assert_false(stream.locked);
|
|
const writer = stream.getWriter();
|
|
assert_true(stream.locked);
|
|
|
|
await writer.write('foo');
|
|
await writer.write(new Blob(['bar']));
|
|
await writer.write({type: 'seek', position: 0});
|
|
await writer.write({type: 'write', data: 'baz'});
|
|
await writer.close();
|
|
|
|
assert_equals(await getFileContents(handle), 'bazbar');
|
|
assert_equals(await getFileSize(handle), 6);
|
|
}, 'getWriter() can be used');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle =
|
|
await createFileWithContents('content.txt', 'very long string', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await promise_rejects_dom(
|
|
t, 'SyntaxError', stream.write({type: 'truncate'}),
|
|
'truncate without size');
|
|
}, 'WriteParams: truncate missing size param');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('content.txt', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await promise_rejects_dom(
|
|
t, 'SyntaxError', stream.write({type: 'write'}), 'write without data');
|
|
}, 'WriteParams: write missing data param');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createEmptyFile('content.txt', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await promise_rejects_js(
|
|
t, TypeError, stream.write({type: 'write', data: null}),
|
|
'write with null data');
|
|
}, 'WriteParams: write null data param');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createFileWithContents('content.txt', 'seekable', root);
|
|
const stream = await handle.createWritable();
|
|
|
|
await promise_rejects_dom(
|
|
t, 'SyntaxError', stream.write({type: 'seek'}), 'seek without position');
|
|
}, 'WriteParams: seek missing position param');
|
|
|
|
directory_test(async (t, root) => {
|
|
const source_file =
|
|
await createFileWithContents('source_file', 'source data', root);
|
|
const source_blob = await source_file.getFile();
|
|
await root.removeEntry(source_file.name);
|
|
|
|
const handle = await createEmptyFile('invalid_blob_test', root);
|
|
const stream = await handle.createWritable();
|
|
await promise_rejects_dom(t, "NotFoundError", stream.write(source_blob));
|
|
await promise_rejects_js(t, TypeError, stream.close());
|
|
|
|
assert_equals(await getFileContents(handle), '');
|
|
assert_equals(await getFileSize(handle), 0);
|
|
}, 'write() with an invalid blob to an empty file should reject');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createFileWithContents('file.txt', 'contents', root);
|
|
const stream = await handle.createWritable({mode: 'exclusive'});
|
|
|
|
await stream.write('12345');
|
|
await promise_rejects_js(
|
|
t, TypeError, stream.write({type: 'write', data: null}),
|
|
'write with null data');
|
|
|
|
// The file contents should not have been changed.
|
|
assert_equals(await getFileContents(handle), 'contents');
|
|
|
|
// The file's lock was released.
|
|
const newStream = await handle.createWritable({mode: 'exclusive'});
|
|
await newStream.close();
|
|
}, 'an errored writable stream releases its lock');
|
|
|
|
directory_test(async (t, root) => {
|
|
const handle = await createFileWithContents('file.txt', 'contents', root);
|
|
const stream = await handle.createWritable({mode: 'exclusive'});
|
|
|
|
const writer = stream.getWriter();
|
|
|
|
await promise_rejects_js(t, TypeError, writer.write(null), 'write with null data');
|
|
await promise_rejects_js(t, TypeError, writer.write("foo"), 'write with text data');
|
|
}, 'an errored writable stream should reject the next write call');
|