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
JavaScript
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"]}