|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
'use strict'; |
|
|
|
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); |
|
|
|
const promises = require('node:fs/promises'); |
|
const process$2 = require('node:process'); |
|
const index = require('./index.js'); |
|
const cli = require('../bin/rollup'); |
|
const rollup = require('./rollup.js'); |
|
const parseAst_js = require('./parseAst.js'); |
|
const loadConfigFile_js = require('./loadConfigFile.js'); |
|
const node_child_process = require('node:child_process'); |
|
const rollup_js = require('../rollup.js'); |
|
require('fs'); |
|
require('util'); |
|
require('stream'); |
|
require('path'); |
|
require('os'); |
|
require('./fsevents-importer.js'); |
|
require('events'); |
|
require('node:path'); |
|
require('tty'); |
|
require('../native.js'); |
|
require('node:perf_hooks'); |
|
require('node:url'); |
|
require('../getLogFilter.js'); |
|
|
|
function timeZone(date = new Date()) { |
|
const offset = date.getTimezoneOffset(); |
|
const absOffset = Math.abs(offset); |
|
const hours = Math.floor(absOffset / 60); |
|
const minutes = absOffset % 60; |
|
const minutesOut = minutes > 0 ? ':' + ('0' + minutes).slice(-2) : ''; |
|
return (offset < 0 ? '+' : '-') + hours + minutesOut; |
|
} |
|
|
|
function dateTime(options = {}) { |
|
let { |
|
date = new Date(), |
|
local = true, |
|
showTimeZone = false, |
|
showMilliseconds = false |
|
} = options; |
|
|
|
if (local) { |
|
|
|
date = new Date(date.getTime() - (date.getTimezoneOffset() * 60000)); |
|
} |
|
|
|
let end = ''; |
|
|
|
if (showTimeZone) { |
|
end = ' UTC' + (local ? timeZone(date) : ''); |
|
} |
|
|
|
if (showMilliseconds && date.getUTCMilliseconds() > 0) { |
|
end = ` ${date.getUTCMilliseconds()}ms${end}`; |
|
} |
|
|
|
return date |
|
.toISOString() |
|
.replace(/T/, ' ') |
|
.replace(/\..+/, end); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const signals = []; |
|
signals.push('SIGHUP', 'SIGINT', 'SIGTERM'); |
|
if (process.platform !== 'win32') { |
|
signals.push('SIGALRM', 'SIGABRT', 'SIGVTALRM', 'SIGXCPU', 'SIGXFSZ', 'SIGUSR2', 'SIGTRAP', 'SIGSYS', 'SIGQUIT', 'SIGIOT' |
|
|
|
|
|
|
|
); |
|
} |
|
if (process.platform === 'linux') { |
|
signals.push('SIGIO', 'SIGPOLL', 'SIGPWR', 'SIGSTKFLT'); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
const processOk = (process) => !!process && |
|
typeof process === 'object' && |
|
typeof process.removeListener === 'function' && |
|
typeof process.emit === 'function' && |
|
typeof process.reallyExit === 'function' && |
|
typeof process.listeners === 'function' && |
|
typeof process.kill === 'function' && |
|
typeof process.pid === 'number' && |
|
typeof process.on === 'function'; |
|
const kExitEmitter = Symbol.for('signal-exit emitter'); |
|
const global = globalThis; |
|
const ObjectDefineProperty = Object.defineProperty.bind(Object); |
|
|
|
class Emitter { |
|
emitted = { |
|
afterExit: false, |
|
exit: false, |
|
}; |
|
listeners = { |
|
afterExit: [], |
|
exit: [], |
|
}; |
|
count = 0; |
|
id = Math.random(); |
|
constructor() { |
|
if (global[kExitEmitter]) { |
|
return global[kExitEmitter]; |
|
} |
|
ObjectDefineProperty(global, kExitEmitter, { |
|
value: this, |
|
writable: false, |
|
enumerable: false, |
|
configurable: false, |
|
}); |
|
} |
|
on(ev, fn) { |
|
this.listeners[ev].push(fn); |
|
} |
|
removeListener(ev, fn) { |
|
const list = this.listeners[ev]; |
|
const i = list.indexOf(fn); |
|
|
|
if (i === -1) { |
|
return; |
|
} |
|
|
|
if (i === 0 && list.length === 1) { |
|
list.length = 0; |
|
} |
|
else { |
|
list.splice(i, 1); |
|
} |
|
} |
|
emit(ev, code, signal) { |
|
if (this.emitted[ev]) { |
|
return false; |
|
} |
|
this.emitted[ev] = true; |
|
let ret = false; |
|
for (const fn of this.listeners[ev]) { |
|
ret = fn(code, signal) === true || ret; |
|
} |
|
if (ev === 'exit') { |
|
ret = this.emit('afterExit', code, signal) || ret; |
|
} |
|
return ret; |
|
} |
|
} |
|
class SignalExitBase { |
|
} |
|
const signalExitWrap = (handler) => { |
|
return { |
|
onExit(cb, opts) { |
|
return handler.onExit(cb, opts); |
|
}, |
|
load() { |
|
return handler.load(); |
|
}, |
|
unload() { |
|
return handler.unload(); |
|
}, |
|
}; |
|
}; |
|
class SignalExitFallback extends SignalExitBase { |
|
onExit() { |
|
return () => { }; |
|
} |
|
load() { } |
|
unload() { } |
|
} |
|
class SignalExit extends SignalExitBase { |
|
|
|
|
|
|
|
#hupSig = process$1.platform === 'win32' ? 'SIGINT' : 'SIGHUP'; |
|
|
|
#emitter = new Emitter(); |
|
#process; |
|
#originalProcessEmit; |
|
#originalProcessReallyExit; |
|
#sigListeners = {}; |
|
#loaded = false; |
|
constructor(process) { |
|
super(); |
|
this.#process = process; |
|
|
|
this.#sigListeners = {}; |
|
for (const sig of signals) { |
|
this.#sigListeners[sig] = () => { |
|
|
|
|
|
|
|
|
|
const listeners = this.#process.listeners(sig); |
|
let { count } = this.#emitter; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const p = process; |
|
if (typeof p.__signal_exit_emitter__ === 'object' && |
|
typeof p.__signal_exit_emitter__.count === 'number') { |
|
count += p.__signal_exit_emitter__.count; |
|
} |
|
|
|
if (listeners.length === count) { |
|
this.unload(); |
|
const ret = this.#emitter.emit('exit', null, sig); |
|
|
|
const s = sig === 'SIGHUP' ? this.#hupSig : sig; |
|
if (!ret) |
|
process.kill(process.pid, s); |
|
|
|
} |
|
}; |
|
} |
|
this.#originalProcessReallyExit = process.reallyExit; |
|
this.#originalProcessEmit = process.emit; |
|
} |
|
onExit(cb, opts) { |
|
|
|
if (!processOk(this.#process)) { |
|
return () => { }; |
|
} |
|
|
|
if (this.#loaded === false) { |
|
this.load(); |
|
} |
|
const ev = opts?.alwaysLast ? 'afterExit' : 'exit'; |
|
this.#emitter.on(ev, cb); |
|
return () => { |
|
this.#emitter.removeListener(ev, cb); |
|
if (this.#emitter.listeners['exit'].length === 0 && |
|
this.#emitter.listeners['afterExit'].length === 0) { |
|
this.unload(); |
|
} |
|
}; |
|
} |
|
load() { |
|
if (this.#loaded) { |
|
return; |
|
} |
|
this.#loaded = true; |
|
|
|
|
|
|
|
|
|
this.#emitter.count += 1; |
|
for (const sig of signals) { |
|
try { |
|
const fn = this.#sigListeners[sig]; |
|
if (fn) |
|
this.#process.on(sig, fn); |
|
} |
|
catch (_) { } |
|
} |
|
this.#process.emit = (ev, ...a) => { |
|
return this.#processEmit(ev, ...a); |
|
}; |
|
this.#process.reallyExit = (code) => { |
|
return this.#processReallyExit(code); |
|
}; |
|
} |
|
unload() { |
|
if (!this.#loaded) { |
|
return; |
|
} |
|
this.#loaded = false; |
|
signals.forEach(sig => { |
|
const listener = this.#sigListeners[sig]; |
|
|
|
if (!listener) { |
|
throw new Error('Listener not defined for signal: ' + sig); |
|
} |
|
|
|
try { |
|
this.#process.removeListener(sig, listener); |
|
|
|
} |
|
catch (_) { } |
|
|
|
}); |
|
this.#process.emit = this.#originalProcessEmit; |
|
this.#process.reallyExit = this.#originalProcessReallyExit; |
|
this.#emitter.count -= 1; |
|
} |
|
#processReallyExit(code) { |
|
|
|
if (!processOk(this.#process)) { |
|
return 0; |
|
} |
|
this.#process.exitCode = code || 0; |
|
|
|
this.#emitter.emit('exit', this.#process.exitCode, null); |
|
return this.#originalProcessReallyExit.call(this.#process, this.#process.exitCode); |
|
} |
|
#processEmit(ev, ...args) { |
|
const og = this.#originalProcessEmit; |
|
if (ev === 'exit' && processOk(this.#process)) { |
|
if (typeof args[0] === 'number') { |
|
this.#process.exitCode = args[0]; |
|
|
|
} |
|
|
|
const ret = og.call(this.#process, ev, ...args); |
|
|
|
this.#emitter.emit('exit', this.#process.exitCode, null); |
|
|
|
return ret; |
|
} |
|
else { |
|
return og.call(this.#process, ev, ...args); |
|
} |
|
} |
|
} |
|
const process$1 = globalThis.process; |
|
|
|
|
|
const { |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
onExit, |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
load, |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unload, } = signalExitWrap(processOk(process$1) ? new SignalExit(process$1) : new SignalExitFallback()); |
|
|
|
const CLEAR_SCREEN = '\u001Bc'; |
|
function getResetScreen(configs, allowClearScreen) { |
|
let clearScreen = allowClearScreen; |
|
for (const config of configs) { |
|
if (config.watch && config.watch.clearScreen === false) { |
|
clearScreen = false; |
|
} |
|
} |
|
if (clearScreen) { |
|
return (heading) => rollup.stderr(CLEAR_SCREEN + heading); |
|
} |
|
let firstRun = true; |
|
return (heading) => { |
|
if (firstRun) { |
|
rollup.stderr(heading); |
|
firstRun = false; |
|
} |
|
}; |
|
} |
|
|
|
function extractWatchHooks(command) { |
|
if (!Array.isArray(command.watch)) |
|
return {}; |
|
return command.watch |
|
.filter(value => typeof value === 'object') |
|
.reduce((accumulator, keyValueOption) => ({ ...accumulator, ...keyValueOption }), {}); |
|
} |
|
function createWatchHooks(command) { |
|
const watchHooks = extractWatchHooks(command); |
|
return function (hook) { |
|
if (watchHooks[hook]) { |
|
const cmd = watchHooks[hook]; |
|
if (!command.silent) { |
|
rollup.stderr(rollup.cyan$1(`watch.${hook} ${rollup.bold(`$ ${cmd}`)}`)); |
|
} |
|
try { |
|
|
|
const stdio = [process.stdin, process.stderr, process.stderr]; |
|
node_child_process.execSync(cmd, { stdio: command.silent ? 'ignore' : stdio }); |
|
} |
|
catch (error) { |
|
rollup.stderr(error.message); |
|
} |
|
} |
|
}; |
|
} |
|
|
|
async function watch(command) { |
|
process$2.env.ROLLUP_WATCH = 'true'; |
|
const isTTY = process$2.stderr.isTTY; |
|
const silent = command.silent; |
|
let watcher; |
|
let configWatcher; |
|
let resetScreen; |
|
const configFile = command.config ? await cli.getConfigPath(command.config) : null; |
|
const runWatchHook = createWatchHooks(command); |
|
onExit(close); |
|
process$2.on('uncaughtException', closeWithError); |
|
if (!process$2.stdin.isTTY) { |
|
process$2.stdin.on('end', close); |
|
process$2.stdin.resume(); |
|
} |
|
async function loadConfigFromFileAndTrack(configFile) { |
|
let configFileData = null; |
|
let configFileRevision = 0; |
|
configWatcher = index.chokidar.watch(configFile).on('change', reloadConfigFile); |
|
await reloadConfigFile(); |
|
async function reloadConfigFile() { |
|
try { |
|
const newConfigFileData = await promises.readFile(configFile, 'utf8'); |
|
if (newConfigFileData === configFileData) { |
|
return; |
|
} |
|
configFileRevision++; |
|
const currentConfigFileRevision = configFileRevision; |
|
if (configFileData) { |
|
rollup.stderr(`\nReloading updated config...`); |
|
} |
|
configFileData = newConfigFileData; |
|
const { options, warnings } = await loadConfigFile_js.loadConfigFile(configFile, command, true); |
|
if (currentConfigFileRevision !== configFileRevision) { |
|
return; |
|
} |
|
if (watcher) { |
|
await watcher.close(); |
|
} |
|
start(options, warnings); |
|
} |
|
catch (error) { |
|
rollup.handleError(error, true); |
|
} |
|
} |
|
} |
|
if (configFile) { |
|
await loadConfigFromFileAndTrack(configFile); |
|
} |
|
else { |
|
const { options, warnings } = await cli.loadConfigFromCommand(command, true); |
|
await start(options, warnings); |
|
} |
|
async function start(configs, warnings) { |
|
watcher = rollup_js.watch(configs); |
|
watcher.on('event', event => { |
|
switch (event.code) { |
|
case 'ERROR': { |
|
warnings.flush(); |
|
rollup.handleError(event.error, true); |
|
runWatchHook('onError'); |
|
break; |
|
} |
|
case 'START': { |
|
if (!silent) { |
|
if (!resetScreen) { |
|
resetScreen = getResetScreen(configs, isTTY); |
|
} |
|
resetScreen(rollup.underline(`rollup v${rollup.version}`)); |
|
} |
|
runWatchHook('onStart'); |
|
break; |
|
} |
|
case 'BUNDLE_START': { |
|
if (!silent) { |
|
let input = event.input; |
|
if (typeof input !== 'string') { |
|
input = Array.isArray(input) |
|
? input.join(', ') |
|
: Object.values(input).join(', '); |
|
} |
|
rollup.stderr(rollup.cyan$1(`bundles ${rollup.bold(input)} → ${rollup.bold(event.output.map(parseAst_js.relativeId).join(', '))}...`)); |
|
} |
|
runWatchHook('onBundleStart'); |
|
break; |
|
} |
|
case 'BUNDLE_END': { |
|
warnings.flush(); |
|
if (!silent) |
|
rollup.stderr(rollup.green(`created ${rollup.bold(event.output.map(parseAst_js.relativeId).join(', '))} in ${rollup.bold(cli.prettyMilliseconds(event.duration))}`)); |
|
runWatchHook('onBundleEnd'); |
|
if (event.result && event.result.getTimings) { |
|
cli.printTimings(event.result.getTimings()); |
|
} |
|
break; |
|
} |
|
case 'END': { |
|
runWatchHook('onEnd'); |
|
if (!silent && isTTY) { |
|
rollup.stderr(`\n[${dateTime()}] waiting for changes...`); |
|
} |
|
} |
|
} |
|
if ('result' in event && event.result) { |
|
event.result.close().catch(error => rollup.handleError(error, true)); |
|
} |
|
}); |
|
} |
|
async function close(code) { |
|
process$2.removeListener('uncaughtException', closeWithError); |
|
|
|
process$2.stdin.removeListener('end', close); |
|
if (watcher) |
|
await watcher.close(); |
|
if (configWatcher) |
|
configWatcher.close(); |
|
if (code) |
|
process$2.exit(code); |
|
} |
|
|
|
return new Promise(() => { }); |
|
} |
|
function closeWithError(error) { |
|
error.name = `Uncaught ${error.name}`; |
|
rollup.handleError(error); |
|
} |
|
|
|
exports.watch = watch; |
|
|
|
|