/** * Copyright JS Foundation and other contributors, http://js.foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. **/ var clone = require("clone"); var jsonata = require("jsonata"); var safeJSONStringify = require("json-stringify-safe"); var util = require("util"); function generateId() { return (1+Math.random()*4294967295).toString(16); } function ensureString(o) { if (Buffer.isBuffer(o)) { return o.toString(); } else if (typeof o === "object") { return JSON.stringify(o); } else if (typeof o === "string") { return o; } return ""+o; } function ensureBuffer(o) { if (Buffer.isBuffer(o)) { return o; } else if (typeof o === "object") { o = JSON.stringify(o); } else if (typeof o !== "string") { o = ""+o; } return new Buffer(o); } function cloneMessage(msg) { // Temporary fix for #97 // TODO: remove this http-node-specific fix somehow var req = msg.req; var res = msg.res; delete msg.req; delete msg.res; var m = clone(msg); if (req) { m.req = req; msg.req = req; } if (res) { m.res = res; msg.res = res; } return m; } function compareObjects(obj1,obj2) { var i; if (obj1 === obj2) { return true; } if (obj1 == null || obj2 == null) { return false; } var isArray1 = Array.isArray(obj1); var isArray2 = Array.isArray(obj2); if (isArray1 != isArray2) { return false; } if (isArray1 && isArray2) { if (obj1.length !== obj2.length) { return false; } for (i=0;i'); expr._legacyMode = /(^|[^a-zA-Z0-9_'"])msg([^a-zA-Z0-9_'"]|$)/.test(value); expr._node = node; return expr; } function evaluateJSONataExpression(expr,msg,callback) { var context = msg; if (expr._legacyMode) { context = {msg:msg}; } var bindings = {}; if (callback) { // If callback provided, need to override the pre-assigned sync // context functions to be their async variants bindings.flowContext = function(val, store) { return new Promise((resolve,reject) => { expr._node.context().flow.get(val, store, function(err,value) { if (err) { reject(err); } else { resolve(value); } }) }); } bindings.globalContext = function(val, store) { return new Promise((resolve,reject) => { expr._node.context().global.get(val, store, function(err,value) { if (err) { reject(err); } else { resolve(value); } }) }); } } return expr.evaluate(context, bindings, callback); } function normaliseNodeTypeName(name) { var result = name.replace(/[^a-zA-Z0-9]/g, " "); result = result.trim(); result = result.replace(/ +/g, " "); result = result.replace(/ ./g, function(s) { return s.charAt(1).toUpperCase(); } ); result = result.charAt(0).toLowerCase() + result.slice(1); return result; } function encodeObject(msg,opts) { var debuglength = 1000; if (opts && opts.hasOwnProperty('maxLength')) { debuglength = opts.maxLength; } var msgType = typeof msg.msg; if (msg.msg instanceof Error) { msg.format = "error"; var errorMsg = {}; if (msg.msg.name) { errorMsg.name = msg.msg.name; } if (msg.msg.hasOwnProperty('message')) { errorMsg.message = msg.msg.message; } else { errorMsg.message = msg.msg.toString(); } msg.msg = JSON.stringify(errorMsg); } else if (msg.msg instanceof Buffer) { msg.format = "buffer["+msg.msg.length+"]"; msg.msg = msg.msg.toString('hex'); if (msg.msg.length > debuglength) { msg.msg = msg.msg.substring(0,debuglength); } } else if (msg.msg && msgType === 'object') { try { msg.format = msg.msg.constructor.name || "Object"; // Handle special case of msg.req/res objects from HTTP In node if (msg.format === "IncomingMessage" || msg.format === "ServerResponse") { msg.format = "Object"; } } catch(err) { msg.format = "Object"; } if (/error/i.test(msg.format)) { msg.msg = JSON.stringify({ name: msg.msg.name, message: msg.msg.message }); } else { var isArray = util.isArray(msg.msg); if (isArray) { msg.format = "array["+msg.msg.length+"]"; if (msg.msg.length > debuglength) { // msg.msg = msg.msg.slice(0,debuglength); msg.msg = { __enc__: true, type: "array", data: msg.msg.slice(0,debuglength), length: msg.msg.length } } } if (isArray || (msg.format === "Object")) { msg.msg = safeJSONStringify(msg.msg, function(key, value) { if (key === '_req' || key === '_res') { value = { __enc__: true, type: "internal" } } else if (value instanceof Error) { value = value.toString() } else if (util.isArray(value) && value.length > debuglength) { value = { __enc__: true, type: "array", data: value.slice(0,debuglength), length: value.length } } else if (typeof value === 'string') { if (value.length > debuglength) { value = value.substring(0,debuglength)+"..."; } } else if (typeof value === 'function') { value = { __enc__: true, type: "function" } } else if (typeof value === 'number') { if (isNaN(value) || value === Infinity || value === -Infinity) { value = { __enc__: true, type: "number", data: value.toString() } } } else if (value && value.constructor) { if (value.type === "Buffer") { value.__enc__ = true; value.length = value.data.length; if (value.length > debuglength) { value.data = value.data.slice(0,debuglength); } } else if (value.constructor.name === "ServerResponse") { value = "[internal]" } else if (value.constructor.name === "Socket") { value = "[internal]" } } return value; }," "); } else { try { msg.msg = msg.msg.toString(); } catch(e) { msg.msg = "[Type not printable]"; } } } } else if (msgType === "function") { msg.format = "function"; msg.msg = "[function]" } else if (msgType === "boolean") { msg.format = "boolean"; msg.msg = msg.msg.toString(); } else if (msgType === "number") { msg.format = "number"; msg.msg = msg.msg.toString(); } else if (msg.msg === null || msgType === "undefined") { msg.format = (msg.msg === null)?"null":"undefined"; msg.msg = "(undefined)"; } else { msg.format = "string["+msg.msg.length+"]"; if (msg.msg.length > debuglength) { msg.msg = msg.msg.substring(0,debuglength)+"..."; } } return msg; } module.exports = { encodeObject: encodeObject, ensureString: ensureString, ensureBuffer: ensureBuffer, cloneMessage: cloneMessage, compareObjects: compareObjects, generateId: generateId, getMessageProperty: getMessageProperty, setMessageProperty: setMessageProperty, getObjectProperty: getObjectProperty, setObjectProperty: setObjectProperty, evaluateNodeProperty: evaluateNodeProperty, normalisePropertyExpression: normalisePropertyExpression, normaliseNodeTypeName: normaliseNodeTypeName, prepareJSONataExpression: prepareJSONataExpression, evaluateJSONataExpression: evaluateJSONataExpression, parseContextStore: parseContextStore };