From 0a234c4dcbfde2879df1ca2dafab08d2a4f0132c Mon Sep 17 00:00:00 2001
From: Jan Travnicek <Jan.Travnicek@fit.cvut.cz>
Date: Wed, 4 Dec 2019 17:58:23 +0100
Subject: [PATCH] while command

---
 alib2cli/src/ast/command/WhileCommand.h | 38 +++++++++++++++++++++++++
 alib2cli/src/parser/Parser.cpp          | 15 +++++++++-
 2 files changed, 52 insertions(+), 1 deletion(-)
 create mode 100644 alib2cli/src/ast/command/WhileCommand.h

diff --git a/alib2cli/src/ast/command/WhileCommand.h b/alib2cli/src/ast/command/WhileCommand.h
new file mode 100644
index 0000000000..0d77e62b62
--- /dev/null
+++ b/alib2cli/src/ast/command/WhileCommand.h
@@ -0,0 +1,38 @@
+#ifndef _CLI_WHILE_COMMAND_H_
+#define _CLI_WHILE_COMMAND_H_
+
+#include <ast/Command.h>
+#include <environment/Environment.h>
+#include <ast/Statement.h>
+#include <common/CastHelper.h>
+
+namespace cli {
+
+class WhileCommand : public Command {
+	std::unique_ptr < Expression > m_condition;
+	std::unique_ptr < Command > m_body;
+
+public:
+	WhileCommand ( std::unique_ptr < Expression > condition, std::unique_ptr < Command > body ) : m_condition ( std::move ( condition ) ), m_body ( std::move ( body ) ) {
+	}
+
+	CommandResult run ( Environment & environment ) const override {
+		CommandResult res = cli::CommandResult::OK;
+		while ( res == cli::CommandResult::OK ) {
+			std::shared_ptr < abstraction::Value > conditionResult = m_condition->translateAndEval ( environment );
+
+			std::shared_ptr < abstraction::Value > castedResult = abstraction::CastHelper::eval ( environment, conditionResult, "bool" );
+
+			if ( ! std::static_pointer_cast < abstraction::ValueHolderInterface < bool > > ( castedResult )->getValue ( ) )
+				break;
+
+			res = m_body->run ( environment );
+		}
+
+		return res;
+	}
+};
+
+} /* namespace cli */
+
+#endif /* _CLI_WHILE_COMMAND_H_ */
diff --git a/alib2cli/src/parser/Parser.cpp b/alib2cli/src/parser/Parser.cpp
index 39ac763c9f..4082f807ce 100644
--- a/alib2cli/src/parser/Parser.cpp
+++ b/alib2cli/src/parser/Parser.cpp
@@ -34,6 +34,7 @@
 #include <ast/command/EvalCommand.h>
 #include <ast/command/InterpretCommand.h>
 #include <ast/command/IfCommand.h>
+#include <ast/command/WhileCommand.h>
 #include <ast/command/ReturnCommand.h>
 
 #include <ast/expression/BatchExpression.h>
@@ -369,7 +370,7 @@ std::unique_ptr < CommandList > Parser::block ( ) {
 }
 
 std::unique_ptr < Command > Parser::semicolon_command ( ) {
-	bool semicolonFreeCommand = check_nonreserved_kw ( "begin", "if" );
+	bool semicolonFreeCommand = check_nonreserved_kw ( "begin", "if", "while" );
 	std::unique_ptr < Command > res = command ( );
 	if ( ! semicolonFreeCommand )
 		match ( cli::Lexer::TokenType::SEMICOLON_SIGN );
@@ -510,6 +511,18 @@ std::unique_ptr < Command > Parser::command ( ) {
 		}
 
 		return std::make_unique < IfCommand > ( std::move ( condition ), std::move ( thenBranch ), std::move ( elseBranch ) );
+	} else if ( check_nonreserved_kw ( "while" ) ) {
+		if ( globalScope ( ) )
+			throw exception::CommonException ( "Statement not available in global scope." );
+		match_nonreserved_kw ( "while" );
+		match ( cli::Lexer::TokenType::LEFT_PAREN );
+		std::unique_ptr < Expression > condition = expression_or_batch ( );
+		match ( cli::Lexer::TokenType::RIGHT_PAREN );
+
+		match_nonreserved_kw ( "do" );
+		std::unique_ptr < Command > body = semicolon_command ( );
+
+		return std::make_unique < WhileCommand > ( std::move ( condition ), std::move ( body ) );
 	} else {
 		throw exception::CommonException ( "Mismatched set " + ext::to_string ( getCheckOptions ( ) ) + " while expanding parse rule. Token is " + ( ( std::string ) m_current ) + "." );
 	}
-- 
GitLab