Skip to content

Commit c3fa0c1

Browse files
authored
Merge pull request #25 from jeremydaly/v0.4.0
v0.4.0
2 parents 514d73a + 0516b7b commit c3fa0c1

26 files changed

+1861
-161
lines changed

README.md

+161-24
Large diffs are not rendered by default.

index.js

+47-17
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,12 @@ class API {
1919

2020
// Set the version and base paths
2121
this._version = props && props.version ? props.version : 'v1'
22-
this._base = props && props.base ? props.base.trim() : ''
22+
this._base = props && props.base && typeof props.base === 'string' ? props.base.trim() : ''
2323
this._callbackName = props && props.callback ? props.callback.trim() : 'callback'
24+
this._mimeTypes = props && props.mimeTypes && typeof props.mimeTypes === 'object' ? props.mimeTypes : {}
25+
26+
// Prefix stack w/ base
27+
this._prefix = this.parseRoute(this._base)
2428

2529
// Stores timers for debugging
2630
this._timers = {}
@@ -56,6 +60,7 @@ class API {
5660

5761
// Testing flag
5862
this._test = false
63+
5964
} // end constructor
6065

6166
// GET: convenience method
@@ -73,6 +78,11 @@ class API {
7378
this.METHOD('PUT', path, handler)
7479
}
7580

81+
// PATCH: convenience method
82+
patch(path, handler) {
83+
this.METHOD('PATCH', path, handler)
84+
}
85+
7686
// DELETE: convenience method
7787
delete(path, handler) {
7888
this.METHOD('DELETE', path, handler)
@@ -86,8 +96,14 @@ class API {
8696
// METHOD: Adds method and handler to routes
8797
METHOD(method, path, handler) {
8898

99+
// Parse the path
100+
let parsedPath = this.parseRoute(path)
101+
89102
// Split the route and clean it up
90-
let route = path.trim().replace(/^\/(.*?)(\/)*$/,'$1').split('/')
103+
let route = this._prefix.concat(parsedPath)
104+
105+
// For root path support
106+
if (route.length === 0) { route.push('')}
91107

92108
// Keep track of path variables
93109
let pathVars = {}
@@ -106,7 +122,7 @@ class API {
106122
// Add the route to the global _routes
107123
this.setRoute(
108124
this._routes,
109-
(i === route.length-1 ? { ['__'+method.toUpperCase()]: { vars: pathVars, handler: handler, route: path } } : {}),
125+
(i === route.length-1 ? { ['__'+method.toUpperCase()]: { vars: pathVars, handler: handler, route: '/'+parsedPath.join('/') } } : {}),
110126
route.slice(0,i+1)
111127
);
112128

@@ -141,6 +157,12 @@ class API {
141157

142158
}).catch((e) => {
143159

160+
// Error messages should never be base64 encoded
161+
response._isBase64 = false
162+
163+
// Strip the headers (TODO: find a better way to handle this)
164+
response._headers = {}
165+
144166
let message;
145167

146168
if (e instanceof Error) {
@@ -298,20 +320,11 @@ class API {
298320
// UTILITY FUNCTIONS
299321
//-------------------------------------------------------------------------//
300322

301-
deepFind(obj, path) {
302-
let paths = path//.split('.'),
303-
let current = obj
304-
305-
for (let i = 0; i < paths.length; ++i) {
306-
if (current[paths[i]] == undefined) {
307-
return undefined
308-
} else {
309-
current = current[paths[i]]
310-
}
311-
}
312-
return current
323+
parseRoute(path) {
324+
return path.trim().replace(/^\/(.*?)(\/)*$/,'$1').split('/').filter(x => x.trim() !== '')
313325
}
314326

327+
315328
setRoute(obj, value, path) {
316329
if (typeof path === "string") {
317330
let path = path.split('.')
@@ -354,9 +367,26 @@ class API {
354367
return this._app
355368
}
356369

370+
371+
// Register routes with options
372+
register(fn,options) {
373+
374+
// Extract Prefix
375+
let prefix = options.prefix && options.prefix.toString().trim() !== '' ?
376+
this.parseRoute(options.prefix) : []
377+
378+
// Concat to existing prefix
379+
this._prefix = this._prefix.concat(prefix)
380+
381+
// Execute the routing function
382+
fn(this,options)
383+
384+
// Remove the last prefix
385+
this._prefix = this._prefix.slice(0,-(prefix.length))
386+
387+
} // end register
388+
357389
} // end API class
358390

359391
// Export the API class
360392
module.exports = opts => new API(opts)
361-
362-
// console.error('DEPRECATED: constructor method. Use require(\'lambda-api\')({ version: \'v1.0\', base: \'v1\' }) to initialize the framework instead')

mimemap.js

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
'use strict'
2+
3+
/**
4+
* Lightweight web framework for your serverless applications
5+
* @author Jeremy Daly <[email protected]>
6+
* @license MIT
7+
*/
8+
9+
// Minimal mime map for common file types
10+
11+
module.exports = {
12+
// images
13+
gif: 'image/gif',
14+
ico: 'image/x-icon',
15+
jpg: 'image/jpeg',
16+
jpeg: 'image/jpeg',
17+
png: 'image/png',
18+
svg: 'image/svg+xml',
19+
svgz: 'image/svg+xml',
20+
21+
// text
22+
atom: 'application/atom+xml',
23+
css: 'text/css',
24+
csv: 'text/csv',
25+
html: 'text/html',
26+
htm: 'text/html',
27+
js: 'application/javascript',
28+
json: 'application/json',
29+
map: 'application/json',
30+
rdf: 'application/rdf+xml',
31+
rss: 'application/rss+xml',
32+
txt: 'text/plain',
33+
webmanifest: 'application/manifest+json',
34+
xml: 'application/xml',
35+
xls: 'application/xml',
36+
37+
// other binary
38+
gz: 'application/gzip',
39+
pdf: 'application/pdf',
40+
zip: 'application/zip',
41+
42+
// fonts
43+
woff: 'application/font-woff',
44+
45+
// MS file Types
46+
doc: 'application/msword',
47+
dot: 'application/msword',
48+
docx: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
49+
dotx: 'application/vnd.openxmlformats-officedocument.wordprocessingml.template',
50+
docm: 'application/vnd.ms-word.document.macroEnabled.12',
51+
dotm: 'application/vnd.ms-word.template.macroEnabled.12',
52+
xls: 'application/vnd.ms-excel',
53+
xlt: 'application/vnd.ms-excel',
54+
xla: 'application/vnd.ms-excel',
55+
xlsx: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
56+
xltx: 'application/vnd.openxmlformats-officedocument.spreadsheetml.template',
57+
xlsm: 'application/vnd.ms-excel.sheet.macroEnabled.12',
58+
xltm: 'application/vnd.ms-excel.template.macroEnabled.12',
59+
xlam: 'application/vnd.ms-excel.addin.macroEnabled.12',
60+
xlsb: 'application/vnd.ms-excel.sheet.binary.macroEnabled.12',
61+
ppt: 'application/vnd.ms-powerpoint',
62+
pot: 'application/vnd.ms-powerpoint',
63+
pps: 'application/vnd.ms-powerpoint',
64+
ppa: 'application/vnd.ms-powerpoint',
65+
pptx: 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
66+
potx: 'application/vnd.openxmlformats-officedocument.presentationml.template',
67+
ppsx: 'application/vnd.openxmlformats-officedocument.presentationml.slideshow',
68+
ppam: 'application/vnd.ms-powerpoint.addin.macroEnabled.12',
69+
pptm: 'application/vnd.ms-powerpoint.presentation.macroEnabled.12',
70+
potm: 'application/vnd.ms-powerpoint.template.macroEnabled.12',
71+
ppsm: 'application/vnd.ms-powerpoint.slideshow.macroEnabled.12',
72+
mdb: 'application/vnd.ms-access'
73+
74+
}

package.json

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "lambda-api",
3-
"version": "0.3.1",
3+
"version": "0.4.0",
44
"description": "Lightweight web framework for your serverless applications",
55
"main": "index.js",
66
"scripts": {
@@ -32,8 +32,10 @@
3232
"bluebird": "^3.5.1"
3333
},
3434
"devDependencies": {
35+
"aws-sdk": "^2.218.1",
3536
"chai": "^4.1.2",
36-
"mocha": "^4.0.1"
37+
"mocha": "^4.0.1",
38+
"sinon": "^4.5.0"
3739
},
3840
"engines": {
3941
"node": ">= 6.10.0"

request.js

+10-9
Original file line numberDiff line numberDiff line change
@@ -55,23 +55,24 @@ class REQUEST {
5555
// Set the requestContext
5656
this.requestContext = app._event.requestContext
5757

58+
// Capture the raw body
59+
this.rawBody = app._event.body
60+
61+
// Set the body (decode it if base64 encoded)
62+
this.body = app._event.isBase64Encoded ? Buffer.from(app._event.body, 'base64').toString() : app._event.body
63+
5864
// Set the body
5965
if (this.headers['content-type'] && this.headers['content-type'].includes("application/x-www-form-urlencoded")) {
60-
this.body = QS.parse(app._event.body)
61-
} else if (typeof app._event.body === 'object') {
62-
this.body = app._event.body
66+
this.body = QS.parse(this.body)
67+
} else if (typeof this.body === 'object') {
68+
this.body = this.body
6369
} else {
64-
this.body = parseBody(app._event.body)
70+
this.body = parseBody(this.body)
6571
}
6672

6773
// Extract path from event (strip querystring just in case)
6874
let path = app._event.path.trim().split('?')[0].replace(/^\/(.*?)(\/)*$/,'$1').split('/')
6975

70-
// Remove base if it exists
71-
if (app._base && app._base === path[0]) {
72-
path.shift()
73-
} // end remove base
74-
7576
// Init the route
7677
this.route = null
7778

0 commit comments

Comments
 (0)