毕设专用git仓库
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.
 
 
 
 
 
 

128 lines
4.9 KiB

"use strict";
// Copyright 2021 Google LLC. Use of this source code is governed by an
// MIT-style license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT.
Object.defineProperty(exports, "__esModule", { value: true });
exports.SyncChildProcess = void 0;
const fs = require("fs");
const p = require("path");
const stream = require("stream");
const worker_threads_1 = require("worker_threads");
const worker_threads = require("worker_threads");
const sync_message_port_1 = require("sync-message-port");
/** Whether {@link object} can't be transferred between threads, only cloned. */
function isMarkedAsUntransferable(object) {
// TODO: Remove this check when we no longer support Node v20 (after
// 2026-04-30).
return 'isMarkedAsUntransferable' in worker_threads
? // TODO - DefinitelyTyped/DefinitelyTyped#71033: Remove this cast
worker_threads.isMarkedAsUntransferable(object)
: false;
}
/**
* A child process that runs synchronously while also allowing the user to
* interact with it before it shuts down.
*/
class SyncChildProcess {
/** The port that communicates with the worker thread. */
port;
/** The worker in which the child process runs. */
worker;
/** The standard input stream to write to the process. */
stdin;
constructor(command, argsOrOptions, options) {
let args;
if (Array.isArray(argsOrOptions)) {
args = argsOrOptions;
}
else {
args = [];
options = argsOrOptions;
}
const { port1, port2 } = sync_message_port_1.SyncMessagePort.createChannel();
this.port = new sync_message_port_1.SyncMessagePort(port1);
this.worker = spawnWorker(p.join(p.dirname(__filename), 'worker'), {
workerData: { port: port2, command, args, options },
transferList: [port2],
});
// The worker shouldn't emit any errors unless it breaks in development.
this.worker.on('error', console.error);
this.stdin = new stream.Writable({
write: (chunk, encoding, callback) => {
this.port.postMessage({
type: 'stdin',
data: chunk,
}, isMarkedAsUntransferable(chunk.buffer) ? undefined : [chunk.buffer]);
callback();
},
final: () => this.port.postMessage({ type: 'stdinClosed' }),
});
}
/**
* Blocks until the child process is ready to emit another event, then returns
* that event. This will return an [IteratorReturnResult] with an [ExitEvent]
* once when the process exits. If it's called again after that, it will
* return `{done: true}` without a value.
*
* If there's an error running the child process, this will throw that error.
*/
next() {
if (this.stdin.destroyed)
return { done: true, value: undefined };
const message = this.port.receiveMessage();
switch (message.type) {
case 'stdout':
return {
value: { type: 'stdout', data: Buffer.from(message.data.buffer) },
};
case 'stderr':
return {
value: { type: 'stderr', data: Buffer.from(message.data.buffer) },
};
case 'error':
this.close();
throw message.error;
case 'exit':
this.close();
return { done: true, value: message };
}
}
// TODO(nex3): Add a non-blocking `yieldIfReady()` function that returns
// `null` if the worker hasn't queued up an event.
// TODO(nex3): Add a `yieldAsync()` function that returns a `Promise<Event>`.
/**
* Sends a signal (`SIGTERM` by default) to the child process.
*
* This has no effect if the process has already exited.
*/
kill(signal) {
this.port.postMessage({ type: 'kill', signal });
}
/** Closes down the worker thread and the stdin stream. */
close() {
this.port.close();
void this.worker.terminate();
this.stdin.destroy();
}
}
exports.SyncChildProcess = SyncChildProcess;
/**
* Spawns a worker for the given `fileWithoutExtension` in either a JS or TS
* worker, depending on which file exists.
*/
function spawnWorker(fileWithoutExtension, options) {
// The released version always spawns the JS worker. The TS worker is only
// used for development.
const jsFile = fileWithoutExtension + '.js';
if (fs.existsSync(jsFile))
return new worker_threads_1.Worker(jsFile, options);
const tsFile = fileWithoutExtension + '.ts';
if (fs.existsSync(tsFile)) {
return new worker_threads_1.Worker(`
require('ts-node').register();
require(${JSON.stringify(tsFile)});
`, { ...options, eval: true });
}
throw new Error(`Neither "${jsFile}" nor ".ts" exists.`);
}
//# sourceMappingURL=index.js.map