UNPKG

node-remote-repl

Version:

Node.js remote code execution via inspect protocol

120 lines (119 loc) 4.83 kB
#!/usr/bin/env node "use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const path_1 = __importDefault(require("path")); const fs_1 = __importDefault(require("fs")); const commander_1 = require("commander"); const chrome_remote_interface_1 = __importDefault(require("chrome-remote-interface")); const package_json_1 = __importDefault(require("./package.json")); commander_1.program .version(package_json_1.default.version) .name('node-remote-repl') .usage('[options] <file>') .option('-h, --host <string>', 'host of inspect protocol', 'localhost') .option('-p, --port <number>', 'port of inspect protocol', Number, 9229) .parse(process.argv); const file = commander_1.program.args[0]; if (!file) { console.error('Need to pass <file> argument'); process.exit(1); } const fileExtension = file.match(/(\.js|\.jsx|\.ts|\.tsx|\.mjs|\.cjs|\.mts|\.cts)$/)?.[1]; if (!fileExtension) { console.error('Support only js or ts file extensions'); process.exit(1); } const filePath = path_1.default.resolve(process.cwd(), file); if (!fs_1.default.existsSync(filePath)) { console.error(`File on path: "${filePath}" not exists`); process.exit(1); } const fileContent = fs_1.default.readFileSync(filePath, { encoding: 'utf8' }); (async () => { const client = await (0, chrome_remote_interface_1.default)({ host: commander_1.program.opts().host, port: commander_1.program.opts().port, }); const expression = fileExtension.endsWith('js') ? fileContent : await (async () => { if (process.env['USE_SWC']) { const swc = await Promise.resolve().then(() => __importStar(require('@swc/core'))); return swc.transformSync(fileContent, { module: { type: fileExtension === '.mts' ? 'es6' : 'commonjs', }, jsc: { target: 'es2022', parser: { decorators: true, dynamicImport: true, syntax: 'typescript', tsx: fileExtension.endsWith('x'), }, transform: { decoratorMetadata: true, legacyDecorator: true, }, keepClassNames: true, }, }).code; } const ts = await Promise.resolve().then(() => __importStar(require('typescript'))); return ts.transpileModule(fileContent, { compilerOptions: { lib: ['es2022'], module: fileExtension === '.mts' ? ts.ModuleKind.ES2020 : ts.ModuleKind.CommonJS, moduleResolution: ts.ModuleResolutionKind.NodeJs, target: ts.ScriptTarget.ES2022, esModuleInterop: true, jsx: fileExtension.endsWith('x') ? ts.JsxEmit.React : void 0, }, }).outputText; })(); const resp = await client.Runtime.evaluate({ expression: `(async () => { const exports = {}; ${expression} if (exports.$replJson) { return JSON.stringify(await exports.$replJson(), null, 2); } if (exports.$repl) { return await exports.$repl(); } })()`, includeCommandLineAPI: true, awaitPromise: true, }); console.log(resp.result.value || resp); client.close(); })().catch((e) => { console.error(e); process.exit(1); });