2020-03-09 01:55:02 +01:00
|
|
|
const fs = require('fs')
|
|
|
|
const gulp = require('gulp')
|
|
|
|
const watch = require('gulp-watch')
|
2020-11-20 16:26:40 +01:00
|
|
|
const pify = require('pify')
|
|
|
|
const pump = pify(require('pump'))
|
2020-03-09 01:55:02 +01:00
|
|
|
const source = require('vinyl-source-stream')
|
|
|
|
const buffer = require('vinyl-buffer')
|
|
|
|
const log = require('fancy-log')
|
|
|
|
const { assign } = require('lodash')
|
|
|
|
const watchify = require('watchify')
|
|
|
|
const browserify = require('browserify')
|
2020-12-09 06:10:22 +01:00
|
|
|
const envify = require('loose-envify/custom')
|
2020-03-09 01:55:02 +01:00
|
|
|
const sourcemaps = require('gulp-sourcemaps')
|
|
|
|
const sesify = require('sesify')
|
|
|
|
const terser = require('gulp-terser-js')
|
|
|
|
const { makeStringTransform } = require('browserify-transform-tools')
|
|
|
|
|
2020-09-10 18:16:00 +02:00
|
|
|
const conf = require('rc')('metamask', {
|
|
|
|
INFURA_PROJECT_ID: process.env.INFURA_PROJECT_ID,
|
2020-11-09 22:45:23 +01:00
|
|
|
SEGMENT_HOST: process.env.SEGMENT_HOST,
|
2020-09-14 19:04:05 +02:00
|
|
|
SEGMENT_WRITE_KEY: process.env.SEGMENT_WRITE_KEY,
|
2020-10-26 20:05:57 +01:00
|
|
|
SEGMENT_LEGACY_WRITE_KEY: process.env.SEGMENT_LEGACY_WRITE_KEY,
|
2020-09-10 18:16:00 +02:00
|
|
|
})
|
2020-03-09 01:55:02 +01:00
|
|
|
|
2020-12-09 06:10:22 +01:00
|
|
|
const baseManifest = require('../../app/manifest/_base.json')
|
|
|
|
|
2020-03-09 01:55:02 +01:00
|
|
|
const packageJSON = require('../../package.json')
|
2020-11-03 00:41:28 +01:00
|
|
|
const {
|
|
|
|
createTask,
|
|
|
|
composeParallel,
|
|
|
|
composeSeries,
|
|
|
|
runInChildProcess,
|
|
|
|
} = require('./task')
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
module.exports = createScriptTasks
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const dependencies = Object.keys(
|
|
|
|
(packageJSON && packageJSON.dependencies) || {},
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
const materialUIDependencies = ['@material-ui/core']
|
2020-08-14 13:48:42 +02:00
|
|
|
const reactDepenendencies = dependencies.filter((dep) => dep.match(/react/u))
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
const externalDependenciesMap = {
|
2020-11-03 00:41:28 +01:00
|
|
|
background: ['3box'],
|
2020-12-03 00:25:19 +01:00
|
|
|
ui: [...materialUIDependencies, ...reactDepenendencies],
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function createScriptTasks({ browserPlatforms, livereload }) {
|
2020-03-09 01:55:02 +01:00
|
|
|
// internal tasks
|
|
|
|
const core = {
|
|
|
|
// dev tasks (live reload)
|
2020-11-03 00:41:28 +01:00
|
|
|
dev: createTasksForBuildJsExtension({
|
|
|
|
taskPrefix: 'scripts:core:dev',
|
|
|
|
devMode: true,
|
|
|
|
}),
|
|
|
|
testDev: createTasksForBuildJsExtension({
|
|
|
|
taskPrefix: 'scripts:core:test-live',
|
|
|
|
devMode: true,
|
|
|
|
testing: true,
|
|
|
|
}),
|
2020-03-09 01:55:02 +01:00
|
|
|
// built for CI tests
|
2020-11-03 00:41:28 +01:00
|
|
|
test: createTasksForBuildJsExtension({
|
|
|
|
taskPrefix: 'scripts:core:test',
|
|
|
|
testing: true,
|
|
|
|
}),
|
2020-03-09 01:55:02 +01:00
|
|
|
// production
|
|
|
|
prod: createTasksForBuildJsExtension({ taskPrefix: 'scripts:core:prod' }),
|
|
|
|
}
|
|
|
|
const deps = {
|
2020-11-03 00:41:28 +01:00
|
|
|
background: createTasksForBuildJsDeps({
|
|
|
|
filename: 'bg-libs',
|
|
|
|
key: 'background',
|
|
|
|
}),
|
2020-03-09 01:55:02 +01:00
|
|
|
ui: createTasksForBuildJsDeps({ filename: 'ui-libs', key: 'ui' }),
|
|
|
|
}
|
|
|
|
|
|
|
|
// high level tasks
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const prod = composeParallel(deps.background, deps.ui, core.prod)
|
2020-03-09 01:55:02 +01:00
|
|
|
|
2020-08-18 22:06:58 +02:00
|
|
|
const { dev, testDev } = core
|
2020-03-09 01:55:02 +01:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const test = composeParallel(deps.background, deps.ui, core.test)
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
return { prod, dev, testDev, test }
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function createTasksForBuildJsDeps({ key, filename }) {
|
|
|
|
return createTask(
|
|
|
|
`scripts:deps:${key}`,
|
|
|
|
bundleTask({
|
|
|
|
label: filename,
|
|
|
|
filename: `${filename}.js`,
|
|
|
|
buildLib: true,
|
|
|
|
dependenciesToBundle: externalDependenciesMap[key],
|
|
|
|
devMode: false,
|
|
|
|
}),
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function createTasksForBuildJsExtension({ taskPrefix, devMode, testing }) {
|
2020-12-09 06:10:22 +01:00
|
|
|
const standardBundles = [
|
|
|
|
'background',
|
|
|
|
'ui',
|
|
|
|
'phishing-detect',
|
|
|
|
'initSentry',
|
|
|
|
]
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
const standardSubtasks = standardBundles.map((filename) => {
|
2020-11-03 00:41:28 +01:00
|
|
|
return createTask(
|
|
|
|
`${taskPrefix}:${filename}`,
|
|
|
|
createBundleTaskForBuildJsExtensionNormal({
|
|
|
|
filename,
|
|
|
|
devMode,
|
|
|
|
testing,
|
|
|
|
}),
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
})
|
|
|
|
// inpage must be built before contentscript
|
|
|
|
// because inpage bundle result is included inside contentscript
|
2020-11-03 00:41:28 +01:00
|
|
|
const contentscriptSubtask = createTask(
|
|
|
|
`${taskPrefix}:contentscript`,
|
|
|
|
createTaskForBuildJsExtensionContentscript({ devMode, testing }),
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
// task for initiating livereload
|
|
|
|
const initiateLiveReload = async () => {
|
|
|
|
if (devMode) {
|
|
|
|
// trigger live reload when the bundles are updated
|
|
|
|
// this is not ideal, but overcomes the limitations:
|
|
|
|
// - run from the main process (not child process tasks)
|
|
|
|
// - after the first build has completed (thus the timeout)
|
|
|
|
// - build tasks never "complete" when run with livereload + child process
|
|
|
|
setTimeout(() => {
|
|
|
|
watch('./dist/*/*.js', (event) => {
|
|
|
|
livereload.changed(event.path)
|
|
|
|
})
|
|
|
|
}, 75e3)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// make each bundle run in a separate process
|
2020-11-03 00:41:28 +01:00
|
|
|
const allSubtasks = [
|
|
|
|
...standardSubtasks,
|
|
|
|
contentscriptSubtask,
|
|
|
|
].map((subtask) => runInChildProcess(subtask))
|
2020-03-09 01:55:02 +01:00
|
|
|
// const allSubtasks = [...standardSubtasks, contentscriptSubtask].map(subtask => (subtask))
|
|
|
|
// make a parent task that runs each task in a child thread
|
|
|
|
return composeParallel(initiateLiveReload, ...allSubtasks)
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function createBundleTaskForBuildJsExtensionNormal({
|
|
|
|
filename,
|
|
|
|
devMode,
|
|
|
|
testing,
|
|
|
|
}) {
|
2020-03-09 01:55:02 +01:00
|
|
|
return bundleTask({
|
|
|
|
label: filename,
|
|
|
|
filename: `${filename}.js`,
|
|
|
|
filepath: `./app/scripts/${filename}.js`,
|
2020-11-03 00:41:28 +01:00
|
|
|
externalDependencies: devMode
|
|
|
|
? undefined
|
|
|
|
: externalDependenciesMap[filename],
|
2020-03-09 01:55:02 +01:00
|
|
|
devMode,
|
|
|
|
testing,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function createTaskForBuildJsExtensionContentscript({ devMode, testing }) {
|
2020-03-09 01:55:02 +01:00
|
|
|
const inpage = 'inpage'
|
|
|
|
const contentscript = 'contentscript'
|
|
|
|
return composeSeries(
|
|
|
|
bundleTask({
|
|
|
|
label: inpage,
|
|
|
|
filename: `${inpage}.js`,
|
|
|
|
filepath: `./app/scripts/${inpage}.js`,
|
2020-11-03 00:41:28 +01:00
|
|
|
externalDependencies: devMode
|
|
|
|
? undefined
|
|
|
|
: externalDependenciesMap[inpage],
|
2020-03-09 01:55:02 +01:00
|
|
|
devMode,
|
|
|
|
testing,
|
|
|
|
}),
|
|
|
|
bundleTask({
|
|
|
|
label: contentscript,
|
|
|
|
filename: `${contentscript}.js`,
|
|
|
|
filepath: `./app/scripts/${contentscript}.js`,
|
2020-11-03 00:41:28 +01:00
|
|
|
externalDependencies: devMode
|
|
|
|
? undefined
|
|
|
|
: externalDependenciesMap[contentscript],
|
2020-03-09 01:55:02 +01:00
|
|
|
devMode,
|
|
|
|
testing,
|
2020-07-14 17:20:41 +02:00
|
|
|
}),
|
2020-03-09 01:55:02 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function bundleTask(opts) {
|
2020-03-09 01:55:02 +01:00
|
|
|
let bundler
|
|
|
|
|
|
|
|
return performBundle
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
async function performBundle() {
|
2020-03-09 01:55:02 +01:00
|
|
|
// initialize bundler if not available yet
|
|
|
|
// dont create bundler until task is actually run
|
|
|
|
if (!bundler) {
|
|
|
|
bundler = generateBundler(opts, performBundle)
|
|
|
|
// output build logs to terminal
|
|
|
|
bundler.on('log', log)
|
|
|
|
}
|
|
|
|
|
2020-11-20 16:26:40 +01:00
|
|
|
const buildPipeline = [
|
|
|
|
bundler.bundle(),
|
2020-03-09 01:55:02 +01:00
|
|
|
// convert bundle stream to gulp vinyl stream
|
2020-11-20 16:26:40 +01:00
|
|
|
source(opts.filename),
|
|
|
|
// Initialize Source Maps
|
|
|
|
buffer(),
|
2020-03-09 01:55:02 +01:00
|
|
|
// loads map from browserify file
|
2020-11-20 16:26:40 +01:00
|
|
|
sourcemaps.init({ loadMaps: true }),
|
|
|
|
]
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
// Minification
|
|
|
|
if (!opts.devMode) {
|
2020-11-20 16:26:40 +01:00
|
|
|
buildPipeline.push(
|
2020-11-03 00:41:28 +01:00
|
|
|
terser({
|
2020-03-09 01:55:02 +01:00
|
|
|
mangle: {
|
2020-08-19 18:27:05 +02:00
|
|
|
reserved: ['MetamaskInpageProvider'],
|
2020-03-09 01:55:02 +01:00
|
|
|
},
|
2020-07-29 22:31:01 +02:00
|
|
|
sourceMap: {
|
|
|
|
content: true,
|
|
|
|
},
|
2020-11-03 00:41:28 +01:00
|
|
|
}),
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Finalize Source Maps
|
|
|
|
if (opts.devMode) {
|
|
|
|
// Use inline source maps for development due to Chrome DevTools bug
|
|
|
|
// https://bugs.chromium.org/p/chromium/issues/detail?id=931675
|
2020-11-20 16:26:40 +01:00
|
|
|
// note: sourcemaps call arity is important
|
|
|
|
buildPipeline.push(sourcemaps.write())
|
2020-03-09 01:55:02 +01:00
|
|
|
} else {
|
2020-11-20 16:26:40 +01:00
|
|
|
buildPipeline.push(sourcemaps.write('../sourcemaps'))
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// write completed bundles
|
|
|
|
browserPlatforms.forEach((platform) => {
|
|
|
|
const dest = `./dist/${platform}`
|
2020-11-20 16:26:40 +01:00
|
|
|
buildPipeline.push(gulp.dest(dest))
|
2020-03-09 01:55:02 +01:00
|
|
|
})
|
|
|
|
|
2020-11-20 16:26:40 +01:00
|
|
|
// process bundles
|
|
|
|
if (opts.devMode) {
|
|
|
|
try {
|
|
|
|
await pump(buildPipeline)
|
|
|
|
} catch (err) {
|
|
|
|
gracefulError(err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
await pump(buildPipeline)
|
|
|
|
}
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function configureBundleForSesify({ browserifyOpts, bundleName }) {
|
2020-03-09 01:55:02 +01:00
|
|
|
// add in sesify args for better globalRef usage detection
|
|
|
|
Object.assign(browserifyOpts, sesify.args)
|
|
|
|
|
|
|
|
// ensure browserify uses full paths
|
|
|
|
browserifyOpts.fullPaths = true
|
|
|
|
|
|
|
|
// record dependencies used in bundle
|
|
|
|
fs.mkdirSync('./sesify', { recursive: true })
|
2020-11-03 00:41:28 +01:00
|
|
|
browserifyOpts.plugin.push([
|
|
|
|
'deps-dump',
|
|
|
|
{
|
|
|
|
filename: `./sesify/deps-${bundleName}.json`,
|
|
|
|
},
|
|
|
|
])
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
const sesifyConfigPath = `./sesify/${bundleName}.json`
|
|
|
|
|
|
|
|
// add sesify plugin
|
2020-11-03 00:41:28 +01:00
|
|
|
browserifyOpts.plugin.push([
|
|
|
|
sesify,
|
|
|
|
{
|
|
|
|
writeAutoConfig: sesifyConfigPath,
|
|
|
|
},
|
|
|
|
])
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
// remove html comments that SES is alergic to
|
2020-11-03 00:41:28 +01:00
|
|
|
const removeHtmlComment = makeStringTransform(
|
|
|
|
'remove-html-comment',
|
|
|
|
{ excludeExtension: ['.json'] },
|
|
|
|
(content, _, cb) => {
|
|
|
|
const result = content.split('-->').join('-- >')
|
|
|
|
cb(null, result)
|
|
|
|
},
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
browserifyOpts.transform.push([removeHtmlComment, { global: true }])
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function generateBundler(opts, performBundle) {
|
2020-03-09 01:55:02 +01:00
|
|
|
const browserifyOpts = assign({}, watchify.args, {
|
|
|
|
plugin: [],
|
|
|
|
transform: [],
|
|
|
|
debug: true,
|
|
|
|
fullPaths: opts.devMode,
|
|
|
|
})
|
|
|
|
|
|
|
|
const bundleName = opts.filename.split('.')[0]
|
|
|
|
|
|
|
|
// activate sesify
|
|
|
|
const activateAutoConfig = Boolean(process.env.SESIFY_AUTOGEN)
|
|
|
|
// const activateSesify = activateAutoConfig
|
2020-11-03 00:41:28 +01:00
|
|
|
const activateSesify =
|
|
|
|
activateAutoConfig && ['background'].includes(bundleName)
|
2020-03-09 01:55:02 +01:00
|
|
|
if (activateSesify) {
|
|
|
|
configureBundleForSesify({ browserifyOpts, bundleName })
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!activateSesify) {
|
|
|
|
browserifyOpts.plugin.push('browserify-derequire')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!opts.buildLib) {
|
|
|
|
if (opts.devMode && opts.filename === 'ui.js') {
|
2020-11-03 00:41:28 +01:00
|
|
|
browserifyOpts.entries = [
|
|
|
|
'./development/require-react-devtools.js',
|
|
|
|
opts.filepath,
|
|
|
|
]
|
2020-03-09 01:55:02 +01:00
|
|
|
} else {
|
2020-08-19 18:27:05 +02:00
|
|
|
browserifyOpts.entries = [opts.filepath]
|
2020-03-09 01:55:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let bundler = browserify(browserifyOpts)
|
|
|
|
.transform('babelify')
|
|
|
|
.transform('brfs')
|
|
|
|
|
|
|
|
if (opts.buildLib) {
|
|
|
|
bundler = bundler.require(opts.dependenciesToBundle)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opts.externalDependencies) {
|
|
|
|
bundler = bundler.external(opts.externalDependencies)
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const environment = getEnvironment({
|
|
|
|
devMode: opts.devMode,
|
|
|
|
test: opts.testing,
|
|
|
|
})
|
2020-07-29 18:14:08 +02:00
|
|
|
if (environment === 'production' && !process.env.SENTRY_DSN) {
|
|
|
|
throw new Error('Missing SENTRY_DSN environment variable')
|
|
|
|
}
|
|
|
|
|
2020-03-09 01:55:02 +01:00
|
|
|
// Inject variables into bundle
|
2020-11-03 00:41:28 +01:00
|
|
|
bundler.transform(
|
|
|
|
envify({
|
|
|
|
METAMASK_DEBUG: opts.devMode,
|
|
|
|
METAMASK_ENVIRONMENT: environment,
|
2020-12-09 06:10:22 +01:00
|
|
|
METAMASK_VERSION: baseManifest.version,
|
2020-11-03 00:41:28 +01:00
|
|
|
METAMETRICS_PROJECT_ID: process.env.METAMETRICS_PROJECT_ID,
|
|
|
|
NODE_ENV: opts.devMode ? 'development' : 'production',
|
|
|
|
IN_TEST: opts.testing ? 'true' : false,
|
|
|
|
PUBNUB_SUB_KEY: process.env.PUBNUB_SUB_KEY || '',
|
|
|
|
PUBNUB_PUB_KEY: process.env.PUBNUB_PUB_KEY || '',
|
|
|
|
ETH_GAS_STATION_API_KEY: process.env.ETH_GAS_STATION_API_KEY || '',
|
|
|
|
CONF: opts.devMode ? conf : {},
|
|
|
|
SENTRY_DSN: process.env.SENTRY_DSN,
|
|
|
|
INFURA_PROJECT_ID: opts.testing
|
2020-09-10 18:16:00 +02:00
|
|
|
? '00000000000000000000000000000000'
|
2020-11-03 00:41:28 +01:00
|
|
|
: conf.INFURA_PROJECT_ID,
|
2020-11-09 22:45:23 +01:00
|
|
|
SEGMENT_HOST: conf.SEGMENT_HOST,
|
2020-11-03 21:41:24 +01:00
|
|
|
// When we're in the 'production' environment we will use a specific key only set in CI
|
|
|
|
// Otherwise we'll use the key from .metamaskrc or from the environment variable. If
|
|
|
|
// the value of SEGMENT_WRITE_KEY that we envify is undefined then no events will be tracked
|
|
|
|
// in the build. This is intentional so that developers can contribute to MetaMask without
|
|
|
|
// inflating event volume.
|
2020-11-03 00:41:28 +01:00
|
|
|
SEGMENT_WRITE_KEY:
|
|
|
|
environment === 'production'
|
2020-11-03 21:41:24 +01:00
|
|
|
? process.env.SEGMENT_PROD_WRITE_KEY
|
|
|
|
: conf.SEGMENT_WRITE_KEY,
|
2020-11-03 00:41:28 +01:00
|
|
|
SEGMENT_LEGACY_WRITE_KEY:
|
|
|
|
environment === 'production'
|
2020-11-03 21:41:24 +01:00
|
|
|
? process.env.SEGMENT_PROD_LEGACY_WRITE_KEY
|
|
|
|
: conf.SEGMENT_LEGACY_WRITE_KEY,
|
2020-11-03 00:41:28 +01:00
|
|
|
}),
|
|
|
|
{
|
|
|
|
global: true,
|
|
|
|
},
|
|
|
|
)
|
2020-03-09 01:55:02 +01:00
|
|
|
|
|
|
|
// Live reload - minimal rebundle on change
|
|
|
|
if (opts.devMode) {
|
|
|
|
bundler = watchify(bundler)
|
|
|
|
// on any file update, re-runs the bundler
|
|
|
|
bundler.on('update', () => {
|
|
|
|
performBundle()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return bundler
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
function getEnvironment({ devMode, test }) {
|
2020-07-30 21:19:42 +02:00
|
|
|
// get environment slug
|
|
|
|
if (devMode) {
|
|
|
|
return 'development'
|
|
|
|
} else if (test) {
|
|
|
|
return 'testing'
|
|
|
|
} else if (process.env.CIRCLE_BRANCH === 'master') {
|
|
|
|
return 'production'
|
2020-11-03 00:41:28 +01:00
|
|
|
} else if (
|
|
|
|
/^Version-v(\d+)[.](\d+)[.](\d+)/u.test(process.env.CIRCLE_BRANCH)
|
|
|
|
) {
|
2020-07-30 21:19:42 +02:00
|
|
|
return 'release-candidate'
|
|
|
|
} else if (process.env.CIRCLE_BRANCH === 'develop') {
|
|
|
|
return 'staging'
|
|
|
|
} else if (process.env.CIRCLE_PULL_REQUEST) {
|
|
|
|
return 'pull-request'
|
|
|
|
}
|
2020-08-19 18:27:05 +02:00
|
|
|
return 'other'
|
2020-07-30 21:19:42 +02:00
|
|
|
}
|
2020-11-20 16:26:40 +01:00
|
|
|
|
|
|
|
function beep() {
|
|
|
|
process.stdout.write('\x07')
|
|
|
|
}
|
|
|
|
|
|
|
|
function gracefulError(err) {
|
|
|
|
console.warn(err)
|
|
|
|
beep()
|
|
|
|
}
|