You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
274 lines
5.5 KiB
274 lines
5.5 KiB
'use strict'
|
|
|
|
const { test } = require('tap')
|
|
const { join } = require('path')
|
|
const { readFile } = require('fs')
|
|
const { file } = require('./helper')
|
|
const ThreadStream = require('..')
|
|
const { MessageChannel } = require('worker_threads')
|
|
const { once } = require('events')
|
|
|
|
test('base sync=true', function (t) {
|
|
t.plan(15)
|
|
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: true
|
|
})
|
|
|
|
t.same(stream.writableObjectMode, false)
|
|
|
|
t.same(stream.writableFinished, false)
|
|
stream.on('finish', () => {
|
|
t.same(stream.writableFinished, true)
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data, 'hello world\nsomething else\n')
|
|
})
|
|
})
|
|
|
|
t.same(stream.closed, false)
|
|
stream.on('close', () => {
|
|
t.same(stream.closed, true)
|
|
t.notOk(stream.writable)
|
|
t.pass('close emitted')
|
|
})
|
|
|
|
t.same(stream.writableNeedDrain, false)
|
|
t.ok(stream.write('hello world\n'))
|
|
t.ok(stream.write('something else\n'))
|
|
t.ok(stream.writable)
|
|
|
|
t.same(stream.writableEnded, false)
|
|
stream.end()
|
|
t.same(stream.writableEnded, true)
|
|
})
|
|
|
|
test('overflow sync=true', function (t) {
|
|
t.plan(3)
|
|
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
bufferSize: 128,
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: true
|
|
})
|
|
|
|
let count = 0
|
|
|
|
// Write 10 chars, 20 times
|
|
function write () {
|
|
if (count++ === 20) {
|
|
stream.end()
|
|
return
|
|
}
|
|
|
|
stream.write('aaaaaaaaaa')
|
|
// do not wait for drain event
|
|
setImmediate(write)
|
|
}
|
|
|
|
write()
|
|
|
|
stream.on('finish', () => {
|
|
t.pass('finish emitted')
|
|
})
|
|
|
|
stream.on('close', () => {
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data.length, 200)
|
|
})
|
|
})
|
|
})
|
|
|
|
test('overflow sync=false', function (t) {
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
bufferSize: 128,
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: false
|
|
})
|
|
|
|
let count = 0
|
|
|
|
t.same(stream.writableNeedDrain, false)
|
|
|
|
// Write 10 chars, 20 times
|
|
function write () {
|
|
if (count++ === 20) {
|
|
t.pass('end sent')
|
|
stream.end()
|
|
return
|
|
}
|
|
|
|
if (!stream.write('aaaaaaaaaa')) {
|
|
t.same(stream.writableNeedDrain, true)
|
|
}
|
|
// do not wait for drain event
|
|
setImmediate(write)
|
|
}
|
|
|
|
write()
|
|
|
|
stream.on('drain', () => {
|
|
t.same(stream.writableNeedDrain, false)
|
|
t.pass('drain')
|
|
})
|
|
|
|
stream.on('finish', () => {
|
|
t.pass('finish emitted')
|
|
})
|
|
|
|
stream.on('close', () => {
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data.length, 200)
|
|
t.end()
|
|
})
|
|
})
|
|
})
|
|
|
|
test('over the bufferSize at startup', function (t) {
|
|
t.plan(6)
|
|
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
bufferSize: 10,
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: true
|
|
})
|
|
|
|
stream.on('finish', () => {
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data, 'hello world\nsomething else\n')
|
|
})
|
|
})
|
|
|
|
stream.on('close', () => {
|
|
t.pass('close emitted')
|
|
})
|
|
|
|
t.ok(stream.write('hello'))
|
|
t.ok(stream.write(' world\n'))
|
|
t.ok(stream.write('something else\n'))
|
|
|
|
stream.end()
|
|
})
|
|
|
|
test('over the bufferSize at startup (async)', function (t) {
|
|
t.plan(6)
|
|
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
bufferSize: 10,
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: false
|
|
})
|
|
|
|
t.ok(stream.write('hello'))
|
|
t.notOk(stream.write(' world\n'))
|
|
t.notOk(stream.write('something else\n'))
|
|
|
|
stream.end()
|
|
|
|
stream.on('finish', () => {
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data, 'hello world\nsomething else\n')
|
|
})
|
|
})
|
|
|
|
stream.on('close', () => {
|
|
t.pass('close emitted')
|
|
})
|
|
})
|
|
|
|
test('flushSync sync=false', function (t) {
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
bufferSize: 128,
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: false
|
|
})
|
|
|
|
stream.on('drain', () => {
|
|
t.pass('drain')
|
|
stream.end()
|
|
})
|
|
|
|
stream.on('finish', () => {
|
|
t.pass('finish emitted')
|
|
})
|
|
|
|
stream.on('close', () => {
|
|
readFile(dest, 'utf8', (err, data) => {
|
|
t.error(err)
|
|
t.equal(data.length, 200)
|
|
t.end()
|
|
})
|
|
})
|
|
|
|
for (let count = 0; count < 20; count++) {
|
|
stream.write('aaaaaaaaaa')
|
|
}
|
|
stream.flushSync()
|
|
})
|
|
|
|
test('pass down MessagePorts', async function (t) {
|
|
t.plan(3)
|
|
|
|
const { port1, port2 } = new MessageChannel()
|
|
const stream = new ThreadStream({
|
|
filename: join(__dirname, 'port.js'),
|
|
workerData: { port: port1 },
|
|
workerOpts: {
|
|
transferList: [port1]
|
|
},
|
|
sync: false
|
|
})
|
|
t.teardown(() => {
|
|
stream.end()
|
|
})
|
|
|
|
t.ok(stream.write('hello world\n'))
|
|
t.ok(stream.write('something else\n'))
|
|
|
|
const [strings] = await once(port2, 'message')
|
|
|
|
t.equal(strings, 'hello world\nsomething else\n')
|
|
})
|
|
|
|
test('destroy does not error', function (t) {
|
|
t.plan(5)
|
|
|
|
const dest = file()
|
|
const stream = new ThreadStream({
|
|
filename: join(__dirname, 'to-file.js'),
|
|
workerData: { dest },
|
|
sync: false
|
|
})
|
|
|
|
stream.on('ready', () => {
|
|
t.pass('ready emitted')
|
|
stream.worker.terminate()
|
|
})
|
|
|
|
stream.on('error', (err) => {
|
|
t.equal(err.message, 'the worker thread exited')
|
|
stream.flush((err) => {
|
|
t.equal(err.message, 'the worker has exited')
|
|
})
|
|
t.doesNotThrow(() => stream.flushSync())
|
|
t.doesNotThrow(() => stream.end())
|
|
})
|
|
})
|