304 lines
7.0 KiB
JavaScript
304 lines
7.0 KiB
JavaScript
'use strict';
|
|
|
|
var isArray = require('lodash/isArray');
|
|
var isObject = require('lodash/isObject');
|
|
|
|
var DROPDOWN_TYPE = 'Dropdown';
|
|
|
|
var VALID_TYPES = [ 'String', 'Text', 'Boolean', 'Hidden', DROPDOWN_TYPE ];
|
|
|
|
var PROPERTY_TYPE = 'property',
|
|
ACTIVITI_PROPERTY_TYPE = 'activiti:property',
|
|
ACTIVITI_INPUT_PARAMETER_TYPE = 'activiti:inputParameter',
|
|
ACTIVITI_OUTPUT_PARAMETER_TYPE = 'activiti:outputParameter',
|
|
ACTIVITI_IN_TYPE = 'activiti:in',
|
|
ACTIVITI_OUT_TYPE = 'activiti:out',
|
|
ACTIVITI_IN_BUSINESS_KEY_TYPE = 'activiti:in:businessKey',
|
|
ACTIVITI_EXECUTION_LISTENER = 'activiti:executionListener',
|
|
ACTIVITI_FIELD = 'activiti:field';
|
|
|
|
var VALID_BINDING_TYPES = [
|
|
PROPERTY_TYPE,
|
|
ACTIVITI_PROPERTY_TYPE,
|
|
ACTIVITI_INPUT_PARAMETER_TYPE,
|
|
ACTIVITI_OUTPUT_PARAMETER_TYPE,
|
|
ACTIVITI_IN_TYPE,
|
|
ACTIVITI_OUT_TYPE,
|
|
ACTIVITI_IN_BUSINESS_KEY_TYPE,
|
|
ACTIVITI_EXECUTION_LISTENER,
|
|
ACTIVITI_FIELD
|
|
];
|
|
|
|
|
|
/**
|
|
* A element template validator.
|
|
*/
|
|
function Validator() {
|
|
|
|
this._templatesById = {};
|
|
|
|
this._validTemplates = [];
|
|
this._errors = [];
|
|
|
|
|
|
/**
|
|
* Adds the templates.
|
|
*
|
|
* @param {Array<TemplateDescriptor>} templates
|
|
*
|
|
* @return {Validator} self
|
|
*/
|
|
this.addAll = function(templates) {
|
|
|
|
if (!isArray(templates)) {
|
|
this._logError('templates must be []');
|
|
} else {
|
|
templates.forEach(this.add, this);
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Add the given element template, if it is valid.
|
|
*
|
|
* @param {TemplateDescriptor} template
|
|
*
|
|
* @return {Validator} self
|
|
*/
|
|
this.add = function(template) {
|
|
|
|
var err = this._validateTemplate(template);
|
|
|
|
if (!err) {
|
|
this._templatesById[template.id] = template;
|
|
|
|
this._validTemplates.push(template);
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Validate given template and return error (if any).
|
|
*
|
|
* @param {TemplateDescriptor} template
|
|
*
|
|
* @return {Error} validation error, if any
|
|
*/
|
|
this._validateTemplate = function(template) {
|
|
|
|
var err,
|
|
id = template.id,
|
|
appliesTo = template.appliesTo,
|
|
properties = template.properties,
|
|
scopes = template.scopes;
|
|
|
|
if (!id) {
|
|
return this._logError('missing template id');
|
|
}
|
|
|
|
if (id in this._templatesById) {
|
|
return this._logError('template id <' + id + '> already used');
|
|
}
|
|
|
|
if (!isArray(appliesTo)) {
|
|
err = this._logError('missing appliesTo=[]', template);
|
|
}
|
|
|
|
if (!isArray(properties)) {
|
|
err = this._logError('missing properties=[]', template);
|
|
} else {
|
|
if (!this._validateProperties(properties)) {
|
|
err = new Error('invalid properties');
|
|
}
|
|
}
|
|
|
|
if (scopes) {
|
|
err = this._validateScopes(template, scopes);
|
|
}
|
|
|
|
return err;
|
|
};
|
|
|
|
this._validateScopes = function(template, scopes) {
|
|
|
|
var err,
|
|
scope,
|
|
scopeName;
|
|
|
|
if (!isObject(scopes) || isArray(scopes)) {
|
|
return this._logError('invalid scopes, should be scopes={}', template);
|
|
}
|
|
|
|
for (scopeName in scopes) {
|
|
scope = scopes[scopeName];
|
|
|
|
if (!isObject(scope) || isArray(scope)) {
|
|
err = this._logError('invalid scope, should be scope={}', template);
|
|
}
|
|
|
|
if (!isArray(scope.properties)) {
|
|
err = this._logError(
|
|
'missing properties=[] in scope <' + scopeName + '>', template
|
|
);
|
|
} else {
|
|
if (!this._validateProperties(scope.properties)) {
|
|
err = new Error('invalid properties in scope <' + scopeName + '>');
|
|
}
|
|
}
|
|
}
|
|
|
|
return err;
|
|
};
|
|
|
|
/**
|
|
* Validate properties and return false if any is invalid.
|
|
*
|
|
* @param {Array<PropertyDescriptor>} properties
|
|
*
|
|
* @return {Boolean} true if all properties are valid
|
|
*/
|
|
this._validateProperties = function(properties) {
|
|
var validProperties = properties.filter(this._validateProperty, this);
|
|
|
|
return properties.length === validProperties.length;
|
|
};
|
|
|
|
/**
|
|
* Validate property and return false, if there was
|
|
* a validation error.
|
|
*
|
|
* @param {PropertyDescriptor} property
|
|
*
|
|
* @return {Boolean} true if property is valid
|
|
*/
|
|
this._validateProperty = function(property) {
|
|
|
|
var type = property.type,
|
|
binding = property.binding;
|
|
|
|
var err;
|
|
|
|
var bindingType = binding.type;
|
|
|
|
if (VALID_TYPES.indexOf(type) === -1) {
|
|
err = this._logError(
|
|
'invalid property type <' + type + '>; ' +
|
|
'must be any of { ' + VALID_TYPES.join(', ') + ' }'
|
|
);
|
|
}
|
|
|
|
if (type === DROPDOWN_TYPE && bindingType !== ACTIVITI_EXECUTION_LISTENER) {
|
|
if (!isArray(property.choices)) {
|
|
err = this._logError(
|
|
'must provide choices=[] with ' + DROPDOWN_TYPE + ' type'
|
|
);
|
|
} else
|
|
|
|
if (!property.choices.every(isDropdownChoiceValid)) {
|
|
err = this._logError(
|
|
'{ name, value } must be specified for ' +
|
|
DROPDOWN_TYPE + ' choices'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (!binding) {
|
|
return this._logError('property missing binding');
|
|
}
|
|
|
|
if (VALID_BINDING_TYPES.indexOf(bindingType) === -1) {
|
|
err = this._logError(
|
|
'invalid property.binding type <' + bindingType + '>; ' +
|
|
'must be any of { ' + VALID_BINDING_TYPES.join(', ') + ' }'
|
|
);
|
|
}
|
|
|
|
if (bindingType === PROPERTY_TYPE ||
|
|
bindingType === ACTIVITI_PROPERTY_TYPE ||
|
|
bindingType === ACTIVITI_INPUT_PARAMETER_TYPE ||
|
|
bindingType === ACTIVITI_FIELD) {
|
|
|
|
if (!binding.name) {
|
|
err = this._logError(
|
|
'property.binding <' + bindingType + '> requires name'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (bindingType === ACTIVITI_OUTPUT_PARAMETER_TYPE) {
|
|
if (!binding.source) {
|
|
err = this._logError(
|
|
'property.binding <' + bindingType + '> requires source'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (bindingType === ACTIVITI_IN_TYPE) {
|
|
|
|
if (!binding.variables && !binding.target) {
|
|
err = this._logError(
|
|
'property.binding <' + bindingType + '> requires ' +
|
|
'variables or target'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (bindingType === ACTIVITI_OUT_TYPE) {
|
|
|
|
if (!binding.variables && !binding.source && !binding.sourceExpression) {
|
|
err = this._logError(
|
|
'property.binding <' + bindingType + '> requires ' +
|
|
'variables, sourceExpression or source'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (bindingType === ACTIVITI_EXECUTION_LISTENER) {
|
|
|
|
if (type !== 'Hidden') {
|
|
err = this._logError(
|
|
'invalid property type <' + type + '> for ' + ACTIVITI_EXECUTION_LISTENER + '; ' +
|
|
'must be <Hidden>'
|
|
);
|
|
}
|
|
}
|
|
|
|
return !err;
|
|
};
|
|
|
|
|
|
this._logError = function(err, template) {
|
|
|
|
if (typeof err === 'string') {
|
|
if (template) {
|
|
err = 'template(id: ' + template.id + ') ' + err;
|
|
}
|
|
|
|
err = new Error(err);
|
|
}
|
|
|
|
this._errors.push(err);
|
|
|
|
return err;
|
|
};
|
|
|
|
this.getErrors = function() {
|
|
return this._errors;
|
|
};
|
|
|
|
this.getValidTemplates = function() {
|
|
return this._validTemplates;
|
|
};
|
|
}
|
|
|
|
module.exports = Validator;
|
|
|
|
|
|
// helpers ///////////////////////////////////
|
|
|
|
function isDropdownChoiceValid(c) {
|
|
return 'name' in c && 'value' in c;
|
|
}
|