Skip to content
This repository has been archived by the owner on Sep 12, 2019. It is now read-only.

Commit

Permalink
Fix async and callback combo handling (#166)
Browse files Browse the repository at this point in the history
Fix async and callback combo handling
  • Loading branch information
swyxio authored Apr 25, 2019
2 parents c4477da + f3e7620 commit 1f37157
Show file tree
Hide file tree
Showing 3 changed files with 90 additions and 58 deletions.
148 changes: 90 additions & 58 deletions src/utils/serve-functions.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ const path = require("path");
const getPort = require("get-port");
const chokidar = require("chokidar");
const jwtDecode = require("jwt-decode");
// const chalk = require("chalk");
const chalk = require("chalk");
const {
NETLIFYDEVLOG,
// NETLIFYDEVWARN,
Expand All @@ -27,60 +27,10 @@ function handleErr(err, response) {
console.log(`${NETLIFYDEVERR} Error during invocation: `, err); // eslint-disable-line no-console
}

function createCallback(response) {
return function(err, lambdaResponse) {
if (err) {
return handleErr(err, response);
}
if (!Number(lambdaResponse.statusCode)) {
console.log(
`${NETLIFYDEVERR} Your function response must have a numerical statusCode. You gave: $`,
lambdaResponse.statusCode
);
return handleErr("Incorrect function response statusCode", response);
}
if (typeof lambdaResponse.body !== "string") {
console.log(
`${NETLIFYDEVERR} Your function response must have a string body. You gave:`,
lambdaResponse.body
);
return handleErr("Incorrect function response body", response);
}

response.statusCode = lambdaResponse.statusCode;
// eslint-disable-line guard-for-in
for (const key in lambdaResponse.headers) {
response.setHeader(key, lambdaResponse.headers[key]);
}
response.write(
lambdaResponse.isBase64Encoded
? Buffer.from(lambdaResponse.body, "base64")
: lambdaResponse.body
);
response.end();
};
}

function promiseCallback(promise, callback) {
if (!promise) return;
if (typeof promise.then !== "function") return;
if (typeof callback !== "function") return;

promise.then(
function(data) {
callback(null, data);
},
function(err) {
callback(err, null);
}
);
}

// function getHandlerPath(functionPath) {
// if (functionPath.match(/\.js$/)) {
// return functionPath;
// }

// return path.join(functionPath, `${path.basename(functionPath)}.js`);
// }

Expand Down Expand Up @@ -130,7 +80,12 @@ function createHandler(dir) {

Object.keys(functions).forEach(name => {
const fn = functions[name];
const clearCache = () => {
const clearCache = action => () => {
console.log(
`${NETLIFYDEVLOG} function ${chalk.yellow(
name
)} ${action}, reloading...`
); // eslint-disable-line no-console
const before = module.paths;
module.paths = [fn.moduleDir];
delete require.cache[require.resolve(fn.functionPath)];
Expand All @@ -144,9 +99,9 @@ function createHandler(dir) {
ignored: /node_modules/
});
fn.watcher
.on("add", clearCache)
.on("change", clearCache)
.on("unlink", clearCache);
.on("add", clearCache("added"))
.on("change", clearCache("modified"))
.on("unlink", clearCache("deleted"));
});

return function(request, response) {
Expand All @@ -167,6 +122,11 @@ function createHandler(dir) {
try {
module.paths = [moduleDir];
handler = require(functionPath);
if (typeof handler.handler !== "function") {
throw new Error(
`function ${functionPath} must export a function named handler`
);
}
module.paths = before;
} catch (error) {
module.paths = before;
Expand All @@ -180,7 +140,7 @@ function createHandler(dir) {
if (body instanceof Buffer) {
isBase64Encoded = true;
body = body.toString("base64");
} else if(typeof(body) === "string") {
} else if (typeof body === "string") {
// body is already processed as string
} else {
body = "";
Expand All @@ -195,24 +155,96 @@ function createHandler(dir) {
isBase64Encoded: isBase64Encoded
};

let callbackWasCalled = false;
const callback = createCallback(response);
const promise = handler.handler(
lambdaRequest,
{ clientContext: buildClientContext(request.headers) || {} },
callback
);
promiseCallback(promise, callback);
/** guard against using BOTH async and callback */
if (callbackWasCalled && promise && typeof promise.then === "function") {
throw new Error(
"Error: your function seems to be using both a callback and returning a promise (aka async function). This is invalid, pick one. (Hint: async!)"
);
} else {
// it is definitely an async function with no callback called, good.
promiseCallback(promise, callback);
}

/** need to keep createCallback in scope so we can know if cb was called AND handler is async */
function createCallback(response) {
return function(err, lambdaResponse) {
callbackWasCalled = true;
if (err) {
return handleErr(err, response);
}
if (lambdaResponse === undefined) {
return handleErr(
"lambda response was undefined. check your function code again.",
response
);
}
if (!Number(lambdaResponse.statusCode)) {
console.log(
`${NETLIFYDEVERR} Your function response must have a numerical statusCode. You gave: $`,
lambdaResponse.statusCode
);
return handleErr("Incorrect function response statusCode", response);
}
if (typeof lambdaResponse.body !== "string") {
console.log(
`${NETLIFYDEVERR} Your function response must have a string body. You gave:`,
lambdaResponse.body
);
return handleErr("Incorrect function response body", response);
}

response.statusCode = lambdaResponse.statusCode;
// eslint-disable-line guard-for-in
for (const key in lambdaResponse.headers) {
response.setHeader(key, lambdaResponse.headers[key]);
}
response.write(
lambdaResponse.isBase64Encoded
? Buffer.from(lambdaResponse.body, "base64")
: lambdaResponse.body
);
response.end();
};
}
};
}

function promiseCallback(promise, callback) {
if (!promise) return; // means no handler was written
if (typeof promise.then !== "function") return;
if (typeof callback !== "function") return;

promise.then(
function(data) {
console.log("hellooo");
callback(null, data);
},
function(err) {
callback(err, null);
}
);
}

async function serveFunctions(settings) {
const app = express();
const dir = settings.functionsDir;
const port = await getPort({
port: assignLoudly(settings.port, defaultPort)
});

app.use(bodyParser.text({ limit: "6mb", type: ["text/*", "application/json", "multipart/form-data"] }));
app.use(
bodyParser.text({
limit: "6mb",
type: ["text/*", "application/json", "multipart/form-data"]
})
);
app.use(bodyParser.raw({ limit: "6mb", type: "*/*" }));
app.use(
expressLogging(console, {
Expand Down

0 comments on commit 1f37157

Please sign in to comment.