import { IVProgParser } from "./../ast/ivprogParser"; import { SemanticAnalyser } from "./../processor/semantic/semanticAnalyser"; import { IVProgProcessor } from "./../processor/ivprogProcessor"; import { InputTest } from "./../util/inputTest"; import { OutputTest } from "./../util/outputTest"; import * as LocalizedStringsService from "../services/localizedStringsService"; import { Decimal } from 'decimal.js'; import { Config } from "../util/config"; const LocalizedStrings = LocalizedStringsService.getInstance(); export class IVProgAssessment { constructor (textCode, testCases, domConsole) { this.textCode = textCode; this.testCases = testCases; this.domConsole = domConsole; } runTest () { try { // try and show error messages through domconsole const parser = IVProgParser.createParser(this.textCode); const semantic = new SemanticAnalyser(parser.parseTree()); const validTree = semantic.analyseTree(); // loop test cases and show messages through domconsole const partialTests = this.testCases.map( (t, name) => { return this.partialEvaluateTestCase(new IVProgProcessor(validTree), t.input, t.output, name); }); const testResult = partialTests.reduce((acc, curr) => acc.then(curr), Promise.resolve(0)); return testResult.then(total => Promise.resolve(total / this.testCases.length)) .catch(err => { this.domConsole.err("Erro durante a execução do programa");// try and show error messages through domconsole this.domConsole.err(err.message); return Promise.resolve(0); }); } catch (error) { this.domConsole.err("Erro durante a execução do programa");// try and show error messages through domconsole this.domConsole.err(error.message); return Promise.resolve(0); } } evaluateTestCase (prog, inputList, outputList, name, accumulator) { const outerThis = this; const input = new InputTest(inputList); const output = new OutputTest(); prog.registerInput(input); prog.registerOutput(output); const startTime = Date.now() return prog.interpretAST().then( _ => { const millis = Date.now() - startTime; if (input.inputList.length !== input.index) { outerThis.showErrorMessage('test_case_few_reads', name+1); outerThis.showMessage('test_case_duration', millis); return Promise.resolve(accumulator + 1 * (input.index/inputList.length)); } else if (output.list.length < outputList.length) { outerThis.showErrorMessage('test_case_failed', name + 1, inputList.join(','), outputList.join(','), output.list.join(',')); outerThis.showMessage('test_case_duration', millis); return Promise.resolve(accumulator + 1 * (output.list.length/outputList.length)); } else if (output.list.length > outputList.length) { outerThis.showErrorMessage('test_case_failed', name + 1, inputList.join(','), outputList.join(','), output.list.join(',')); outerThis.showMessage('test_case_duration', millis); return Promise.resolve(accumulator + 1 * (outputList.length/output.list.length)); } else { const isOk = outerThis.checkOutput(output.list, outputList); if(!isOk) { outerThis.showErrorMessage('test_case_failed', name + 1, inputList.join(','), outputList.join(','), output.list.join(',')); outerThis.showMessage('test_case_duration', millis); return Promise.resolve(accumulator); } else { outerThis.showMessage('test_case_success', name + 1); outerThis.showMessage('test_case_duration', millis); return Promise.resolve(accumulator + 1); } } }).catch( error => { outerThis.showErrorMessage('test_case_failed_exception', name + 1, error.message); return Promise.resolve(accumulator); }); } partialEvaluateTestCase (prog, inputList, outputList, name) { return this.evaluateTestCase.bind(this, prog, inputList, outputList, name); } checkOutput (aList, bList) { for (let i = 0; i < aList.length; i++) { const outValue = aList[i]; let castNumberA = parseFloat(outValue); if(!Number.isNaN(castNumberA)) { let castNumberB = parseFloat(bList[i]); if(Number.isNaN(castNumberB)) { return false; } castNumberA = new Decimal(castNumberA); castNumberB = new Decimal(castNumberB); const decimalPlaces = Math.min(castNumberB.dp(), Config.decimalPlaces); Decimal.set({ rounding: Decimal.ROUND_FLOOR}); castNumberA = new Decimal(castNumberA.toFixed(decimalPlaces)); castNumberB = new Decimal(castNumberB.toFixed(decimalPlaces)); const aEqualsB = castNumberA.eq(castNumberB); Decimal.set({ rounding: Decimal.ROUND_HALF_UP}); if (!aEqualsB) { return false; } } else if(outValue != bList[i]) { return false; } } return true; } showErrorMessage (errorID, ...args) { this.domConsole.err(LocalizedStrings.getError(errorID, args)); } showMessage (msgID, ...args) { this.domConsole.info(LocalizedStrings.getMessage(msgID, args)); } }