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.

228 lines 35.2 kB
import { getConnection } from './iot-connection'; import { getTopic } from './topic'; export class WsListener { constructor() { this.messages = []; this.trackers = []; this.closed = true; this.functionPrefix = 'waitFor'; this.debugMode = false; this.fragments = new Map(); } async start(params) { this.debugMode = !!params.debugMode; try { this.connection = await getConnection(this.debugMode, params.serverlessSpyWsUrl); this.closed = false; const topic = getTopic(params.scope || '#'); this.log(`Subscribing to topic ${topic}`); const connectionOpenResolve = this.connectionOpenResolve || params.connectionOpenResolve; const localConnection = this.connection; this.connection.on('connect', () => { this.closed = false; this.log('Connection opened'); localConnection.subscribe(topic); if (connectionOpenResolve) { connectionOpenResolve(); } }); this.connection.on('message', (_topic, data) => { if (this.closed) return; this.log('Message received', data.toString()); const fragment = JSON.parse(data.toString()); let message = undefined; if (!fragment.id) { message = JSON.parse(fragment.data); } let pending = this.fragments.get(fragment.id); if (!pending) { pending = new Map(); this.fragments.set(fragment.id, pending); } pending.set(fragment.index, fragment); if (pending.size === fragment.count) { const data = [...pending.values()] .sort((a, b) => a.index - b.index) .map((item) => item.data) .join(''); this.fragments.delete(fragment.id); message = JSON.parse(data); } if (message) { message.serviceKeyForFunction = message.serviceKey.replace(/#/g, ''); if (message.serviceKey.startsWith('Function')) { message.functionContextAwsRequestId = message.data.context.awsRequestId; } this.messages.push(message); this.resolveOldTrackerWithNewMessage(message); } }); this.connection.on('close', () => { this.log('Connection closed'); this.closed = true; }); const connectionOpenReject = this.connectionOpenReject || params.connectionOpenReject; this.connection.on('error', (error) => { this.log('Connection error:', error); connectionOpenReject?.(error); }); } catch (e) { console.error('Failed to get connection', e); throw e; } } async stop() { this.closed = true; this.connection.end(true); } trackerMatchMessage(tracker, message) { if (tracker.finished) return; if ((tracker.serviceKey && tracker.serviceKey === message.serviceKey) || (tracker.serviceKeyForFunction && tracker.serviceKeyForFunction === message.serviceKeyForFunction)) { if (this.trackerMatchCondition(tracker, message)) { tracker.finished = true; const spyAndJestMatchers = { getData: () => message.data, }; const serviceKeyForFunction = tracker.serviceKeyForFunction; if (serviceKeyForFunction && serviceKeyForFunction.startsWith('Function') && (serviceKeyForFunction.endsWith('Request') || serviceKeyForFunction.endsWith('Console'))) { let serviceKeyForFunctionChain = serviceKeyForFunction; if (serviceKeyForFunctionChain.endsWith('Request')) { serviceKeyForFunctionChain = serviceKeyForFunctionChain.substring(0, serviceKeyForFunctionChain.length - 'Request'.length); } else if (serviceKeyForFunctionChain.endsWith('Console')) { serviceKeyForFunctionChain = serviceKeyForFunctionChain.substring(0, serviceKeyForFunctionChain.length - 'Console'.length); } spyAndJestMatchers.followedByConsole = (paramsW) => { return this.createWaitForXXXFunc(`${serviceKeyForFunctionChain}Console`, message.data.context.awsRequestId)(paramsW); }; spyAndJestMatchers.followedByResponse = (paramsW) => { return this.createWaitForXXXFunc(`${serviceKeyForFunctionChain}Response`, message.data.context.awsRequestId)(paramsW); }; } const proxy = new Proxy(spyAndJestMatchers, { get: function (target, objectKey) { if (target.hasOwnProperty(objectKey)) { return target[objectKey]; } else if (objectKey !== 'then') { return function () { const jestFunctionToExecute = expect(message.data)[objectKey]; jestFunctionToExecute.apply(undefined, arguments); return proxy; }; } }, }); tracker.promiseResolve(proxy); return true; } } return false; } resolveTrackerInOldMessages(tracker) { for (const message of this.messages) { if (this.trackerMatchMessage(tracker, message)) { return true; } } return false; } resolveOldTrackerWithNewMessage(message) { for (let index = 0; index < this.trackers.length; index++) { const tracker = this.trackers[index]; if (this.trackerMatchMessage(tracker, message)) { this.trackers = this.trackers.splice(index, 1); return true; } } return false; } trackerMatchCondition(tracker, message) { const matchCondition = (tracker.condition && tracker.condition(message.data)) || !tracker.condition; const matchRequestId = (tracker.functionContextAwsRequestId && tracker.functionContextAwsRequestId === message.functionContextAwsRequestId) || !tracker.functionContextAwsRequestId; if (matchCondition && matchRequestId) { return true; } else { if (!matchCondition && matchRequestId && !tracker.possibleSpyMessageDataForDebugging) { tracker.possibleSpyMessageDataForDebugging = message.data; } return false; } } createWaitForXXXFunc(serviceKeyForFunction, functionContextAwsRequestId) { return (paramsW) => { let resolve; const promise = new Promise((res) => { resolve = res; }); const tracker = { finished: false, // @ts-ignore promiseResolve: resolve, serviceKeyForFunction, functionContextAwsRequestId, }; tracker.condition = paramsW?.condition; let timeoutPid; const timer = new Promise((_, reject) => { timeoutPid = setTimeout(() => { if (tracker.finished) return; tracker.finished = true; let message = `Timeout waiting for Serverless Spy message ${serviceKeyForFunction}.`; if (tracker.possibleSpyMessageDataForDebugging) { message += ` Similar matching spy event data: ${JSON.stringify(tracker.possibleSpyMessageDataForDebugging, null, 2)}`; } reject(new Error(message)); }, paramsW?.timoutMs || 10000); }); if (!this.resolveTrackerInOldMessages(tracker)) { this.trackers.push(tracker); } return Promise.race([promise, timer]).finally(() => { if (!!timeoutPid) { clearTimeout(timeoutPid); } }); }; } createProxy() { const spyListener = {}; spyListener.stop = async () => { await this.stop(); }; return new Proxy(spyListener, { get: (target, objectKey) => { if (target.hasOwnProperty(objectKey)) { return target[objectKey].bind(target); } else if (typeof objectKey === 'string' && objectKey.startsWith(this.functionPrefix)) { const serviceKeyForFunction = objectKey.substring(this.functionPrefix.length); return this.createWaitForXXXFunc(serviceKeyForFunction); } }, }); } log(message, ...optionalParams) { if (this.debugMode && !this.closed) { console.debug('SSPY', message, new Date().toISOString(), ...optionalParams); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"WsListener.js","sourceRoot":"","sources":["../../listener/WsListener.ts"],"names":[],"mappings":"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","sourcesContent":["import { device } from 'aws-iot-device-sdk';\nimport { fragment, getConnection } from './iot-connection';\nimport { ServerlessSpyListener } from './ServerlessSpyListener';\nimport { ServerlessSpyListenerParams } from './ServerlessSpyListenerParams';\nimport { getTopic } from './topic';\nimport { WaitForParams } from './WaitForParams';\nimport { FunctionRequestSpyEvent } from '../common/spyEvents/FunctionRequestSpyEvent';\nimport { SpyMessage } from '../common/spyEvents/SpyMessage';\n\nexport class WsListener<TSpyEvents> {\n  private messages: SpyMessageStorage[] = [];\n  private trackers: Tracker[] = [];\n\n  private connectionOpenResolve?: () => void;\n  private connectionOpenReject?: (reason?: any) => void;\n  private closed = true;\n  private functionPrefix = 'waitFor';\n  private debugMode = false;\n  private connection: device | undefined;\n\n  private fragments = new Map<string, Map<number, fragment>>();\n\n  public async start(params: ServerlessSpyListenerParams) {\n    this.debugMode = !!params.debugMode;\n    try {\n      this.connection = await getConnection(\n        this.debugMode,\n        params.serverlessSpyWsUrl\n      );\n      this.closed = false;\n      const topic = getTopic(params.scope || '#');\n      this.log(`Subscribing to topic ${topic}`);\n      const connectionOpenResolve =\n        this.connectionOpenResolve || params.connectionOpenResolve;\n      const localConnection = this.connection;\n      this.connection.on('connect', () => {\n        this.closed = false;\n        this.log('Connection opened');\n        localConnection.subscribe(topic);\n        if (connectionOpenResolve) {\n          connectionOpenResolve();\n        }\n      });\n      this.connection.on('message', (_topic: string, data: Buffer) => {\n        if (this.closed) return;\n\n        this.log('Message received', data.toString());\n        const fragment = JSON.parse(data.toString());\n        let message: SpyMessageStorage | undefined = undefined;\n        if (!fragment.id) {\n          message = JSON.parse(fragment.data) as SpyMessageStorage;\n        }\n\n        let pending = this.fragments.get(fragment.id);\n        if (!pending) {\n          pending = new Map();\n          this.fragments.set(fragment.id, pending);\n        }\n        pending.set(fragment.index, fragment);\n\n        if (pending.size === fragment.count) {\n          const data = [...pending.values()]\n            .sort((a, b) => a.index - b.index)\n            .map((item) => item.data)\n            .join('');\n          this.fragments.delete(fragment.id);\n          message = JSON.parse(data) as SpyMessageStorage;\n        }\n\n        if (message) {\n          message.serviceKeyForFunction = message.serviceKey.replace(/#/g, '');\n\n          if (message.serviceKey.startsWith('Function')) {\n            message.functionContextAwsRequestId = (\n              message.data as FunctionRequestSpyEvent\n            ).context.awsRequestId;\n          }\n\n          this.messages.push(message);\n          this.resolveOldTrackerWithNewMessage(message);\n        }\n      });\n      this.connection.on('close', () => {\n        this.log('Connection closed');\n\n        this.closed = true;\n      });\n\n      const connectionOpenReject =\n        this.connectionOpenReject || params.connectionOpenReject;\n      this.connection.on('error', (error) => {\n        this.log('Connection error:', error);\n        connectionOpenReject?.(error);\n      });\n    } catch (e) {\n      console.error('Failed to get connection', e);\n      throw e;\n    }\n  }\n\n  public async stop() {\n    this.closed = true;\n    this.connection!.end(true);\n  }\n\n  private trackerMatchMessage(tracker: Tracker, message: SpyMessageStorage) {\n    if (tracker.finished) return;\n\n    if (\n      (tracker.serviceKey && tracker.serviceKey === message.serviceKey) ||\n      (tracker.serviceKeyForFunction &&\n        tracker.serviceKeyForFunction === message.serviceKeyForFunction)\n    ) {\n      if (this.trackerMatchCondition(tracker, message)) {\n        tracker.finished = true;\n\n        const spyAndJestMatchers: any = {\n          getData: () => message.data,\n        };\n\n        const serviceKeyForFunction = tracker.serviceKeyForFunction;\n        if (\n          serviceKeyForFunction &&\n          serviceKeyForFunction.startsWith('Function') &&\n          (serviceKeyForFunction.endsWith('Request') ||\n            serviceKeyForFunction.endsWith('Console'))\n        ) {\n          let serviceKeyForFunctionChain = serviceKeyForFunction;\n\n          if (serviceKeyForFunctionChain.endsWith('Request')) {\n            serviceKeyForFunctionChain = serviceKeyForFunctionChain.substring(\n              0,\n              serviceKeyForFunctionChain.length - 'Request'.length\n            );\n          } else if (serviceKeyForFunctionChain.endsWith('Console')) {\n            serviceKeyForFunctionChain = serviceKeyForFunctionChain.substring(\n              0,\n              serviceKeyForFunctionChain.length - 'Console'.length\n            );\n          }\n\n          spyAndJestMatchers.followedByConsole = (paramsW: WaitForParams) => {\n            return this.createWaitForXXXFunc(\n              `${serviceKeyForFunctionChain}Console`,\n              (message.data as FunctionRequestSpyEvent).context.awsRequestId\n            )(paramsW);\n          };\n\n          spyAndJestMatchers.followedByResponse = (paramsW: WaitForParams) => {\n            return this.createWaitForXXXFunc(\n              `${serviceKeyForFunctionChain}Response`,\n              (message.data as FunctionRequestSpyEvent).context.awsRequestId\n            )(paramsW);\n          };\n        }\n\n        const proxy = new Proxy(spyAndJestMatchers, {\n          get: function (target: any, objectKey: string) {\n            if (target.hasOwnProperty(objectKey)) {\n              return target[objectKey];\n            } else if (objectKey !== 'then') {\n              return function () {\n                const jestFunctionToExecute = (expect(message.data) as any)[\n                  objectKey\n                ];\n                jestFunctionToExecute.apply(undefined, arguments);\n                return proxy;\n              };\n            }\n          },\n        });\n\n        tracker.promiseResolve(proxy);\n        return true;\n      }\n    }\n    return false;\n  }\n\n  private resolveTrackerInOldMessages(tracker: Tracker) {\n    for (const message of this.messages) {\n      if (this.trackerMatchMessage(tracker, message)) {\n        return true;\n      }\n    }\n\n    return false;\n  }\n\n  private resolveOldTrackerWithNewMessage(message: SpyMessageStorage) {\n    for (let index = 0; index < this.trackers.length; index++) {\n      const tracker = this.trackers[index];\n      if (this.trackerMatchMessage(tracker, message)) {\n        this.trackers = this.trackers.splice(index, 1);\n        return true;\n      }\n    }\n\n    return false;\n  }\n\n  private trackerMatchCondition(tracker: Tracker, message: SpyMessageStorage) {\n    const matchCondition =\n      (tracker.condition && tracker.condition(message.data)) ||\n      !tracker.condition;\n\n    const matchRequestId =\n      (tracker.functionContextAwsRequestId &&\n        tracker.functionContextAwsRequestId ===\n          message.functionContextAwsRequestId) ||\n      !tracker.functionContextAwsRequestId;\n\n    if (matchCondition && matchRequestId) {\n      return true;\n    } else {\n      if (\n        !matchCondition &&\n        matchRequestId &&\n        !tracker.possibleSpyMessageDataForDebugging\n      ) {\n        tracker.possibleSpyMessageDataForDebugging = message.data;\n      }\n      return false;\n    }\n  }\n\n  private createWaitForXXXFunc(\n    serviceKeyForFunction: string,\n    functionContextAwsRequestId?: string\n  ) {\n    return (paramsW?: WaitForParams) => {\n      let resolve: (value: void | PromiseLike<any>) => void;\n      const promise = new Promise((res) => {\n        resolve = res;\n      });\n      const tracker: Tracker = {\n        finished: false,\n        // @ts-ignore\n        promiseResolve: resolve,\n        serviceKeyForFunction,\n        functionContextAwsRequestId,\n      };\n\n      tracker.condition = paramsW?.condition;\n\n      let timeoutPid: NodeJS.Timeout | undefined;\n      const timer = new Promise((_, reject) => {\n        timeoutPid = setTimeout(() => {\n          if (tracker.finished) return;\n          tracker.finished = true;\n          let message = `Timeout waiting for Serverless Spy message ${serviceKeyForFunction}.`;\n\n          if (tracker.possibleSpyMessageDataForDebugging) {\n            message += ` Similar matching spy event data: ${JSON.stringify(\n              tracker.possibleSpyMessageDataForDebugging,\n              null,\n              2\n            )}`;\n          }\n\n          reject(new Error(message));\n        }, paramsW?.timoutMs || 10000);\n      });\n\n      if (!this.resolveTrackerInOldMessages(tracker)) {\n        this.trackers.push(tracker);\n      }\n\n      return Promise.race([promise, timer]).finally(() => {\n        if (!!timeoutPid) {\n          clearTimeout(timeoutPid);\n        }\n      });\n    };\n  }\n\n  public createProxy() {\n    const spyListener = {} as ServerlessSpyListener<TSpyEvents>;\n\n    spyListener.stop = async () => {\n      await this.stop();\n    };\n\n    return new Proxy<ServerlessSpyListener<TSpyEvents>>(spyListener, {\n      get: (target: any, objectKey: string) => {\n        if (target.hasOwnProperty(objectKey)) {\n          return target[objectKey].bind(target);\n        } else if (\n          typeof objectKey === 'string' &&\n          objectKey.startsWith(this.functionPrefix)\n        ) {\n          const serviceKeyForFunction = objectKey.substring(\n            this.functionPrefix.length\n          );\n\n          return this.createWaitForXXXFunc(serviceKeyForFunction);\n        }\n      },\n    });\n  }\n\n  private log(message: string, ...optionalParams: any[]) {\n    if (this.debugMode && !this.closed) {\n      console.debug(\n        'SSPY',\n        message,\n        new Date().toISOString(),\n        ...optionalParams\n      );\n    }\n  }\n}\n\ntype Tracker = {\n  promiseResolve: (data: any) => void;\n  finished: boolean;\n  serviceKey?: string;\n  serviceKeyForFunction?: string;\n  condition?: (data: any) => boolean;\n  timoutMs?: number;\n  functionContextAwsRequestId?: string;\n  possibleSpyMessageDataForDebugging?: any;\n};\n\ntype SpyMessageStorage = SpyMessage & {\n  serviceKeyForFunction: string;\n  functionContextAwsRequestId?: string;\n};\n"]}