| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194 | import { CommonTokenStream, InputStream } from 'antlr4/index';import * as Expressions from './expressions/';import * as Commands from './commands/';import { recover } from "./ivprogLexer";import { toInt, toString, toBool, toReal } from './../typeSystem/parsers';import { Types } from "./../typeSystem/types";import { CompoundType } from "./../typeSystem/compoundType";import { SourceInfo } from './sourceInfo';import { convertFromString } from './operators';import { SyntaxErrorFactory } from './error/syntaxErrorFactory';import { LanguageDefinedFunction } from './../processor/definedFunctions';import { LanguageService } from '../services/languageService';export class IVProgParser {  static createParser (input) {    const lexerClass = LanguageService.getCurrentLexer();    return new IVProgParser(input, lexerClass);  }  // <BEGIN scope consts>  static get BASE () {    return 0;  }  static get FUNCTION () {    return 1;  }  static get COMMAND () {    return 2;  }  static get BREAKABLE () {    return 4;  }  // </ END scope consts>  constructor (input, lexerClass) {    this.lexerClass = lexerClass;    this.lexer = new lexerClass(new InputStream(input));    this.lexer.recover = recover.bind(this.lexer);    this.tokenStream = new CommonTokenStream(this.lexer);    this.tokenStream.fill();    this.pos = 1;    this.variableTypes = [this.lexerClass.RK_INTEGER,      this.lexerClass.RK_REAL,      this.lexerClass.RK_BOOLEAN,      this.lexerClass.RK_STRING    ];    this.functionTypes = this.variableTypes.concat(this.lexerClass.RK_VOID);    this.parsingArrayDimension = 0;    this.scope = [];    this.langFuncs = LanguageService.getCurrentLangFuncs();    this.definedFuncsNameList = [];    this.definedVariablesStack = [];  }  parseTree () {    return this.parseProgram();  }  getToken (index = this.pos) {    // if(index === null)    //   index = this.pos;    return this.tokenStream.LT(index);  }  insideScope (scope) {    if(this.scope.length <= 0) {      return IVProgParser.BASE === scope;    } else {      return this.scope[this.scope.length-1] === scope;    }  }  pushScope (scope) {    this.scope.push(scope);  }  pushVariableStack () {    this.definedVariablesStack.push([]);  }  popScope () {    return this.scope.pop();  }  popVariableStack () {    return this.definedVariablesStack.pop();  }  getCurrentVariableStack () {    return this.definedVariablesStack[this.definedVariablesStack.length - 1];  }  isEOF () {    this.getToken(this.pos);    return this.tokenStream.fetchedEOF;  }  parseProgram () {    this.consumeNewLines();    const token = this.getToken();    let globalVars = [];    let functions = [];    if(this.lexerClass.RK_PROGRAM === token.type) {      this.pos++;      this.consumeNewLines();      this.checkOpenCurly();      this.pos++;      this.pushVariableStack();      while(true) {        this.consumeNewLines();        const token = this.getToken();        if (token.type === this.lexerClass.RK_CONST || this.isVariableType(token)) {          globalVars = globalVars.concat(this.parseGlobalVariables());        } else if (token.type === this.lexerClass.RK_FUNCTION) {          this.pushVariableStack();          functions = functions.concat(this.parseFunction());          this.popVariableStack();        } else {          break;        }      }      this.consumeNewLines();      this.checkCloseCurly();      this.pos++;      this.consumeNewLines();      if(!this.isEOF()) {        throw SyntaxErrorFactory.extra_lines();      }      this.popVariableStack();      return {global: globalVars, functions: functions};    } else {      throw SyntaxErrorFactory.token_missing_one(this.lexer.literalNames[this.lexerClass.RK_PROGRAM], token);    }  }  checkOpenCurly (attempt = false) {    const token = this.getToken();    if(this.lexerClass.OPEN_CURLY !== token.type){      if(!attempt)        throw SyntaxErrorFactory.token_missing_one('{', token);      else        return false;    }    return true;  }  checkCloseCurly (attempt = false) {    const token = this.getToken();    if(this.lexerClass.CLOSE_CURLY !== token.type){      if(!attempt)        throw SyntaxErrorFactory.token_missing_one('}', token);      else        return false;    }    return true;  }  /* It checks if the current token at position pos is a ']'.  * As a check function it doesn't increment pos.  *  * @params bool:attempt, indicates that the token is optional. Defaults: false  *  * @returns true if the attempt is true and current token is '[',  *   false is attempt is true and current token is not '['  **/  checkOpenBrace (attempt = false) {    const token = this.getToken();    if(this.lexerClass.OPEN_BRACE !== token.type){      if (!attempt) {        throw SyntaxErrorFactory.token_missing_one('[', token);      } else {        return false;      }    }    return true;  }  checkCloseBrace (attempt = false) {    const token = this.getToken();    if(this.lexerClass.CLOSE_BRACE !== token.type){      if (!attempt) {        throw SyntaxErrorFactory.token_missing_one(']', token);      } else {        return false;      }    }    return true;  }  checkOpenParenthesis (attempt = false) {    const token = this.getToken();    if(this.lexerClass.OPEN_PARENTHESIS !== token.type){      if (!attempt) {        throw SyntaxErrorFactory.token_missing_one('(', token);      } else {        return false;      }    }    return true;  }  checkCloseParenthesis (attempt = false) {    const token = this.getToken();    if(this.lexerClass.CLOSE_PARENTHESIS !== token.type){      if (!attempt) {        throw SyntaxErrorFactory.token_missing_one(')', token);      } else {        return false;      }    }    return true;  }  checkEOS (attempt = false)  {    const eosToken = this.getToken();    if (eosToken.type !== this.lexerClass.EOS) {      if (!attempt)        throw SyntaxErrorFactory.eos_missing(eosToken);      else        return false;    }    return true;  }  checkFunctionDuplicate (functionID, funcIDToken) {    const id = functionID === null ? "$main" : functionID;    const index = this.definedFuncsNameList.indexOf(id);    if(index !== -1) {      throw SyntaxErrorFactory.duplicate_function(funcIDToken);    }    this.definedFuncsNameList.push(id);  }  checkVariableDuplicate (variableID, variableIDToken) {    const index = this.getCurrentVariableStack().indexOf(variableID);    if(index !== -1) {      throw SyntaxErrorFactory.duplicate_variable(variableIDToken);    }    this.getCurrentVariableStack().push(variableID);  }  consumeForSemiColon () {    const eosToken = this.getToken();    if (eosToken.type === this.lexerClass.EOS && eosToken.text.match(';')) {      this.pos++;      return;      }    throw SyntaxErrorFactory.token_missing_one(';', eosToken);  }  parseGlobalVariables () {    const decl = this.parseMaybeConst();    this.checkEOS();    this.pos++;    return decl;  }  /*  * Checks if the next token is PR_CONST. It's only available  * at global variables declaration level  * @returns Declararion(const, type, id, initVal?)  **/  parseMaybeConst () {    const constToken = this.getToken();    if(constToken.type === this.lexerClass.RK_CONST) {      this.pos++;      const typeString = this.parseType();      return this.parseDeclaration(typeString, true);    } else if(this.isVariableType(constToken)) {      const typeString = this.parseType();      return this.parseDeclaration(typeString);    } else {      throw SyntaxErrorFactory.token_missing_list(        [this.lexer.literalNames[this.lexerClass.RK_CONST]].concat(this.getTypeArray()), constToken);    }  }  /*  * Parses a declarion of the form: type --- id --- (= --- EAnd)?  * @returns a list of Declararion(const, type, id, initVal?)  **/  parseDeclaration (typeString, isConst = false) {    let initial = null;    let dim1 = null;    let dim2 = null;    const idToken = this.getToken();    const idString = this.parseID();    this.checkVariableDuplicate(idString,idToken);    // Check for array or vector    // ID[int/IDi][int/IDj]    if (this.checkOpenBrace(true)) {      this.pos++;      this.consumeNewLines();      dim1 = this.parseArrayDimension();      this.consumeNewLines();      this.checkCloseBrace();      this.pos++;      if(this.checkOpenBrace(true)) {        this.pos++;        this.consumeNewLines();        dim2 = this.parseArrayDimension();        this.consumeNewLines();        this.checkCloseBrace();        this.pos++;      }    }    const equalsToken = this.getToken();    if(isConst && equalsToken.type !== this.lexerClass.EQUAL ) {      throw SyntaxErrorFactory.const_not_init(idToken);    }    if(equalsToken.type === this.lexerClass.EQUAL) {      this.pos++;      initial = this.parseExpressionOR();    }    let declaration = null;    let dimensions = 0;    if (dim1 !== null) {      dimensions++;      if(dim2 !== null) {        dimensions++;      }      declaration = new Commands.ArrayDeclaration(idString,        new CompoundType(typeString, dimensions), dim1, dim2, initial, isConst);    } else {      declaration = new Commands.Declaration(idString, typeString, initial, isConst);    }    declaration.sourceInfo = SourceInfo.createSourceInfo(idToken);    const commaToken = this.getToken();    if(commaToken.type === this.lexerClass.COMMA) {      this.pos++;      this.consumeNewLines();      return [declaration]      .concat(this.parseDeclaration(typeString, isConst));    } else {       return [declaration]    }  }  consumeNewLines () {    let token = this.getToken();    while(token.type === this.lexerClass.EOS && token.text.match('[\r\n]+')) {      this.pos++;      token = this.getToken();    }  }  isVariableType (token) {    return this.variableTypes.find(v => v === token.type);  }  /*  * Reads the next token of the stream to check if it is a Integer or an ID.  * @returns Integer | ID  **/  parseArrayDimension () {    const dimToken = this.getToken();    if(dimToken.type === this.lexerClass.INTEGER) {      //parse as int literal      this.pos++;      return this.getIntLiteral(dimToken);    } else if(dimToken.type === this.lexerClass.ID) {      //parse as variable      this.pos++;      return this.parseVariable(dimToken);    } else {      throw SyntaxErrorFactory.invalid_array_dimension(this.lexer.literalNames[this.lexerClass.RK_INTEGER], dimToken);    }  }  /*  * Returns an object {type: 'int', value: value}.  * It checks for binary and hexadecimal integers.  * @returns object with fields type and value  **/  getIntLiteral (token) {    const text = token.text;    const sourceInfo = SourceInfo.createSourceInfo(token);    const exp = new Expressions.IntLiteral(toInt(text));    exp.sourceInfo = sourceInfo;    return exp;  }  getRealLiteral (token) {    const sourceInfo = SourceInfo.createSourceInfo(token);    const exp = new Expressions.RealLiteral(toReal(token.text));    exp.sourceInfo = sourceInfo;    return exp;  }  getStringLiteral (token) {    const text = token.text;    const sourceInfo = SourceInfo.createSourceInfo(token);    const exp = new Expressions.StringLiteral(toString(text));    exp.sourceInfo = sourceInfo;    return exp;  }  getBoolLiteral (token) {    const val = toBool(token.text);    const exp = new Expressions.BoolLiteral(val);    exp.sourceInfo = SourceInfo.createSourceInfo(token);;    return exp;  }  parseArrayLiteral () {    this.checkOpenCurly();    const beginArray = this.getToken();    if (this.parsingArrayDimension >= 2) {      throw SyntaxErrorFactory.token_missing_list(`Array dimensions exceed maximum size of 2 at line ${beginArray.line}`);    }    this.pos++;    this.parsingArrayDimension++;    this.consumeNewLines();    const data = this.parseExpressionList();    this.consumeNewLines();    this.checkCloseCurly()    const endArray = this.getToken();    this.pos++;    this.parsingArrayDimension--;    if (this.parsingArrayDimension === 0) {      // if (!data.isValid) {      //   // TODO: better error message      //   console.log('invalid array');      //   throw new Error(`Invalid array at line ${beginArray.line}`);      // }    }    const sourceInfo = SourceInfo.createSourceInfoFromList(beginArray, endArray);    let dataDim = 1;    if(data[0] instanceof Expressions.ArrayLiteral) {      dataDim++;    }    const type = new CompoundType(Types.UNDEFINED, dataDim);    const exp = new Expressions.ArrayLiteral(type, data);    exp.sourceInfo = sourceInfo;    return exp;  }  /*  * Returns an object {type: 'variable', value: value}.  * @returns object with fields type and value  **/  parseVariable (token) {    const sourceInfo = SourceInfo.createSourceInfo(token);    const exp = new Expressions.VariableLiteral(token.text);    exp.sourceInfo = sourceInfo;    return exp;  }  /*  * Returns an object representing a function. It has  * four attributes: returnType, id, formalParams and block.  * The block object has two attributes: declarations and commands  **/  parseFunction () {    this.pushScope(IVProgParser.FUNCTION);    let formalParams = [];    const token = this.getToken();    if(token.type !== this.lexerClass.RK_FUNCTION) {      //throw SyntaxError.createError(this.lexer.literalNames[this.lexerClass.PR_FUNCAO], token);      return null;    }    this.pos++;    const funType = this.parseType();    let dimensions = 0;    if(this.checkOpenBrace(true)) {      this.pos++;      this.checkCloseBrace();      this.pos++;      dimensions++;      if(this.checkOpenBrace(true)) {        this.pos++;        this.checkCloseBrace();        this.pos++;        dimensions++;      }    }    const funcIDToken = this.getToken();    const functionID = this.parseID();    this.checkFunctionDuplicate(functionID, funcIDToken);    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    if (!this.checkCloseParenthesis(true)) {      formalParams = this.parseFormalParameters(); // formal parameters       this.consumeNewLines();      this.checkCloseParenthesis();      this.pos++;    } else {      this.pos++;    }    this.consumeNewLines();    const commandsBlock = this.parseCommandBlock();    let returnType = funType;    if(dimensions > 0) {      returnType = new CompoundType(funType, dimensions);    }    const func = new Commands.Function(functionID, returnType, formalParams, commandsBlock);    if (functionID === null && !func.isMain) {      throw SyntaxErrorFactory.invalid_main_return(LanguageDefinedFunction.getMainFunctionName(),        this.lexer.literalNames[this.lexerClass.RK_VOID],        token.line);    } else if (func.isMain && formalParams.length !== 0) {      throw SyntaxErrorFactory.main_parameters();    }    this.popScope();    return func;  }  /*  * Parse the formal parameters of a function.  * @returns a list of objects with the following attributes: type, id and dimensions.  **/  parseFormalParameters () {    const list = [];    while(true) {      let dimensions = 0;      const typeString = this.parseType();      const idToken = this.getToken();      const idString = this.parseID();      this.checkVariableDuplicate(idString, idToken);      if (this.checkOpenBrace(true)) {        this.pos++;        dimensions++;        this.checkCloseBrace();        this.pos++;        if(this.checkOpenBrace(true)) {          this.pos++;          dimensions++;          this.checkCloseBrace();          this.pos++;        }      }      let type = null;      if(dimensions > 0) {        type = new CompoundType(typeString, dimensions);      } else {        type = typeString;      }      list.push(new Commands.FormalParameter(type, idString));      const commaToken = this.getToken();      if (commaToken.type !== this.lexerClass.COMMA)        break;      this.pos++;      this.consumeNewLines();    }    return list;  }  parseID () {    const token = this.getToken();    if(token.type !== this.lexerClass.ID) {      throw SyntaxErrorFactory.id_missing(token);    }    this.pos++;    if (this.insideScope(IVProgParser.FUNCTION)) {      if (token.text === LanguageDefinedFunction.getMainFunctionName()){        return null;      }    }    return token.text;  }  parseMaybeLibID () {    const token = this.getToken();    if(token.type !== this.lexerClass.ID && token.type !== this.lexerClass.LIB_ID) {      throw SyntaxErrorFactory.id_missing(token);    }     this.pos++;    return token.text;  }  parseType () {    const token = this.getToken();    if(token.type === this.lexerClass.ID && this.insideScope(IVProgParser.FUNCTION)) {      return Types.VOID;    } else if (token.type === this.lexerClass.RK_VOID && this.insideScope(IVProgParser.FUNCTION)) {      this.pos++;      return Types.VOID;    } else if (this.isVariableType(token)) {      this.pos++;      switch(token.type) {        case this.lexerClass.RK_INTEGER:          return Types.INTEGER;        case this.lexerClass.RK_BOOLEAN:          return Types.BOOLEAN;        case this.lexerClass.RK_REAL:          return Types.REAL;        case this.lexerClass.RK_STRING:          return Types.STRING;        default:          break;      }    }        throw SyntaxErrorFactory.invalid_type(this.getTypeArray(), token);  }  parseCommandBlock (optionalCurly = false) {    let variablesDecl = [];    const commands = [];    let hasOpen = false;    if (this.checkOpenCurly(optionalCurly)) {      this.pos++;      hasOpen = true;    }    this.consumeNewLines();    while(true) {      const cmd = this.parseCommand();      if (cmd === null)        break;      if(cmd !== -1) {        if (cmd instanceof Array) {          variablesDecl = variablesDecl.concat(cmd);        } else {          commands.push(cmd);        }      }    }    this.consumeNewLines();    if (hasOpen) {      this.checkCloseCurly()      this.pos++;      this.consumeNewLines();    }    return new Commands.CommandBlock(variablesDecl, commands);  }  parseCommand () {    const token = this.getToken();    if (this.isVariableType(token)) {      if(!this.insideScope(IVProgParser.FUNCTION)) {        throw SyntaxErrorFactory.invalid_var_declaration(token.line);      }      this.pushScope(IVProgParser.BASE);      const varType = this.parseType();      this.popScope();      const cmd = this.parseDeclaration(varType);      this.checkEOS();      this.pos++;      return cmd;    } else if (token.type === this.lexerClass.ID) {      return this.parseIDCommand();    } else if (token.type === this.lexerClass.LIB_ID) {      return this.parseIDCommand();    } else if (token.type === this.lexerClass.RK_RETURN) {      return this.parseReturn();    } else if (token.type === this.lexerClass.RK_WHILE) {      return this.parseWhile();    } else if (token.type === this.lexerClass.RK_FOR) {      return this.parseFor();    } else if (token.type === this.lexerClass.RK_BREAK ) {      if(!this.insideScope(IVProgParser.BREAKABLE)) {        throw SyntaxErrorFactory.invalid_break_command(          this.lexer.literalNames[this.lexerClass.RK_BREAK],          token        );      }      return this.parseBreak();    } else if (token.type === this.lexerClass.RK_SWITCH) {      return this.parseSwitchCase();    } else if (token.type === this.lexerClass.RK_DO) {      return this.parseDoWhile();    } else if (token.type === this.lexerClass.RK_IF) {      return this.parseIfThenElse();    } else if (this.checkEOS(true)){      this.pos++;      return -1;    } else {      return null;    }  }  parseSwitchCase () {    this.pushScope(IVProgParser.BREAKABLE);    this.pos++;    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    const exp = this.parseExpressionOR();    this.consumeNewLines();    this.checkCloseParenthesis();    this.pos++;    this.consumeNewLines();    this.checkOpenCurly();    this.pos++;    this.consumeNewLines();    const casesList = this.parseCases();    this.consumeNewLines();    this.checkCloseCurly();    this.pos++;    this.consumeNewLines();    this.popScope();    return new Commands.Switch(exp, casesList);  }  parseDoWhile () {    this.pos++;    this.consumeNewLines();    this.pushScope(IVProgParser.BREAKABLE);    const commandsBlock = this.parseCommandBlock();    this.consumeNewLines(); //Maybe not...    const whileToken = this.getToken();    if (whileToken.type !== this.lexerClass.RK_WHILE) {      throw SyntaxErrorFactory.token_missing_one(this.lexer.literalNames[this.lexerClass.RK_WHILE], whileToken);    }    this.pos++;    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    const condition = this.parseExpressionOR();    this.consumeNewLines();    this.checkCloseParenthesis();    this.pos++;    this.checkEOS();    this.popScope();    return new Commands.DoWhile(condition, commandsBlock);  }  parseIfThenElse () {    if(this.insideScope(IVProgParser.BREAKABLE)) {      this.pushScope(IVProgParser.BREAKABLE);    } else {      this.pushScope(IVProgParser.COMMAND);    }    const token = this.getToken();    this.pos++;    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    const logicalExpression = this.parseExpressionOR();    this.consumeNewLines();    this.checkCloseParenthesis();    this.pos++;    this.consumeNewLines();    const cmdBlocks = this.parseCommandBlock();    const maybeElse = this.getToken();    if(maybeElse.type === this.lexerClass.RK_ELSE) {      this.pos++;      this.consumeNewLines();      const maybeIf = this.getToken();      let elseBlock = null;      if(this.checkOpenCurly(true)) {        elseBlock = this.parseCommandBlock();      } else if(maybeIf.type === this.lexerClass.RK_IF) {        elseBlock = this.parseIfThenElse();      } else {        throw SyntaxErrorFactory.token_missing_list([this.lexer.literalNames[this.lexerClass.RK_IF], '{'], maybeIf);      }      this.popScope();      const cmd = new Commands.IfThenElse(logicalExpression, cmdBlocks, elseBlock);      cmd.sourceInfo = SourceInfo.createSourceInfo(token);      return cmd;    }    this.popScope();    const cmd = new Commands.IfThenElse(logicalExpression, cmdBlocks, null);    cmd.sourceInfo = SourceInfo.createSourceInfo(token);    return cmd;  }  parseFor () {    this.pushScope(IVProgParser.BREAKABLE);    this.pos++;    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    const attribution = this.parseForAssign();    this.consumeNewLines();    const condition = this.parseExpressionOR();    this.consumeForSemiColon();    const increment = this.parseForAssign(true);    this.checkCloseParenthesis()    this.pos++;    this.consumeNewLines();    const commandsBlock = this.parseCommandBlock();    this.popScope();    return new Commands.For(attribution, condition, increment, commandsBlock);  }  parseWhile () {    this.pushScope(IVProgParser.BREAKABLE);    const token = this.getToken();    this.pos++;    this.checkOpenParenthesis();    this.pos++;    this.consumeNewLines();    const logicalExpression = this.parseExpressionOR();    this.consumeNewLines();    this.checkCloseParenthesis();    this.pos++;    this.consumeNewLines();    const cmdBlocks = this.parseCommandBlock();    this.popScope();    const cmd = new Commands.While(logicalExpression, cmdBlocks);    cmd.sourceInfo = SourceInfo.createSourceInfo(token);    return cmd;  }  parseBreak () {    this.pos++;    this.checkEOS();    this.pos++;    return new Commands.Break();  }  parseReturn () {    this.pos++;    let exp = null;    if(!this.checkEOS(true)) {      exp = this.parseExpressionOR();      this.checkEOS();    }    this.pos++;    return new Commands.Return(exp);  }  parseIDCommand () {    const refToken = this.getToken();    const isID = refToken.type === this.lexerClass.ID;    const id = this.parseMaybeLibID();    if(this.checkOpenBrace(true)) {      this.pos++;      let lineExpression = null;      let columnExpression = null;      this.consumeNewLines();      lineExpression = this.parseExpression()      this.consumeNewLines();      this.checkCloseBrace();      this.pos++;      if (this.checkOpenBrace(true)) {        this.pos++        this.consumeNewLines();        columnExpression = this.parseExpression();        this.consumeNewLines();        this.checkCloseBrace();        this.pos++;      }      const equalToken = this.getToken();      if (equalToken.type !== this.lexerClass.EQUAL) {        throw SyntaxErrorFactory.token_missing_one('=', equalToken);      }      this.pos++;      const exp = this.parseExpressionOR();      this.checkEOS();      this.pos++;      const cmd = new Commands.ArrayIndexAssign(id, lineExpression, columnExpression, exp);      cmd.sourceInfo = SourceInfo.createSourceInfo(equalToken);      return cmd;    }    const equalOrParenthesis = this.getToken();    if (isID && equalOrParenthesis.type === this.lexerClass.EQUAL) {      this.pos++      const exp = this.parseExpressionOR();      this.checkEOS();      this.pos++;      const cmd = new Commands.Assign(id, exp);      cmd.sourceInfo = SourceInfo.createSourceInfo(equalOrParenthesis);      return cmd;    } else if (equalOrParenthesis.type === this.lexerClass.OPEN_PARENTHESIS) {      const funcCall = this.parseFunctionCallCommand(id);      this.checkEOS();      this.pos++;      return funcCall;    } else if (isID) {      throw SyntaxErrorFactory.token_missing_list(['=','('], equalOrParenthesis);    } else {      throw SyntaxErrorFactory.invalid_id_format(refToken);    }  }  parseForAssign (isLast = false) {    if(!isLast)      this.consumeNewLines();    if(this.checkEOS(true)) {      return null;    }    const id = this.parseID();    const equal = this.getToken();    if (equal.type !== this.lexerClass.EQUAL) {      throw SyntaxErrorFactory.token_missing_one('=', equal);    }    this.pos++    const exp = this.parseExpressionOR();    if(!isLast) {      this.consumeForSemiColon();    }    const sourceInfo = SourceInfo.createSourceInfo(equal);    const cmd = new Commands.Assign(id, exp);    cmd.sourceInfo = sourceInfo;    return cmd;  }  parseCases () {    const token = this.getToken();    if(token.type !== this.lexerClass.RK_CASE) {      throw SyntaxErrorFactory.token_missing_one(this.lexer.literalNames[this.lexerClass.RK_CASE], token);    }    this.pos++;    const nextToken = this.getToken();    if(nextToken.type === this.lexerClass.RK_DEFAULT) {      this.pos++;      const colonToken = this.getToken();      if (colonToken.type !== this.lexerClass.COLON) {        throw SyntaxErrorFactory.token_missing_one(':', colonToken);      }      this.pos++;      this.consumeNewLines();      const block = this.parseCommandBlock(true);      const defaultCase = new Commands.Case(null);      defaultCase.setCommands(block.commands);      return [defaultCase];    } else {      const exp = this.parseExpressionOR();      const colonToken = this.getToken();      if (colonToken.type !== this.lexerClass.COLON) {        throw SyntaxErrorFactory.token_missing_one(':', colonToken);      }      this.pos++;      this.consumeNewLines();      const block = this.parseCommandBlock(true);      const aCase = new Commands.Case(exp);      aCase.setCommands(block.commands);      const caseToken = this.getToken();      if(caseToken.type === this.lexerClass.RK_CASE) {        return [aCase].concat(this.parseCases());      } else {        return [aCase];      }    }  }  /*  * Parses an Expression following the structure:  *  * EOR  => EAnd ( 'or' EOR)? #expression and  *  * EOR   => ENot ('and' EOR)? #expression or  *  * ENot  => 'not'? ER #expression not  *  * ER    => E ((>=, <=, ==, >, <) E)? #expression relational  *  * E     => factor ((+, -) E)? #expression  *  * factor=> term ((*, /, %) factor)?  *  * term  => literal || arrayAccess || FuncCall || ID || '('EAnd')'  **/  parseExpressionOR () {    let exp1 = this.parseExpressionAND();    while (this.getToken().type === this.lexerClass.OR_OPERATOR) {      const opToken = this.getToken();      this.pos++;      const or = convertFromString('or');      this.consumeNewLines();      const exp2 = this.parseExpressionAND();      const finalExp = new Expressions.InfixApp(or, exp1, exp2);      finalExp.sourceInfo = SourceInfo.createSourceInfo(opToken);      exp1 = finalExp    }    return exp1;  }  parseExpressionAND () {    let exp1 = this.parseExpressionNot();    while (this.getToken().type === this.lexerClass.AND_OPERATOR) {      const opToken = this.getToken();      this.pos++;      const and = convertFromString('and');      this.consumeNewLines();      const exp2 = this.parseExpressionNot();      const finalExp = new Expressions.InfixApp(and, exp1, exp2);      finalExp.sourceInfo = SourceInfo.createSourceInfo(opToken);      exp1 = finalExp;    }    return exp1;  }  parseExpressionNot () {    const maybeNotToken = this.getToken();    if (maybeNotToken.type === this.lexerClass.NOT_OPERATOR) {      const opToken = this.getToken();      this.pos++;      const not = convertFromString('not');      const exp1 = this.parseExpressionRel();      const finalExp = new Expressions.UnaryApp(not, exp1);      finalExp.sourceInfo = SourceInfo.createSourceInfo(opToken);      return finalExp;          } else {      return this.parseExpressionRel();    }  }  parseExpressionRel () {    let exp1 = this.parseExpression();    while (this.getToken().type === this.lexerClass.RELATIONAL_OPERATOR) {      const relToken = this.getToken();      this.pos++;      const rel = convertFromString(relToken.text);      const exp2 = this.parseExpression();      const finalExp = new Expressions.InfixApp(rel, exp1, exp2);      finalExp.sourceInfo = SourceInfo.createSourceInfo(relToken);      exp1 = finalExp;    }    return exp1;  }  parseExpression () {    let factor = this.parseFactor();    while (this.getToken().type === this.lexerClass.SUM_OP) {      const sumOpToken = this.getToken();      this.pos++;      const op = convertFromString(sumOpToken.text);      const factor2 = this.parseFactor();      const finalExp = new Expressions.InfixApp(op, factor, factor2);      finalExp.sourceInfo = SourceInfo.createSourceInfo(sumOpToken);      factor = finalExp;    }    return factor;  }  parseFactor () {    let term = this.parseTerm();    while (this.getToken().type === this.lexerClass.MULTI_OP) {      const multOpToken = this.getToken();      this.pos++;      const op = convertFromString(multOpToken.text);      const term2 =this.parseTerm();      const finalExp = new Expressions.InfixApp(op, term, term2);      finalExp.sourceInfo = SourceInfo.createSourceInfo(multOpToken);      term = finalExp;    }    return term;  }  parseTerm () {    const token = this.getToken();    let sourceInfo = null;    switch(token.type) {      case this.lexerClass.SUM_OP:        this.pos++;        sourceInfo = SourceInfo.createSourceInfo(token);        const exp = new Expressions.UnaryApp(convertFromString(token.text), this.parseTerm());        exp.sourceInfo = sourceInfo;        return exp;      case this.lexerClass.INTEGER:        this.pos++;        return this.getIntLiteral(token);      case this.lexerClass.REAL:        this.pos++;        return this.getRealLiteral(token);      case this.lexerClass.STRING:        this.pos++;        return this.getStringLiteral(token);      case this.lexerClass.RK_TRUE:      case this.lexerClass.RK_FALSE:        this.pos++;        return this.getBoolLiteral(token);      case this.lexerClass.OPEN_CURLY:        return this.parseArrayLiteral();      case this.lexerClass.ID:      case this.lexerClass.LIB_ID:        return this.parseIDTerm();      case this.lexerClass.OPEN_PARENTHESIS:        return this.parseParenthesisExp();      default:        throw SyntaxErrorFactory.invalid_terminal(token);    }  }  parseIDTerm () {    const tokenA = this.getToken();    const id = this.parseMaybeLibID();    const isID = tokenA.type === this.lexerClass.ID;    if(isID && this.checkOpenBrace(true)) {      let tokenB = null;      this.pos++;      const firstIndex = this.parseExpression();      let secondIndex = null;      this.consumeNewLines();      this.checkCloseBrace();      tokenB = this.getToken();      this.pos++;      if(this.checkOpenBrace(true)){        this.pos++;        secondIndex = this.parseExpression();        this.consumeNewLines();        this.checkCloseBrace();        tokenB = this.getToken();        this.pos++;      }      const sourceInfo = SourceInfo.createSourceInfoFromList(tokenA, tokenB);       const exp = new Expressions.ArrayAccess(id, firstIndex, secondIndex);      exp.sourceInfo = sourceInfo;      return exp;    } else if (this.checkOpenParenthesis(true)) {      return this.parseFunctionCallExpression(id);    } else if (isID) {      const sourceInfo = SourceInfo.createSourceInfo(tokenA);      const exp = new Expressions.VariableLiteral(id);      exp.sourceInfo = sourceInfo;      return exp;    } else {      throw SyntaxErrorFactory.invalid_id_format(tokenA);    }  }  getFunctionName (id) {    const name = LanguageDefinedFunction.getInternalName(id);    if (name === null) {      if (id === LanguageDefinedFunction.getMainFunctionName()) {        return null;      }      return id;    } else {      return name;    }  }  parseFunctionCallExpression (id) {    const tokenA = this.getToken(this.pos - 1);    const actualParameters = this.parseActualParameters();    const tokenB = this.getToken(this.pos - 1);    const funcName = this.getFunctionName(id);    const sourceInfo = SourceInfo.createSourceInfoFromList(tokenA, tokenB);    const cmd = new Expressions.FunctionCall(funcName, actualParameters);    cmd.sourceInfo = sourceInfo;    return cmd;  }  parseFunctionCallCommand (id) {    return this.parseFunctionCallExpression(id);  }  parseParenthesisExp () {    this.checkOpenParenthesis();    const tokenA = this.getToken();    this.pos++;    this.consumeNewLines();    const exp = this.parseExpressionOR();    this.consumeNewLines();    this.checkCloseParenthesis();    const tokenB = this.getToken();    const sourceInfo = SourceInfo.createSourceInfoFromList(tokenA, tokenB);    this.pos++;    exp.sourceInfo = sourceInfo;    return exp;  }  parseActualParameters () {    this.checkOpenParenthesis();    this.pos++;    if(this.checkCloseParenthesis(true)) {      this.pos++;      return [];    }    this.consumeNewLines();    const list = this.parseExpressionList();    this.consumeNewLines();    this.checkCloseParenthesis();    this.pos++;    return list;  }  parseExpressionList () {    const list = [];    while(true) {      const exp = this.parseExpressionOR();      list.push(exp);      const maybeToken = this.getToken();      if (maybeToken.type !== this.lexerClass.COMMA) {        break;      } else {        this.pos++;        this.consumeNewLines();      }    }    return list;  }  getTypeArray () {    const types = this.insideScope(IVProgParser.FUNCTION) ? this.functionTypes : this.variableTypes;    return types.map( x => this.lexer.literalNames[x]);  }}
 |