This commit is contained in:
2022-02-26 10:50:40 +03:00
parent f5de7299ec
commit d047ebf453

View File

@@ -31,7 +31,7 @@ ExprAST::~ExprAST() {}
IoTValue* ExprAST::exec() {return nullptr;} IoTValue* ExprAST::exec() {return nullptr;}
int ExprAST::setValue(IoTValue *val) {return 0;} // 0 - установка значения не поддерживается наследником int ExprAST::setValue(IoTValue *val) {return 0;} // 0 - установка значения не поддерживается наследником
bool ExprAST::hasEventIdName(String eventIdName) {return false;} // по умолчанию все узлы не связаны с ИД события, для которого выполняется сценарий bool ExprAST::hasEventIdName(String eventIdName) {return false;} // по умолчанию все узлы не связаны с ИД события, для которого выполняется сценарий
struct IoTValue zeroIotVal; //struct IoTValue zeroIotVal;
/// NumberExprAST - Класс узла выражения для числовых литералов (Например, "1.0"). /// NumberExprAST - Класс узла выражения для числовых литералов (Например, "1.0").
class NumberExprAST : public ExprAST { class NumberExprAST : public ExprAST {
@@ -61,25 +61,32 @@ public:
class VariableExprAST : public ExprAST { class VariableExprAST : public ExprAST {
String Name; String Name;
IoTItem* Item; // ссылка на объект модуля (прямой доступ к идентификатору указанному в сценарии), если получилось найти модуль по ID IoTItem* Item; // ссылка на объект модуля (прямой доступ к идентификатору указанному в сценарии), если получилось найти модуль по ID
bool ItemIsLocal = false;
public: public:
VariableExprAST(const String &name, IoTItem* item) : Name(name), Item(item) {} VariableExprAST(const String &name, IoTItem* item) : Name(name), Item(item) {
if (item) ItemIsLocal = item->iAmLocal;
}
int setValue(IoTValue *val) { int setValue(IoTValue *val) {
if (Item) { if (!ItemIsLocal) Item = findIoTItem(Name);
//Item->value = *val; // устанавливаем значение в связанном Item модуля напрямую if (Item) Item->setValue(*val);
Item->setValue(*val); else return 0;
}
return 1; return 1;
} }
IoTValue* exec() { IoTValue* exec() {
if (!ItemIsLocal) Item = findIoTItem(Name);
if (Item) {
if (Item->value.isDecimal) if (Item->value.isDecimal)
Serial.printf("Call from VariableExprAST: %s = %f\n", Name.c_str(), Item->value.valD); Serial.printf("Call from VariableExprAST: %s = %f\n", Name.c_str(), Item->value.valD);
else Serial.printf("Call from VariableExprAST: %s = %s\n", Name.c_str(), Item->value.valS.c_str()); else Serial.printf("Call from VariableExprAST: %s = %s\n", Name.c_str(), Item->value.valS.c_str());
return &(Item->value); return &(Item->value);
} }
return nullptr; // Item не найден.
}
}; };
/// BinaryExprAST - Класс узла выражения для бинарных операторов. /// BinaryExprAST - Класс узла выражения для бинарных операторов.
@@ -109,6 +116,8 @@ public:
Serial.printf("Call from BinaryExprAST: %s\n", printStr.c_str()); Serial.printf("Call from BinaryExprAST: %s\n", printStr.c_str());
if (RHS == nullptr || LHS == nullptr) return nullptr;
IoTValue* rhs = RHS->exec(); // получаем значение правого операнда для возможного использования в операции присваивания IoTValue* rhs = RHS->exec(); // получаем значение правого операнда для возможного использования в операции присваивания
if (Op == '=' && LHS->setValue(rhs)) { // если установка значения не поддерживается, т.е. слева не переменная, то работаем по другим комбинациям далее if (Op == '=' && LHS->setValue(rhs)) { // если установка значения не поддерживается, т.е. слева не переменная, то работаем по другим комбинациям далее
@@ -189,21 +198,26 @@ class CallExprAST : public ExprAST {
std::vector<ExprAST*> Args; std::vector<ExprAST*> Args;
IoTItem *Item; // ссылка на объект модуля (прямой доступ к идентификатору указанному в сценарии), если получилось найти модуль по ID IoTItem *Item; // ссылка на объект модуля (прямой доступ к идентификатору указанному в сценарии), если получилось найти модуль по ID
IoTValue ret; // хранение возвращаемого значения, т.к. возврат по ссылке осуществляется IoTValue ret; // хранение возвращаемого значения, т.к. возврат по ссылке осуществляется
bool ItemIsLocal = false;
public: public:
CallExprAST(const String &callee, String &cmd, std::vector<ExprAST*> &args, IoTItem *item) CallExprAST(const String &callee, String &cmd, std::vector<ExprAST*> &args, IoTItem *item)
: Callee(callee), Cmd(cmd), Args(args), Item(item) {} : Callee(callee), Cmd(cmd), Args(args), Item(item) {
if (item) ItemIsLocal = item->iAmLocal;
}
IoTValue* exec() { IoTValue* exec() {
if (Item) { if (!ItemIsLocal) Item = findIoTItem(Callee);
if (!Item) return nullptr; //ret = zeroIotVal;
std::vector<IoTValue> ArgsAsIoTValue; std::vector<IoTValue> ArgsAsIoTValue;
for (unsigned int i = 0; i < Args.size(); i++) { for (unsigned int i = 0; i < Args.size(); i++) {
if (Args[i] == nullptr) return nullptr;
IoTValue *tmp = Args[i]->exec(); IoTValue *tmp = Args[i]->exec();
if (tmp != nullptr) ArgsAsIoTValue.push_back(*tmp); if (tmp != nullptr) ArgsAsIoTValue.push_back(*tmp);
else ArgsAsIoTValue.push_back(zeroIotVal); else return nullptr; //ArgsAsIoTValue.push_back(zeroIotVal);
} }
ret = Item->execute(Cmd, ArgsAsIoTValue); // вызываем команду из модуля напрямую с передачей всех аргументов ret = Item->execute(Cmd, ArgsAsIoTValue); // вызываем команду из модуля напрямую с передачей всех аргументов
} else ret = zeroIotVal;
if (ret.isDecimal) Serial.printf("Call from CallExprAST ID = %s, Command = %s, exec result = %f\n", Callee.c_str(), Cmd.c_str(), ret.valD); if (ret.isDecimal) Serial.printf("Call from CallExprAST ID = %s, Command = %s, exec result = %f\n", Callee.c_str(), Cmd.c_str(), ret.valD);
else Serial.printf("Call from CallExprAST ID = %s, Command = %s, exec result = %s\n", Callee.c_str(), Cmd.c_str(), ret.valS.c_str()); else Serial.printf("Call from CallExprAST ID = %s, Command = %s, exec result = %s\n", Callee.c_str(), Cmd.c_str(), ret.valS.c_str());
@@ -241,11 +255,16 @@ public:
if (!cond_ret) { if (!cond_ret) {
Serial.printf("Call from IfExprAST: Skip If\n"); Serial.printf("Call from IfExprAST: Skip If\n");
return &zeroIotVal; return nullptr; //&zeroIotVal;
} }
if (cond_ret->isDecimal && cond_ret->valD) res_ret = Then->exec(); if (cond_ret->isDecimal && cond_ret->valD) {
else if (Else) res_ret = Else->exec(); if (Then == nullptr) return nullptr;
res_ret = Then->exec();
} else {
if (Else == nullptr) return nullptr;
res_ret = Else->exec();
}
if (!res_ret) Serial.printf("Call from IfExprAST: Cond result = %f, no body result\n", cond_ret->valD); if (!res_ret) Serial.printf("Call from IfExprAST: Cond result = %f, no body result\n", cond_ret->valD);
else if (res_ret->isDecimal) Serial.printf("Call from IfExprAST: Cond result = %f, result = %f\n", cond_ret->valD, res_ret->valD); else if (res_ret->isDecimal) Serial.printf("Call from IfExprAST: Cond result = %f, result = %f\n", cond_ret->valD, res_ret->valD);
@@ -275,6 +294,7 @@ public:
IoTValue* lastExecValue = nullptr; IoTValue* lastExecValue = nullptr;
for (unsigned int i = 0; i < BracketsList.size(); i++) { for (unsigned int i = 0; i < BracketsList.size(); i++) {
if (BracketsList[i] == nullptr) return nullptr;
lastExecValue = BracketsList[i]->exec(); lastExecValue = BracketsList[i]->exec();
} }
@@ -442,8 +462,11 @@ public:
} }
if (CurTok != '(') { // Обычная переменная. if (CurTok != '(') { // Обычная переменная.
if (tmpItem) return new VariableExprAST(IdName, tmpItem); // if (tmpItem) return new VariableExprAST(IdName, tmpItem);
else return new StringExprAST("id " + IdName + " not_found"); // else return new StringExprAST("id " + IdName + " not_found");
// создаем экземпляр переменной в любом случае, даж если не нашли (tmpItem = nulptr), т.к. переменная может придти из сети позже
return new VariableExprAST(IdName, tmpItem);
} }
// Вызов функции. // Вызов функции.