1
1
import type { ArgumentPlaceholder , Expression , SpreadElement , JSXNamespacedName } from '@babel/types'
2
+ import mergeOptions from 'merge-options'
3
+ import { z } from 'zod'
2
4
5
+ import { FunctionConfig , functionConfig } from '../../../config.js'
3
6
import { InvocationMode , INVOCATION_MODE } from '../../../function.js'
4
- import { TrafficRules } from '../../../manifest.js'
5
- import { RateLimitAction , RateLimitAggregator , RateLimitAlgorithm } from '../../../rate_limit.js'
7
+ import { rateLimit } from '../../../rate_limit.js'
6
8
import { FunctionBundlingUserError } from '../../../utils/error.js'
7
- import { nonNullable } from '../../../utils/non_nullable.js'
8
- import { getRoutes , Route } from '../../../utils/routes.js'
9
+ import { Route , getRoutes } from '../../../utils/routes.js'
9
10
import { RUNTIME } from '../../runtime.js'
10
- import { NODE_BUNDLER } from '../bundlers/types.js'
11
11
import { createBindingsMethod } from '../parser/bindings.js'
12
12
import { traverseNodes } from '../parser/exports.js'
13
13
import { getImports } from '../parser/imports.js'
@@ -18,26 +18,55 @@ import { parse as parseSchedule } from './properties/schedule.js'
18
18
19
19
export const IN_SOURCE_CONFIG_MODULE = '@netlify/functions'
20
20
21
- export type ISCValues = {
22
- routes ?: Route [ ]
23
- schedule ?: string
24
- methods ?: string [ ]
25
- trafficRules ?: TrafficRules
26
- name ?: string
27
- generator ?: string
28
- timeout ?: number
29
- }
30
-
31
- export interface StaticAnalysisResult extends ISCValues {
21
+ export interface StaticAnalysisResult {
22
+ config : InSourceConfig
32
23
inputModuleFormat ?: ModuleFormat
33
24
invocationMode ?: InvocationMode
25
+ routes ?: Route [ ]
34
26
runtimeAPIVersion ?: number
35
27
}
36
28
37
29
interface FindISCDeclarationsOptions {
38
30
functionName : string
39
31
}
40
32
33
+ const ensureArray = ( input : unknown ) => ( Array . isArray ( input ) ? input : [ input ] )
34
+
35
+ const httpMethods = z . preprocess (
36
+ ( input ) => ( typeof input === 'string' ? input . toUpperCase ( ) : input ) ,
37
+ z . enum ( [ 'GET' , 'POST' , 'PUT' , 'PATCH' , 'OPTIONS' , 'DELETE' , 'HEAD' ] ) ,
38
+ )
39
+ const path = z . string ( ) . startsWith ( '/' , { message : "Must start with a '/'" } )
40
+
41
+ export const inSourceConfig = functionConfig
42
+ . pick ( {
43
+ externalNodeModules : true ,
44
+ generator : true ,
45
+ includedFiles : true ,
46
+ ignoredNodeModules : true ,
47
+ name : true ,
48
+ nodeBundler : true ,
49
+ nodeVersion : true ,
50
+ schedule : true ,
51
+ timeout : true ,
52
+ } )
53
+ . extend ( {
54
+ method : z
55
+ . union ( [ httpMethods , z . array ( httpMethods ) ] , {
56
+ errorMap : ( ) => ( { message : 'Must be a string or array of strings' } ) ,
57
+ } )
58
+ . transform ( ensureArray )
59
+ . optional ( ) ,
60
+ path : z
61
+ . union ( [ path , z . array ( path ) ] , { errorMap : ( ) => ( { message : 'Must be a string or array of strings' } ) } )
62
+ . transform ( ensureArray )
63
+ . optional ( ) ,
64
+ preferStatic : z . boolean ( ) . optional ( ) . catch ( undefined ) ,
65
+ rateLimit : rateLimit . optional ( ) . catch ( undefined ) ,
66
+ } )
67
+
68
+ export type InSourceConfig = z . infer < typeof inSourceConfig >
69
+
41
70
const validateScheduleFunction = ( functionFound : boolean , scheduleFound : boolean , functionName : string ) : void => {
42
71
if ( ! functionFound ) {
43
72
throw new FunctionBundlingUserError (
@@ -54,83 +83,6 @@ const validateScheduleFunction = (functionFound: boolean, scheduleFound: boolean
54
83
}
55
84
}
56
85
57
- /**
58
- * Normalizes method names into arrays of uppercase strings.
59
- * (e.g. "get" becomes ["GET"])
60
- */
61
- const normalizeMethods = ( input : unknown , name : string ) : string [ ] | undefined => {
62
- const methods = Array . isArray ( input ) ? input : [ input ]
63
-
64
- return methods . map ( ( method ) => {
65
- if ( typeof method !== 'string' ) {
66
- throw new FunctionBundlingUserError (
67
- `Could not parse method declaration of function '${ name } '. Expecting HTTP Method, got ${ method } ` ,
68
- {
69
- functionName : name ,
70
- runtime : RUNTIME . JAVASCRIPT ,
71
- bundler : NODE_BUNDLER . ESBUILD ,
72
- } ,
73
- )
74
- }
75
-
76
- return method . toUpperCase ( )
77
- } )
78
- }
79
-
80
- /**
81
- * Extracts the `ratelimit` configuration from the exported config.
82
- */
83
- const getTrafficRulesConfig = ( input : unknown , name : string ) : TrafficRules | undefined => {
84
- if ( typeof input !== 'object' || input === null ) {
85
- throw new FunctionBundlingUserError (
86
- `Could not parse ratelimit declaration of function '${ name } '. Expecting an object, got ${ input } ` ,
87
- {
88
- functionName : name ,
89
- runtime : RUNTIME . JAVASCRIPT ,
90
- bundler : NODE_BUNDLER . ESBUILD ,
91
- } ,
92
- )
93
- }
94
-
95
- const { windowSize, windowLimit, algorithm, aggregateBy, action } = input as Record < string , unknown >
96
-
97
- if (
98
- typeof windowSize !== 'number' ||
99
- typeof windowLimit !== 'number' ||
100
- ! Number . isInteger ( windowSize ) ||
101
- ! Number . isInteger ( windowLimit )
102
- ) {
103
- throw new FunctionBundlingUserError (
104
- `Could not parse ratelimit declaration of function '${ name } '. Expecting 'windowSize' and 'limitSize' integer properties, got ${ input } ` ,
105
- {
106
- functionName : name ,
107
- runtime : RUNTIME . JAVASCRIPT ,
108
- bundler : NODE_BUNDLER . ESBUILD ,
109
- } ,
110
- )
111
- }
112
-
113
- const rateLimitAgg = Array . isArray ( aggregateBy ) ? aggregateBy : [ RateLimitAggregator . Domain ]
114
- const rewriteConfig = 'to' in input && typeof input . to === 'string' ? { to : input . to } : undefined
115
-
116
- return {
117
- action : {
118
- type : ( action as RateLimitAction ) || RateLimitAction . Limit ,
119
- config : {
120
- ...rewriteConfig ,
121
- rateLimitConfig : {
122
- windowLimit,
123
- windowSize,
124
- algorithm : ( algorithm as RateLimitAlgorithm ) || RateLimitAlgorithm . SlidingWindow ,
125
- } ,
126
- aggregate : {
127
- keys : rateLimitAgg . map ( ( agg ) => ( { type : agg } ) ) ,
128
- } ,
129
- } ,
130
- } ,
131
- }
132
- }
133
-
134
86
/**
135
87
* Loads a file at a given path, parses it into an AST, and returns a series of
136
88
* data points, such as in-source configuration properties and other metadata.
@@ -142,7 +94,9 @@ export const parseFile = async (
142
94
const source = await safelyReadSource ( sourcePath )
143
95
144
96
if ( source === null ) {
145
- return { }
97
+ return {
98
+ config : { } ,
99
+ }
146
100
}
147
101
148
102
return parseSource ( source , { functionName } )
@@ -157,7 +111,9 @@ export const parseSource = (source: string, { functionName }: FindISCDeclaration
157
111
const ast = safelyParseSource ( source )
158
112
159
113
if ( ast === null ) {
160
- return { }
114
+ return {
115
+ config : { } ,
116
+ }
161
117
}
162
118
163
119
const imports = ast . body . flatMap ( ( node ) => getImports ( node , IN_SOURCE_CONFIG_MODULE ) )
@@ -172,92 +128,97 @@ export const parseSource = (source: string, { functionName }: FindISCDeclaration
172
128
173
129
if ( isV2API ) {
174
130
const result : StaticAnalysisResult = {
131
+ config : { } ,
175
132
inputModuleFormat,
176
133
runtimeAPIVersion : 2 ,
177
134
}
135
+ const { data, error, success } = inSourceConfig . safeParse ( configExport )
136
+
137
+ if ( success ) {
138
+ result . config = data
139
+ result . routes = getRoutes ( {
140
+ functionName,
141
+ methods : data . method ?? [ ] ,
142
+ path : data . path ,
143
+ preferStatic : data . preferStatic ,
144
+ } )
145
+ } else {
146
+ // TODO: Handle multiple errors.
147
+ const [ issue ] = error . issues
178
148
179
- if ( typeof configExport . schedule === 'string' ) {
180
- result . schedule = configExport . schedule
181
- }
182
-
183
- if ( typeof configExport . name === 'string' ) {
184
- result . name = configExport . name
149
+ throw new FunctionBundlingUserError (
150
+ `Function ${ functionName } has a configuration error on '${ issue . path . join ( '.' ) } ': ${ issue . message } ` ,
151
+ {
152
+ functionName,
153
+ runtime : RUNTIME . JAVASCRIPT ,
154
+ } ,
155
+ )
185
156
}
186
157
187
- if ( typeof configExport . generator === 'string' ) {
188
- result . generator = configExport . generator
189
- }
158
+ return result
159
+ }
190
160
191
- if ( typeof configExport . timeout === 'number' ) {
192
- result . timeout = configExport . timeout
193
- }
161
+ const result : StaticAnalysisResult = {
162
+ config : { } ,
163
+ inputModuleFormat,
164
+ runtimeAPIVersion : 1 ,
165
+ }
194
166
195
- if ( configExport . method !== undefined ) {
196
- result . methods = normalizeMethods ( configExport . method , functionName )
167
+ handlerExports . forEach ( ( node ) => {
168
+ // We're only interested in exports with call expressions, since that's
169
+ // the pattern we use for the wrapper functions.
170
+ if ( node . type !== 'call-expression' ) {
171
+ return
197
172
}
198
173
199
- result . routes = getRoutes ( {
200
- functionName,
201
- methods : result . methods ?? [ ] ,
202
- path : configExport . path ,
203
- preferStatic : configExport . preferStatic === true ,
204
- } )
174
+ const { args, local : exportName } = node
175
+ const matchingImport = imports . find ( ( { local : importName } ) => importName === exportName )
205
176
206
- if ( configExport . rateLimit ! == undefined ) {
207
- result . trafficRules = getTrafficRulesConfig ( configExport . rateLimit , functionName )
177
+ if ( matchingImport = == undefined ) {
178
+ return
208
179
}
209
180
210
- return result
211
- }
181
+ switch ( matchingImport . imported ) {
182
+ case 'schedule' : {
183
+ const parsed = parseSchedule ( { args } , getAllBindings )
212
184
213
- const iscExports = handlerExports
214
- . map ( ( node ) => {
215
- // We're only interested in exports with call expressions, since that's
216
- // the pattern we use for the wrapper functions.
217
- if ( node . type !== 'call-expression' ) {
218
- return null
219
- }
185
+ scheduledFunctionFound = true
186
+ if ( parsed . schedule ) {
187
+ scheduleFound = true
188
+ }
220
189
221
- const { args, local : exportName } = node
222
- const matchingImport = imports . find ( ( { local : importName } ) => importName === exportName )
190
+ if ( parsed . schedule !== undefined ) {
191
+ result . config . schedule = parsed . schedule
192
+ }
223
193
224
- if ( matchingImport === undefined ) {
225
- return null
194
+ return
226
195
}
227
196
228
- switch ( matchingImport . imported ) {
229
- case 'schedule' : {
230
- const parsed = parseSchedule ( { args } , getAllBindings )
231
-
232
- scheduledFunctionFound = true
233
- if ( parsed . schedule ) {
234
- scheduleFound = true
235
- }
197
+ case 'stream' : {
198
+ result . invocationMode = INVOCATION_MODE . Stream
236
199
237
- return parsed
238
- }
239
-
240
- case 'stream' : {
241
- return {
242
- invocationMode : INVOCATION_MODE . Stream ,
243
- }
244
- }
245
-
246
- default :
247
- // no-op
200
+ return
248
201
}
249
202
250
- return null
251
- } )
252
- . filter ( nonNullable )
203
+ default :
204
+ // no-op
205
+ }
206
+
207
+ return
208
+ } )
253
209
254
210
if ( scheduledFunctionExpected ) {
255
211
validateScheduleFunction ( scheduledFunctionFound , scheduleFound , functionName )
256
212
}
257
213
258
- const mergedExports : ISCValues = iscExports . reduce ( ( acc , obj ) => ( { ...acc , ...obj } ) , { } )
214
+ return result
215
+ }
259
216
260
- return { ...mergedExports , inputModuleFormat, runtimeAPIVersion : 1 }
217
+ export const augmentFunctionConfig = (
218
+ tomlConfig : FunctionConfig ,
219
+ inSourceConfig : InSourceConfig = { } ,
220
+ ) : FunctionConfig & InSourceConfig => {
221
+ return mergeOptions . call ( { concatArrays : true } , tomlConfig , inSourceConfig )
261
222
}
262
223
263
224
export type ISCHandlerArg = ArgumentPlaceholder | Expression | SpreadElement | JSXNamespacedName
0 commit comments