2021-01-18 18:40:19 +01:00
|
|
|
const handlebars = require("handlebars")
|
|
|
|
const { registerAll } = require("./helpers/index")
|
|
|
|
|
|
|
|
const HBS_CLEANING_REGEX = /{{[^}}]*}}/g
|
2021-01-19 19:44:29 +01:00
|
|
|
const FIND_HBS_REGEX = /{{.*}}/
|
2021-01-18 18:40:19 +01:00
|
|
|
|
|
|
|
const hbsInstance = handlebars.create()
|
|
|
|
registerAll(hbsInstance)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When running handlebars statements to execute on the context of the automation it possible user's may input handlebars
|
|
|
|
* in a few different forms, some of which are invalid but are logically valid. An example of this would be the handlebars
|
|
|
|
* statement "{{steps[0].revision}}" here it is obvious the user is attempting to access an array or object using array
|
|
|
|
* like operators. These are not supported by handlebars and therefore the statement will fail. This function will clean up
|
|
|
|
* the handlebars statement so it instead reads as "{{steps.0.revision}}" which is valid and will work. It may also be expanded
|
|
|
|
* to include any other handlebars statement cleanup that has been deemed necessary for the system.
|
|
|
|
*
|
|
|
|
* @param {string} string The string which *may* contain handlebars statements, it is OK if it does not contain any.
|
|
|
|
* @returns {string} The string that was input with cleaned up handlebars statements as required.
|
|
|
|
*/
|
|
|
|
function cleanHandlebars(string) {
|
2021-01-19 19:44:29 +01:00
|
|
|
// TODO: handle these types of statement
|
|
|
|
// every statement must have the "all" helper added e.g.
|
|
|
|
// {{ person }} => {{ html person }}
|
|
|
|
// escaping strings must be handled as such:
|
|
|
|
// {{ person name }} => {{ [person name] }}
|
|
|
|
// {{ obj.person name }} => {{ obj.[person name] }}
|
2021-01-18 18:40:19 +01:00
|
|
|
let charToReplace = {
|
|
|
|
"[": ".",
|
|
|
|
"]": "",
|
|
|
|
}
|
|
|
|
let regex = new RegExp(HBS_CLEANING_REGEX)
|
|
|
|
let matches = string.match(regex)
|
|
|
|
if (matches == null) {
|
|
|
|
return string
|
|
|
|
}
|
|
|
|
for (let match of matches) {
|
|
|
|
let baseIdx = string.indexOf(match)
|
|
|
|
for (let key of Object.keys(charToReplace)) {
|
|
|
|
let idxChar = match.indexOf(key)
|
|
|
|
if (idxChar !== -1) {
|
|
|
|
string =
|
|
|
|
string.slice(baseIdx, baseIdx + idxChar) +
|
|
|
|
charToReplace[key] +
|
|
|
|
string.slice(baseIdx + idxChar + 1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return string
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-20 14:32:15 +01:00
|
|
|
* utility function to check if the object is valid
|
2021-01-18 18:40:19 +01:00
|
|
|
*/
|
2021-01-20 14:32:15 +01:00
|
|
|
function testObject(object) {
|
2021-01-18 18:40:19 +01:00
|
|
|
// JSON stringify will fail if there are any cycles, stops infinite recursion
|
|
|
|
try {
|
|
|
|
JSON.stringify(object)
|
|
|
|
} catch (err) {
|
|
|
|
throw "Unable to process inputs to JSON, cannot recurse"
|
|
|
|
}
|
2021-01-20 14:32:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given an input object this will recurse through all props to try and update any handlebars statements within.
|
|
|
|
* @param {object|array} object The input structure which is to be recursed, it is important to note that
|
|
|
|
* if the structure contains any cycles then this will fail.
|
|
|
|
* @param {object} context The context that handlebars should fill data from.
|
|
|
|
* @returns {Promise<object|array>} The structure input, as fully updated as possible.
|
|
|
|
*/
|
|
|
|
module.exports.processObject = async (object, context) => {
|
|
|
|
testObject(object)
|
|
|
|
// TODO: carry out any async calls before carrying out async call
|
2021-01-18 18:40:19 +01:00
|
|
|
for (let key of Object.keys(object)) {
|
|
|
|
let val = object[key]
|
|
|
|
if (typeof val === "string") {
|
2021-01-20 14:32:15 +01:00
|
|
|
object[key] = await module.exports.processString(object[key], context)
|
2021-01-18 18:40:19 +01:00
|
|
|
}
|
|
|
|
else if (typeof val === "object") {
|
2021-01-20 14:32:15 +01:00
|
|
|
object[key] = await module.exports.processObject(object[key], context)
|
2021-01-18 18:40:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return object
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This will process a single handlebars containing string. If the string passed in has no valid handlebars statements
|
|
|
|
* then nothing will occur.
|
|
|
|
* @param {string} string The template string which is the filled from the context object.
|
|
|
|
* @param {object} context An object of information which will be used to enrich the string.
|
2021-01-20 14:32:15 +01:00
|
|
|
* @returns {Promise<string>} The enriched string, all templates should have been replaced if they can be.
|
|
|
|
*/
|
|
|
|
module.exports.processString = async (string, context) => {
|
|
|
|
// TODO: carry out any async calls before carrying out async call
|
|
|
|
return module.exports.processStringSync(string, context)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given an input object this will recurse through all props to try and update any handlebars statements within. This is
|
|
|
|
* a pure sync call and therefore does not have the full functionality of the async call.
|
|
|
|
* @param {object|array} object The input structure which is to be recursed, it is important to note that
|
|
|
|
* if the structure contains any cycles then this will fail.
|
|
|
|
* @param {object} context The context that handlebars should fill data from.
|
|
|
|
* @returns {object|array} The structure input, as fully updated as possible.
|
|
|
|
*/
|
|
|
|
module.exports.processObjectSync = (object, context) => {
|
|
|
|
testObject(object)
|
|
|
|
for (let key of Object.keys(object)) {
|
|
|
|
let val = object[key]
|
|
|
|
if (typeof val === "string") {
|
|
|
|
object[key] = module.exports.processStringSync(object[key], context)
|
|
|
|
}
|
|
|
|
else if (typeof val === "object") {
|
|
|
|
object[key] = module.exports.processObjectSync(object[key], context)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return object
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This will process a single handlebars containing string. If the string passed in has no valid handlebars statements
|
|
|
|
* then nothing will occur. This is a pure sync call and therefore does not have the full functionality of the async call.
|
|
|
|
* @param {string} string The template string which is the filled from the context object.
|
|
|
|
* @param {object} context An object of information which will be used to enrich the string.
|
2021-01-18 18:40:19 +01:00
|
|
|
* @returns {string} The enriched string, all templates should have been replaced if they can be.
|
|
|
|
*/
|
2021-01-20 14:32:15 +01:00
|
|
|
module.exports.processStringSync = (string, context) => {
|
2021-01-19 18:29:38 +01:00
|
|
|
if (typeof string !== "string") {
|
|
|
|
throw "Cannot process non-string types."
|
|
|
|
}
|
2021-01-19 16:29:49 +01:00
|
|
|
let template
|
2021-01-20 14:32:15 +01:00
|
|
|
string = cleanHandlebars(string)
|
|
|
|
// this does not throw an error when template can't be fulfilled, have to try correct beforehand
|
|
|
|
template = hbsInstance.compile(string)
|
2021-01-18 18:40:19 +01:00
|
|
|
return template(context)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Errors can occur if a user of this library attempts to use a helper that has not been added to the system, these errors
|
|
|
|
* can be captured to alert the user of the mistake.
|
|
|
|
* @param {function} handler a function which will be called every time an error occurs when processing a handlebars
|
2021-01-20 14:32:15 +01:00
|
|
|
* statement.
|
2021-01-18 18:40:19 +01:00
|
|
|
*/
|
|
|
|
module.exports.errorEvents = handler => {
|
|
|
|
hbsInstance.registerHelper("helperMissing", handler)
|
|
|
|
}
|