serverless-spy
Version:
CDK-based library for writing elegant integration tests on AWS serverless architecture and an additional web console to monitor events in real time.
217 lines • 31.2 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
const fs = require("fs");
const http = require("http");
const path = require("path");
const util_1 = require("util");
const progam = require("caporal");
const open = require("open");
const ws_1 = require("ws");
// @ts-ignore
const iot_connection_1 = require("../listener/iot-connection");
const topic_1 = require("../listener/topic");
const readFileAsync = (0, util_1.promisify)(fs.readFile);
//resolve issue with module import
let opener = open;
if (open.default) {
opener = open.default;
}
async function run() {
let stackList;
let cdkOutput;
let options;
progam
.description('ServerlessSpy web console')
.option('--ws <ws>', 'Websocket link')
.option('--cdkoutput <cdkoutput>', 'CDK output file that contains IoT Endpoint link in a property ServerlessSpyWsUrl')
.option('--cdkstack <cdkstack>', 'CDK stack in cdk output file. If not specified the first one is picked.')
.option('--open <open>', 'Open browser', progam.BOOL, true)
.option('--port <p>', `A port on localhost where ServerlessSpy web console is accessible.`, progam.INT, '3456')
.option('--wsport <wsp>', `A port on localhost where ServerlessSpy websocket is accessible.`, progam.INT, '3457')
.action((_args, opt, _logger) => {
options = opt;
});
progam.parse(process.argv);
if (!options.ws && !options.cdkoutput) {
throw new Error('--ws or --cdkoutput parameter not specified');
}
if (options.cdkoutput) {
const rawdata = fs.readFileSync(options.cdkoutput);
cdkOutput = JSON.parse(rawdata.toString());
stackList = Object.keys(cdkOutput);
}
const wss = new ws_1.WebSocketServer({ port: options.wsport });
let connection = undefined;
wss.on('close', async () => {
if (connection)
connection.end(true);
});
wss.on('connection', async function connect(ws) {
console.log('Connection');
ws.on('message', function message(data) {
console.log('received: %s', data);
});
let wsUrl;
if (options.ws) {
wsUrl = options.ws;
}
else if (cdkOutput) {
if (cdkOutput[options.cdkstack]) {
wsUrl = cdkOutput[options.cdkstack].ServerlessSpyWsUrl;
}
else if (cdkOutput[Object.keys(cdkOutput)[0]]) {
wsUrl = cdkOutput[Object.keys(cdkOutput)[0]].ServerlessSpyWsUrl;
}
}
if (!wsUrl) {
throw new Error('Missing IoT endpoint url');
}
const wsUrlWithoutScope = wsUrl.split('/')[0];
connection = await (0, iot_connection_1.getConnection)(true, wsUrlWithoutScope);
const topic = (0, topic_1.getTopic)('#');
console.log(`Subscribing to ${topic}`);
connection.on('connect', () => {
console.log('Connection opened');
if (connection) {
connection.subscribe(topic);
}
});
connection.on('message', (topic, data) => {
ws.send(JSON.stringify({
...JSON.parse(JSON.parse(data.toString()).data),
topic,
}));
});
});
http
.createServer((request, response) => {
void (async () => {
try {
//console.log('request ', request.url);
let filePath = `.${request.url}`;
//remove query parameters
filePath = filePath.split('?')[0];
let rootFolder = __dirname;
if (request.url?.startsWith('/webServerlessSpy.js')) {
//get transpiled TS to JS files
rootFolder = getCompiledJsPath();
}
else if (request.url?.startsWith('/bootstrap/')) {
filePath = filePath.substring('/bootstrap/'.length);
const bootstrapFolder = await getNpmModuleInstalledPath('bootstrap');
rootFolder = bootstrapFolder;
}
else if (request.url?.startsWith('/bootstrap-icons/')) {
filePath = filePath.substring('/bootstrap-icons/'.length);
const bootstrapFolder = await getNpmModuleInstalledPath('bootstrap-icons');
rootFolder = bootstrapFolder;
}
else {
if (filePath === './') {
filePath = './index.html';
}
}
filePath = path.join(rootFolder, filePath);
//console.log(`${request.url} --> ${filePath}`);
const extname = String(path.extname(filePath)).toLowerCase();
const mimeTypes = {
'.html': 'text/html',
'.js': 'text/javascript',
'.css': 'text/css',
'.json': 'application/json',
'.png': 'image/png',
'.jpg': 'image/jpg',
'.gif': 'image/gif',
'.svg': 'image/svg+xml',
'.wav': 'audio/wav',
'.mp4': 'video/mp4',
'.woff': 'application/font-woff',
'.ttf': 'application/font-ttf',
'.eot': 'application/vnd.ms-fontobject',
'.otf': 'application/font-otf',
'.wasm': 'application/wasm',
};
const contentType = mimeTypes[extname] || 'application/octet-stream';
if (request.url === '/stackList') {
response.writeHead(200, { 'Content-Type': 'application/json' });
response.end(JSON.stringify(stackList), 'utf-8');
}
else if (request.url === '/stackTopicMappings') {
response.writeHead(200, { 'Content-Type': 'application/json' });
const mappings = {};
if (cdkOutput) {
for (const [stackName, stack] of Object.entries(cdkOutput)) {
if (stack.ServerlessSpyWsUrl) {
const [_, scope] = stack.ServerlessSpyWsUrl.split('/');
if (scope) {
mappings[stackName] = scope;
}
}
}
}
response.end(JSON.stringify(mappings), 'utf-8');
}
else if (request.url?.match('^/wsUrl')) {
response.writeHead(200, { 'Content-Type': 'text/html' });
response.end(`ws:localhost:${options.wsport}`, 'utf-8');
}
else {
try {
const content = await readFileAsync(filePath);
response.writeHead(200, { 'Content-Type': contentType });
response.end(content, 'utf-8');
}
catch (error) {
if (error.code === 'ENOENT') {
response.writeHead(404, { 'Content-Type': 'text/html' });
response.end(`No such file or directory ${request.url}`, 'utf-8');
}
else {
response.writeHead(500);
response.end(`Error: ${error.code} ..\n`);
}
}
}
}
catch (err) {
response.writeHead(500, { 'Content-Type': 'text/html' });
response.end(err.message, 'utf-8');
}
})();
})
.listen(options.port);
console.log(`ServerlessSpy console runing at http://localhost:${options.port}`);
if (options.open) {
await opener(`http://localhost:${options.port}`);
}
}
run().catch(console.error);
function getNpmModuleInstalledPath(npm) {
let folder = path.join(__dirname, '../', 'node_modules', npm);
if (fs.existsSync(folder)) {
return folder;
}
let folderAsPackage = path.join(__dirname, '../../', 'node_modules', npm);
if (fs.existsSync(folderAsPackage)) {
return folderAsPackage;
}
// When boostrap ends up in importing projects root node_modules and not in serverless-spys node_modules
folderAsPackage = path.join(__dirname, '../../../../', 'node_modules', npm);
if (fs.existsSync(folderAsPackage)) {
return folderAsPackage;
}
throw new Error(`Can not find package in folder ${folder} and ${folderAsPackage}`);
}
function getCompiledJsPath() {
let folder = path.join(__dirname, '../', 'lib/cli');
if (fs.existsSync(folder)) {
return folder;
}
let folderAsPackage = path.join(__dirname, '../../', 'lib/cli');
if (fs.existsSync(folderAsPackage)) {
return folderAsPackage;
}
throw new Error(`Can not find compiled files in folder ${folder} and ${folderAsPackage}`);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../../cli/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\nimport * as fs from 'fs';\nimport * as http from 'http';\nimport * as path from 'path';\nimport { promisify } from 'util';\nimport { device } from 'aws-iot-device-sdk';\nimport * as progam from 'caporal';\nimport * as open from 'open';\nimport { WebSocketServer } from 'ws';\n// @ts-ignore\nimport { getConnection } from '../listener/iot-connection';\nimport { getTopic } from '../listener/topic';\n\nconst readFileAsync = promisify(fs.readFile);\n\n//resolve issue with module import\nlet opener = open;\nif ((open as any).default) {\n  opener = (open as any).default;\n}\n\nasync function run() {\n  let stackList: string[] | undefined;\n  let cdkOutput: Record<string, Record<string, string>>;\n\n  let options: any;\n\n  progam\n    .description('ServerlessSpy web console')\n    .option('--ws <ws>', 'Websocket link')\n    .option(\n      '--cdkoutput <cdkoutput>',\n      'CDK output file that contains IoT Endpoint link in a property ServerlessSpyWsUrl'\n    )\n    .option(\n      '--cdkstack <cdkstack>',\n      'CDK stack in cdk output file. If not specified the first one is picked.'\n    )\n    .option('--open <open>', 'Open browser', progam.BOOL, true)\n    .option(\n      '--port <p>',\n      `A port on localhost where ServerlessSpy web console is accessible.`,\n      progam.INT,\n      '3456'\n    )\n    .option(\n      '--wsport <wsp>',\n      `A port on localhost where ServerlessSpy websocket is accessible.`,\n      progam.INT,\n      '3457'\n    )\n    .action((_args, opt, _logger) => {\n      options = opt;\n    });\n\n  progam.parse(process.argv);\n\n  if (!options.ws && !options.cdkoutput) {\n    throw new Error('--ws or --cdkoutput parameter not specified');\n  }\n\n  if (options.cdkoutput) {\n    const rawdata = fs.readFileSync(options.cdkoutput);\n    cdkOutput = JSON.parse(rawdata.toString());\n    stackList = Object.keys(cdkOutput);\n  }\n\n  const wss = new WebSocketServer({ port: options.wsport });\n  let connection: device | undefined = undefined;\n\n  wss.on('close', async () => {\n    if (connection) connection.end(true);\n  });\n\n  wss.on('connection', async function connect(ws) {\n    console.log('Connection');\n    ws.on('message', function message(data) {\n      console.log('received: %s', data);\n    });\n\n    let wsUrl: string | undefined;\n    if (options.ws) {\n      wsUrl = options.ws;\n    } else if (cdkOutput) {\n      if (cdkOutput[options.cdkstack]) {\n        wsUrl = cdkOutput[options.cdkstack].ServerlessSpyWsUrl;\n      } else if (cdkOutput[Object.keys(cdkOutput)[0]]) {\n        wsUrl = cdkOutput[Object.keys(cdkOutput)[0]].ServerlessSpyWsUrl;\n      }\n    }\n\n    if (!wsUrl) {\n      throw new Error('Missing IoT endpoint url');\n    }\n\n    const wsUrlWithoutScope = wsUrl.split('/')[0];\n\n    connection = await getConnection(true, wsUrlWithoutScope);\n\n    const topic = getTopic('#');\n    console.log(`Subscribing to ${topic}`);\n\n    connection.on('connect', () => {\n      console.log('Connection opened');\n      if (connection) {\n        connection.subscribe(topic);\n      }\n    });\n\n    connection.on('message', (topic: string, data: Buffer) => {\n      ws.send(\n        JSON.stringify({\n          ...JSON.parse(JSON.parse(data.toString()).data),\n          topic,\n        })\n      );\n    });\n  });\n\n  http\n    .createServer((request, response) => {\n      void (async () => {\n        try {\n          //console.log('request ', request.url);\n          let filePath: string = `.${request.url}`;\n          //remove query parameters\n          filePath = filePath.split('?')[0];\n          let rootFolder = __dirname;\n\n          if (request.url?.startsWith('/webServerlessSpy.js')) {\n            //get transpiled TS to JS files\n            rootFolder = getCompiledJsPath();\n          } else if (request.url?.startsWith('/bootstrap/')) {\n            filePath = filePath.substring('/bootstrap/'.length);\n            const bootstrapFolder = await getNpmModuleInstalledPath(\n              'bootstrap'\n            );\n\n            rootFolder = bootstrapFolder;\n          } else if (request.url?.startsWith('/bootstrap-icons/')) {\n            filePath = filePath.substring('/bootstrap-icons/'.length);\n            const bootstrapFolder = await getNpmModuleInstalledPath(\n              'bootstrap-icons'\n            );\n\n            rootFolder = bootstrapFolder;\n          } else {\n            if (filePath === './') {\n              filePath = './index.html';\n            }\n          }\n\n          filePath = path.join(rootFolder, filePath);\n          //console.log(`${request.url} --> ${filePath}`);\n\n          const extname = String(path.extname(filePath)).toLowerCase();\n          const mimeTypes: any = {\n            '.html': 'text/html',\n            '.js': 'text/javascript',\n            '.css': 'text/css',\n            '.json': 'application/json',\n            '.png': 'image/png',\n            '.jpg': 'image/jpg',\n            '.gif': 'image/gif',\n            '.svg': 'image/svg+xml',\n            '.wav': 'audio/wav',\n            '.mp4': 'video/mp4',\n            '.woff': 'application/font-woff',\n            '.ttf': 'application/font-ttf',\n            '.eot': 'application/vnd.ms-fontobject',\n            '.otf': 'application/font-otf',\n            '.wasm': 'application/wasm',\n          };\n\n          const contentType = mimeTypes[extname] || 'application/octet-stream';\n\n          if (request.url === '/stackList') {\n            response.writeHead(200, { 'Content-Type': 'application/json' });\n            response.end(JSON.stringify(stackList), 'utf-8');\n          } else if (request.url === '/stackTopicMappings') {\n            response.writeHead(200, { 'Content-Type': 'application/json' });\n            const mappings: Record<string, string> = {};\n            if (cdkOutput) {\n              for (const [stackName, stack] of Object.entries(cdkOutput)) {\n                if (stack.ServerlessSpyWsUrl) {\n                  const [_, scope] = stack.ServerlessSpyWsUrl.split('/');\n                  if (scope) {\n                    mappings[stackName] = scope;\n                  }\n                }\n              }\n            }\n            response.end(JSON.stringify(mappings), 'utf-8');\n          } else if (request.url?.match('^/wsUrl')) {\n            response.writeHead(200, { 'Content-Type': 'text/html' });\n            response.end(`ws:localhost:${options.wsport}`, 'utf-8');\n          } else {\n            try {\n              const content = await readFileAsync(filePath);\n\n              response.writeHead(200, { 'Content-Type': contentType });\n              response.end(content, 'utf-8');\n            } catch (error: any) {\n              if (error.code === 'ENOENT') {\n                response.writeHead(404, { 'Content-Type': 'text/html' });\n                response.end(\n                  `No such file or directory ${request.url}`,\n                  'utf-8'\n                );\n              } else {\n                response.writeHead(500);\n                response.end(`Error: ${error.code} ..\\n`);\n              }\n            }\n          }\n        } catch (err: any) {\n          response.writeHead(500, { 'Content-Type': 'text/html' });\n          response.end(err.message, 'utf-8');\n        }\n      })();\n    })\n    .listen(options.port);\n\n  console.log(\n    `ServerlessSpy console runing at http://localhost:${options.port}`\n  );\n  if (options.open) {\n    await opener(`http://localhost:${options.port}`);\n  }\n}\n\nrun().catch(console.error);\n\nfunction getNpmModuleInstalledPath(npm: string) {\n  let folder = path.join(__dirname, '../', 'node_modules', npm);\n  if (fs.existsSync(folder)) {\n    return folder;\n  }\n\n  let folderAsPackage = path.join(__dirname, '../../', 'node_modules', npm);\n\n  if (fs.existsSync(folderAsPackage)) {\n    return folderAsPackage;\n  }\n\n  // When boostrap ends up in importing projects root node_modules and not in serverless-spys node_modules\n  folderAsPackage = path.join(__dirname, '../../../../', 'node_modules', npm);\n\n  if (fs.existsSync(folderAsPackage)) {\n    return folderAsPackage;\n  }\n\n  throw new Error(\n    `Can not find package in folder ${folder} and ${folderAsPackage}`\n  );\n}\n\nfunction getCompiledJsPath() {\n  let folder = path.join(__dirname, '../', 'lib/cli');\n  if (fs.existsSync(folder)) {\n    return folder;\n  }\n\n  let folderAsPackage = path.join(__dirname, '../../', 'lib/cli');\n\n  if (fs.existsSync(folderAsPackage)) {\n    return folderAsPackage;\n  }\n\n  throw new Error(\n    `Can not find compiled files in folder ${folder} and ${folderAsPackage}`\n  );\n}\n"]}
;