mirror of
https://github.com/nodejs/node.git
synced 2024-11-21 10:59:27 +00:00
97c29def52
We previously used a text that appears to be an excerpt of https://zh.wikipedia.org/wiki/%E5%8D%97%E8%B6%8A%E5%9B%BD and can have copyright/license complications. It may also include some geopolitical nuances. The text has been repeated through out the code base without much reuse. This patch consolidates the fixtures by adding a common helper string as `fixtures.utf8TestText` which is identical to a copy in test/fixtures/utf8_test_text.txt. It also updates the text to a copy of 蘭亭集序, It was chosen because: 1. It's a well-known Chinese classical piece written in 353 CE and therefore in public domain. The string is copied from https://zh.wikisource.org/zh-hant/%E8%98%AD%E4%BA%AD%E9%9B%86%E5%BA%8F which contains a disclaimer of copyright for this reason. 2. The text is in suitable length for general UTF8 string read/write tests (including punctuations, 389 code points and 1167 bytes). 3. This is also commonly used as reference text for Chinese text layout tests. 4. It's a timeless and harmless preface for a collection of poems, written by a uncontroversial figure who passed away >1600 years ago and contains no geopolitical nuances. Background and an English translation of this text can be found at https://en.wikipedia.org/wiki/Lantingji_Xu PR-URL: https://github.com/nodejs/node/pull/50732 Reviewed-By: Yagiz Nizipli <yagiz.nizipli@sentry.io>
188 lines
5.9 KiB
JavaScript
188 lines
5.9 KiB
JavaScript
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
const common = require('../common');
|
|
const assert = require('assert');
|
|
const fs = require('fs');
|
|
|
|
const tmpdir = require('../common/tmpdir');
|
|
|
|
const currentFileData = 'ABCD';
|
|
const fixtures = require('../common/fixtures');
|
|
const s = fixtures.utf8TestText;
|
|
|
|
tmpdir.refresh();
|
|
|
|
const throwNextTick = (e) => { process.nextTick(() => { throw e; }); };
|
|
|
|
// Test that empty file will be created and have content added (callback API).
|
|
{
|
|
const filename = tmpdir.resolve('append.txt');
|
|
|
|
fs.appendFile(filename, s, common.mustSucceed(() => {
|
|
fs.readFile(filename, common.mustSucceed((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s), buffer.length);
|
|
}));
|
|
}));
|
|
}
|
|
|
|
// Test that empty file will be created and have content added (promise API).
|
|
{
|
|
const filename = tmpdir.resolve('append-promise.txt');
|
|
|
|
fs.promises.appendFile(filename, s)
|
|
.then(common.mustCall(() => fs.promises.readFile(filename)))
|
|
.then((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s), buffer.length);
|
|
})
|
|
.catch(throwNextTick);
|
|
}
|
|
|
|
// Test that appends data to a non-empty file (callback API).
|
|
{
|
|
const filename = tmpdir.resolve('append-non-empty.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
fs.appendFile(filename, s, common.mustSucceed(() => {
|
|
fs.readFile(filename, common.mustSucceed((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s) + currentFileData.length,
|
|
buffer.length);
|
|
}));
|
|
}));
|
|
}
|
|
|
|
// Test that appends data to a non-empty file (promise API).
|
|
{
|
|
const filename = tmpdir.resolve('append-non-empty-promise.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
fs.promises.appendFile(filename, s)
|
|
.then(common.mustCall(() => fs.promises.readFile(filename)))
|
|
.then((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s) + currentFileData.length,
|
|
buffer.length);
|
|
})
|
|
.catch(throwNextTick);
|
|
}
|
|
|
|
// Test that appendFile accepts buffers (callback API).
|
|
{
|
|
const filename = tmpdir.resolve('append-buffer.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
const buf = Buffer.from(s, 'utf8');
|
|
|
|
fs.appendFile(filename, buf, common.mustSucceed(() => {
|
|
fs.readFile(filename, common.mustSucceed((buffer) => {
|
|
assert.strictEqual(buf.length + currentFileData.length, buffer.length);
|
|
}));
|
|
}));
|
|
}
|
|
|
|
// Test that appendFile accepts buffers (promises API).
|
|
{
|
|
const filename = tmpdir.resolve('append-buffer-promises.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
const buf = Buffer.from(s, 'utf8');
|
|
|
|
fs.promises.appendFile(filename, buf)
|
|
.then(common.mustCall(() => fs.promises.readFile(filename)))
|
|
.then((buffer) => {
|
|
assert.strictEqual(buf.length + currentFileData.length, buffer.length);
|
|
})
|
|
.catch(throwNextTick);
|
|
}
|
|
|
|
// Test that appendFile does not accept invalid data type (callback API).
|
|
[false, 5, {}, null, undefined].forEach(async (data) => {
|
|
const errObj = {
|
|
code: 'ERR_INVALID_ARG_TYPE',
|
|
message: /"data"|"buffer"/
|
|
};
|
|
const filename = tmpdir.resolve('append-invalid-data.txt');
|
|
|
|
assert.throws(
|
|
() => fs.appendFile(filename, data, common.mustNotCall()),
|
|
errObj
|
|
);
|
|
|
|
assert.throws(
|
|
() => fs.appendFileSync(filename, data),
|
|
errObj
|
|
);
|
|
|
|
await assert.rejects(
|
|
fs.promises.appendFile(filename, data),
|
|
errObj
|
|
);
|
|
// The filename shouldn't exist if throwing error.
|
|
assert.throws(
|
|
() => fs.statSync(filename),
|
|
{
|
|
code: 'ENOENT',
|
|
message: /no such file or directory/
|
|
}
|
|
);
|
|
});
|
|
|
|
// Test that appendFile accepts file descriptors (callback API).
|
|
{
|
|
const filename = tmpdir.resolve('append-descriptors.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
fs.open(filename, 'a+', common.mustSucceed((fd) => {
|
|
fs.appendFile(fd, s, common.mustSucceed(() => {
|
|
fs.close(fd, common.mustSucceed(() => {
|
|
fs.readFile(filename, common.mustSucceed((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s) + currentFileData.length,
|
|
buffer.length);
|
|
}));
|
|
}));
|
|
}));
|
|
}));
|
|
}
|
|
|
|
// Test that appendFile accepts file descriptors (promises API).
|
|
{
|
|
const filename = tmpdir.resolve('append-descriptors-promises.txt');
|
|
fs.writeFileSync(filename, currentFileData);
|
|
|
|
let fd;
|
|
fs.promises.open(filename, 'a+')
|
|
.then(common.mustCall((fileDescriptor) => {
|
|
fd = fileDescriptor;
|
|
return fs.promises.appendFile(fd, s);
|
|
}))
|
|
.then(common.mustCall(() => fd.close()))
|
|
.then(common.mustCall(() => fs.promises.readFile(filename)))
|
|
.then(common.mustCall((buffer) => {
|
|
assert.strictEqual(Buffer.byteLength(s) + currentFileData.length,
|
|
buffer.length);
|
|
}))
|
|
.catch(throwNextTick);
|
|
}
|
|
|
|
assert.throws(
|
|
() => fs.appendFile(tmpdir.resolve('append6.txt'), console.log),
|
|
{ code: 'ERR_INVALID_ARG_TYPE' });
|