-
Notifications
You must be signed in to change notification settings - Fork 59
/
Copy pathindex.js
147 lines (124 loc) · 4.27 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
'use strict'
/**
* Keep your Lambda functions warm
* @author Jeremy Daly <jeremy@jeremydaly.com>
* @license MIT
*/
const id =
Date.now().toString() +
'-' +
('0000' + Math.floor(Math.random() * 1000).toString()).substr(-4)
let warm = false
let lastAccess = null
const funcName = process.env.AWS_LAMBDA_FUNCTION_NAME
const funcVersion = process.env.AWS_LAMBDA_FUNCTION_VERSION
const delay = ms => new Promise(res => setTimeout(res, ms))
const handleEvent = (event, context, config) => {
const isWarmerPing = event && event[config.flag]
if (isWarmerPing) {
let concurrency =
event[config.concurrency] &&
!isNaN(event[config.concurrency]) &&
event[config.concurrency] > 1
? event[config.concurrency]
: 1
let target = event[config.target] || `${funcName}:${funcVersion}`
let invokeCount =
event['__WARMER_INVOCATION__'] && !isNaN(event['__WARMER_INVOCATION__'])
? event['__WARMER_INVOCATION__']
: 1
let invokeTotal =
event['__WARMER_CONCURRENCY__'] && !isNaN(event['__WARMER_CONCURRENCY__'])
? event['__WARMER_CONCURRENCY__']
: concurrency
let correlationId = event['__WARMER_CORRELATIONID__']
? event['__WARMER_CORRELATIONID__']
: config.correlationId
// Create log record
let log = {
action: 'warmer',
function: `${funcName}:${funcVersion}`,
id,
correlationId,
count: invokeCount,
concurrency: invokeTotal,
warm,
lastAccessed: lastAccess,
lastAccessedSeconds:
lastAccess === null
? null
: ((Date.now() - lastAccess) / 1000).toFixed(1)
}
// Log it
config.log && console.log(log) // eslint-disable-line no-console
// flag as warm
warm = true
const currentFunctionAlias = context && context.invokedFunctionArn ? context.invokedFunctionArn.split(':').pop() : funcVersion
// Check whether this lambda is invoking a different lambda
let isDifferentTarget = !(target === `${funcName}:${funcVersion}` ||
target === `${funcName}:${currentFunctionAlias}` ||
(target === funcName && funcVersion === '$LATEST'))
// Fan out if concurrency is set higher than 1
if ((concurrency > 1 || isDifferentTarget) && !event[config.test]) {
// init Lambda service
let lambda = require('./lib/lambda-service')
// init promise array
let invocations = []
// loop through concurrency count
for (let i = isDifferentTarget ? 1 : 2; i <= concurrency; i++) {
// Set the params and wait for the final function to finish
let params = {
FunctionName: target,
InvocationType: i === concurrency ? 'RequestResponse' : 'Event',
LogType: 'None',
Payload: Buffer.from(
JSON.stringify({
[config.flag]: true, // send warmer flag
__WARMER_INVOCATION__: i, // send invocation number
__WARMER_CONCURRENCY__: concurrency, // send total concurrency
__WARMER_CORRELATIONID__: correlationId // send correlation id
})
)
}
// Add promise to invocations array
invocations.push(lambda.invoke(params))
} // end for
// Invoke concurrent functions
return Promise.all(invocations).then(() => true)
} else if (invokeCount > 1) {
return delay(config.delay).then(() => true)
}
return Promise.resolve(true)
} else {
warm = true
lastAccess = Date.now()
return Promise.resolve(false)
}
}
module.exports = (event, cfg = {}, context) => {
let config = Object.assign(
{},
{
flag: 'warmer', // default test flag
concurrency: 'concurrency', // default concurrency field
target: 'target', // default target field
test: 'test', // default test flag
log: true, // default logging to true
correlationId: id, // default the correlationId
delay: 75 // default the delay to 75ms
},
cfg
)
if (Array.isArray(event)) {
let i = 0
const handleNext = () => {
if (i < event.length) {
return handleEvent(event[i++], context, config).then(handleNext)
}
return Promise.resolve(event.some((e) => e[config.flag]))
}
return handleNext()
} else {
return handleEvent(event, context, config)
}
} // end module