// FunctionsRouter.js var Parse = require('parse/node').Parse, triggers = require('../triggers'); import PromiseRouter from '../PromiseRouter'; import { promiseEnforceMasterKeyAccess } from '../middlewares'; import { jobStatusHandler } from '../StatusHandler'; import _ from 'lodash'; import { logger } from '../logger'; function parseObject(obj) { if (Array.isArray(obj)) { return obj.map((item) => { return parseObject(item); }); } else if (obj && obj.__type == 'Date') { return Object.assign(new Date(obj.iso), obj); } else if (obj && obj.__type == 'File') { return Parse.File.fromJSON(obj); } else if (obj && typeof obj === 'object') { return parseParams(obj); } else { return obj; } } function parseParams(params) { return _.mapValues(params, parseObject); } export class FunctionsRouter extends PromiseRouter { mountRoutes() { this.route('POST', '/functions/:functionName', FunctionsRouter.handleCloudFunction); this.route('POST', '/jobs/:jobName', promiseEnforceMasterKeyAccess, function(req) { return FunctionsRouter.handleCloudJob(req); }); this.route('POST', '/jobs', promiseEnforceMasterKeyAccess, function(req) { return FunctionsRouter.handleCloudJob(req); }); } static handleCloudJob(req) { const jobName = req.params.jobName || req.body.jobName; const applicationId = req.config.applicationId; const jobHandler = jobStatusHandler(req.config); const jobFunction = triggers.getJob(jobName, applicationId); if (!jobFunction) { throw new Parse.Error(Parse.Error.SCRIPT_FAILED, 'Invalid job.'); } let params = Object.assign({}, req.body, req.query); params = parseParams(params); const request = { params: params, log: req.config.loggerController, headers: req.config.headers, ip: req.config.ip, jobName, message: jobHandler.setMessage.bind(jobHandler) }; return jobHandler.setRunning(jobName, params).then((jobStatus) => { request.jobId = jobStatus.objectId // run the function async process.nextTick(() => { Promise.resolve().then(() => { return jobFunction(request); }).then((result) => { jobHandler.setSucceeded(result); }, (error) => { jobHandler.setFailed(error); }); }); return { headers: { 'X-Parse-Job-Status-Id': jobStatus.objectId }, response: {} } }); } static createResponseObject(resolve, reject, message) { return { success: function(result) { resolve({ response: { result: Parse._encode(result) } }); }, error: function(message) { // parse error, process away if (message instanceof Parse.Error) { return reject(message); } const code = Parse.Error.SCRIPT_FAILED; // If it's an error, mark it as a script failed if (typeof message === 'string') { return reject(new Parse.Error(code, message)); } if (message instanceof Error) { message = message.message; } reject(new Parse.Error(code, message)); }, message: message } } static handleCloudFunction(req) { const functionName = req.params.functionName; const applicationId = req.config.applicationId; const theFunction = triggers.getFunction(functionName, applicationId); const theValidator = triggers.getValidator(req.params.functionName, applicationId); if (!theFunction) { throw new Parse.Error(Parse.Error.SCRIPT_FAILED, `Invalid function: "${functionName}"`); } let params = Object.assign({}, req.body, req.query); params = parseParams(params); const request = { params: params, master: req.auth && req.auth.isMaster, user: req.auth && req.auth.user, installationId: req.info.installationId, log: req.config.loggerController, headers: req.config.headers, ip: req.config.ip, functionName }; if (theValidator && typeof theValidator === "function") { var result = theValidator(request); if (!result) { throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Validation failed.'); } } return new Promise(function (resolve, reject) { const userString = (req.auth && req.auth.user) ? req.auth.user.id : undefined; const cleanInput = logger.truncateLogMessage(JSON.stringify(params)); const { success, error, message } = FunctionsRouter.createResponseObject((result) => { try { const cleanResult = logger.truncateLogMessage(JSON.stringify(result.response.result)); logger.info( `Ran cloud function ${functionName} for user ${userString} with:\n Input: ${cleanInput }\n Result: ${cleanResult }`, { functionName, params, user: userString, } ); resolve(result); } catch (e) { reject(e); } }, (error) => { try { logger.error( `Failed running cloud function ${functionName} for user ${userString} with:\n Input: ${cleanInput}\n Error: ` + JSON.stringify(error), { functionName, error, params, user: userString } ); reject(error); } catch (e) { reject(e); } }); return Promise.resolve().then(() => { return theFunction(request, { message }); }).then(success, error); }); } }