UNPKG

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.

215 lines 31.4 kB
#!/usr/bin/env node import * as fs from 'fs'; import * as http from 'http'; import * as path from 'path'; import { promisify } from 'util'; import * as progam from 'caporal'; import * as open from 'open'; import { WebSocketServer } from 'ws'; // @ts-ignore import { getConnection } from '../listener/iot-connection'; import { getTopic } from '../listener/topic'; const readFileAsync = 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 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 getConnection(true, wsUrlWithoutScope); const topic = 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"]}