serverless-offline-kinesis
Version:
Emulate AWS λ and Kinesis locally when developing your Serverless project
202 lines (148 loc) • 4.71 kB
JavaScript
const {assign, omitBy, isUndefined, get, startsWith, pick} = require('lodash/fp');
const log = require('@serverless/utils/log').log;
const Kinesis = require('./kinesis');
const OFFLINE_OPTION = 'serverless-offline';
const CUSTOM_OPTION = 'serverless-offline-kinesis';
const SERVER_SHUTDOWN_TIMEOUT = 5000;
const defaultOptions = {
accountId: '000000000000'
};
const omitUndefined = omitBy(isUndefined);
class ServerlessOfflineKinesis {
constructor(serverless, cliOptions) {
this.cliOptions = null;
this.options = null;
this.kinesis = null;
this.lambda = null;
this.serverless = null;
this.cliOptions = cliOptions;
this.serverless = serverless;
this.hooks = {
'offline:start:init': this.start.bind(this),
'offline:start:ready': this.ready.bind(this),
'offline:start': this._startWithReady.bind(this),
'offline:start:end': this.end.bind(this)
};
}
async start() {
process.env.IS_OFFLINE = true;
this._mergeOptions();
const {kinesisEvents, lambdas} = this._getEvents();
await this._createLambda(lambdas);
const eventModules = [];
if (kinesisEvents.length > 0) {
eventModules.push(this._createKinesis(kinesisEvents));
}
await Promise.all(eventModules);
this.serverless.cli.log(
`Starting Offline Kinesis at stage ${this.options.stage} (${this.options.region})`
);
}
ready() {
if (process.env.NODE_ENV !== 'test') {
this._listenForTermination();
}
}
_listenForTermination() {
const signals = ['SIGINT', 'SIGTERM'];
signals.map(signal =>
process.on(signal, async () => {
this.serverless.cli.log(`Got ${signal} signal. Offline Halting...`);
await this.end();
})
);
}
async _startWithReady() {
await this.start();
this.ready();
}
async end(skipExit) {
if (process.env.NODE_ENV === 'test' && skipExit === undefined) {
return;
}
this.serverless.cli.log('Halting offline server');
const eventModules = [];
if (this.lambda) {
eventModules.push(this.lambda.cleanup());
}
if (this.kinesis) {
eventModules.push(this.kinesis.stop(SERVER_SHUTDOWN_TIMEOUT));
}
await Promise.all(eventModules);
if (!skipExit) {
process.exit(0);
}
}
async _createLambda(lambdas) {
const {default: Lambda} = await import('serverless-offline/lambda');
this.lambda = new Lambda(this.serverless, this.options);
this.lambda.create(lambdas);
}
async _createKinesis(events, skipStart) {
this.kinesis = new Kinesis(this.lambda, this.options);
await this.kinesis.create(events);
if (!skipStart) {
await this.kinesis.start();
}
}
_mergeOptions() {
const {
service: {custom = {}, provider}
} = this.serverless;
const offlineOptions = custom[OFFLINE_OPTION];
const customOptions = custom[CUSTOM_OPTION];
this.options = Object.assign(
{},
omitUndefined(defaultOptions),
omitUndefined(provider),
omitUndefined(pick(['location', 'localEnvironment'], offlineOptions)), // serverless-webpack support
omitUndefined(customOptions),
omitUndefined(this.cliOptions)
);
log.debug('options:', this.options);
}
_getEvents() {
const {service} = this.serverless;
const lambdas = [];
const kinesisEvents = [];
const functionKeys = service.getAllFunctions();
functionKeys.forEach(functionKey => {
const functionDefinition = service.getFunction(functionKey);
lambdas.push({functionKey, functionDefinition});
const events = service.getAllEventsInFunction(functionKey) || [];
events.forEach(event => {
const {stream} = event;
if (
stream &&
(stream.type === 'kinesis' || startsWith('arn:aws:kinesis', stream)) &&
functionDefinition.handler
) {
kinesisEvents.push({
functionKey,
handler: functionDefinition.handler,
kinesis: this._resolveFn(stream)
});
}
});
});
return {
kinesisEvents,
lambdas
};
}
_resolveFn(event) {
if (typeof event.streamName === 'string') return event;
const getAtt = get(['arn', 'Fn::GetAtt'], event);
if (getAtt) {
const [resourceName] = getAtt;
const properties = get(
['service', 'resources', 'Resources', resourceName, 'Properties'],
this.serverless
);
if (!properties) throw new Error(`No resource defined with name ${resourceName}`);
return assign(event, {streamName: properties.Name});
}
return event;
}
}
module.exports = ServerlessOfflineKinesis;