@oat-sa/tao-item-runner-qti
Version:
TAO QTI Item Runner modules
121 lines (105 loc) • 5.58 kB
JavaScript
define(['lodash', 'taoQtiItem/scoring/processor/expressions/preprocessor', 'taoQtiItem/scoring/processor/errorHandler', 'taoQtiItem/scoring/processor/expressions/operators/constraintValidator'], function (_, preProcessorFactory, errorHandler, constraintValidator) { 'use strict';
_ = _ && Object.prototype.hasOwnProperty.call(_, 'default') ? _['default'] : _;
preProcessorFactory = preProcessorFactory && Object.prototype.hasOwnProperty.call(preProcessorFactory, 'default') ? preProcessorFactory['default'] : preProcessorFactory;
errorHandler = errorHandler && Object.prototype.hasOwnProperty.call(errorHandler, 'default') ? errorHandler['default'] : errorHandler;
constraintValidator = constraintValidator && Object.prototype.hasOwnProperty.call(constraintValidator, 'default') ? constraintValidator['default'] : constraintValidator;
/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; under version 2
* of the License (non-upgradable).
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* Copyright (c) 2015-2019 (original work) Open Assessment Technlogies SA (under the project TAO-PRODUCT);
*
*/
/**
* All values handled by a processor
* @typedef ProcessingValue
* @property {String} baseType - the value type in 'identifier', 'boolean', 'integer', 'float', 'string', 'point', 'pair', 'directedPair', 'duration', 'file', 'uri' and 'intOrIdentifier'
* @property {String} cardinality - the value cardinality in 'single', 'multiple', 'ordered' and 'record'
* @property {*} value - the real value
*/
//keeps the references of processors (this is something we may load dynamically)
var processors = {};
/**
* Provides you a processor from an expression (definition) and against a state.
* The Processor must have been registerd previously.
* It will also validate the processor against the given expression and context.
* See it as a kind of factory.
*
* @param {Object} expression - the expression definition
* @param {String} expression.qtiClass - the expression name that should mastch with the processor name
* @param {Object} state - the state to give to the processor to lookup for variables like responses
* @param {Array} [operands] - the operands for an operator processor
* @returns {ExpressionProcessorWrapper} a transparent object that delegates the calls to the processor
* @throws {TypeError} when the validation fails
* @throws {Error} by trying to use an unregistered processor
*/
var expressionProcessor = function expressionProcessor(expression, state, operands) {
var name = expression.qtiClass;
var processor = processors.expression[name] || processors.operator[name];
if (!processor) {
return errorHandler.throw('scoring', new Error('No processor found for ' + name));
}
processor.expression = expression;
processor.preProcessor = preProcessorFactory(state);
processor.state = state;
processor.name = name;
//validate operators
if (processor.operands && processor.constraints) {
constraintValidator(processor, operands);
processor.operands = operands;
}
/**
* Wrap ad forward the processing to the enclosed processor
* @typedef ExpressionProcessorWrapper
* @property {Function} process - call's processor's process
*/
return {
process: function process() {
//forward the call to the related expression processor
return processor.process.apply(processor, [].slice.call(arguments, 1));
}
};
};
/**
*
*/
expressionProcessor.types = {
EXPRESSION: 'expression',
OPERATOR: 'operator'
};
/**
* Register a processor
* @param {String} name - the processor name
* @param {String} type - the processor type see expressionProcessor.types
* @param {ExpressionProcessor|OperatorProcessor} processor - the processor to register
* @throws {TypeError} when a parameter isn't valid
*/
expressionProcessor.register = function register(name, type, processor) {
if (!Object.values(this.types).includes(type)) {
return errorHandler.throw('scoring', new TypeError(type + ' is not a valid expression type'));
}
if (_.isEmpty(name)) {
return errorHandler.throw('scoring', new TypeError('Please give a valid name to your processor'));
}
if (!_.isPlainObject(processor) || !_.isFunction(processor.process)) {
return errorHandler.throw(
'scoring',
new TypeError('The processor must be an object that contains a process method.')
);
}
processors[type] = processors[type] || {};
processors[type][name] = processor;
};
return expressionProcessor;
});