/* * EMS-ESP - https://github.com/emsesp/EMS-ESP * Copyright 2020-2023 Paul Derbyshire * * 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, either version 3 of the License, or * (at your option) any later version. * * 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, see . */ #include "command.h" #include "emsdevice.h" #include "emsesp.h" namespace emsesp { uuid::log::Logger Command::logger_{F_(command), uuid::log::Facility::DAEMON}; std::vector Command::cmdfunctions_; // takes a path and a json body, parses the data and calls the command // the path is leading so if duplicate keys are in the input JSON it will be ignored // the entry point will be either via the Web API (api/) or MQTT (/) // returns a return code and json output uint8_t Command::process(const char * path, const bool is_admin, const JsonObject & input, JsonObject & output) { SUrlParser p; // parse URL for the path names p.parse(path); if (!p.paths().size()) { return message(CommandRet::ERROR, "invalid path", output); } // check first if it's from API, if so strip the "api/" if (p.paths().front() == "api") { p.paths().erase(p.paths().begin()); } else { // not /api, so must be MQTT path. Check for base and remove it. if (!strncmp(path, Mqtt::base().c_str(), Mqtt::base().length())) { char new_path[Mqtt::MQTT_TOPIC_MAX_SIZE]; strlcpy(new_path, path, sizeof(new_path)); p.parse(new_path + Mqtt::base().length() + 1); // re-parse the stripped path } else { return message(CommandRet::ERROR, "unrecognized path", output); // error } } // re-calculate new path // if there is only a path (URL) and no body then error! size_t num_paths = p.paths().size(); if (!num_paths && !input.size()) { return message(CommandRet::ERROR, "missing command in path", output); } std::string cmd_s; int8_t id_n = -1; // default hc // check for a device as first item in the path // if its not a known device (thermostat, boiler etc) look for any special MQTT subscriptions const char * device_s = nullptr; if (!num_paths) { // we must look for the device in the JSON body if (input.containsKey("device")) { device_s = input["device"]; } } else { // extract it from the path device_s = p.paths().front().c_str(); // get the device (boiler, thermostat, system etc) } // validate the device, make sure it exists uint8_t device_type = EMSdevice::device_name_2_device_type(device_s); if (!device_has_commands(device_type)) { LOG_DEBUG("Command failed: unknown device '%s'", device_s); return message(CommandRet::ERROR, "unknown device", output); } // the next value on the path should be the command or entity name const char * command_p = nullptr; if (num_paths == 2) { command_p = p.paths()[1].c_str(); } else if (num_paths == 3) { // concatenate the path into one string as it could be in the format 'hc/XXX' char command[50]; snprintf(command, sizeof(command), "%s/%s", p.paths()[1].c_str(), p.paths()[2].c_str()); command_p = command; } else if (num_paths > 3) { // concatenate the path into one string as it could be in the format 'hc/XXX/attribute' char command[50]; snprintf(command, sizeof(command), "%s/%s/%s", p.paths()[1].c_str(), p.paths()[2].c_str(), p.paths()[3].c_str()); command_p = command; } else { // take it from the JSON if (input.containsKey("entity")) { command_p = input["entity"]; } else if (input.containsKey("cmd")) { command_p = input["cmd"]; } } // some commands may be prefixed with hc. wwc. or hc/ or wwc/ so extract these if they exist // parse_command_string returns the extracted command command_p = parse_command_string(command_p, id_n); if (command_p == nullptr) { // handle dead endpoints like api/system or api/boiler // default to 'info' for SYSTEM, the other devices to 'values' for shortname version if (num_paths < (id_n > 0 ? 4 : 3)) { command_p = device_type == EMSdevice::DeviceType::SYSTEM ? F_(info) : F_(values); } else { return message(CommandRet::NOT_FOUND, "missing or bad command", output); } } // if we don't have an id/hc/wwc try and get it from the JSON input // it's allowed to have no id, and then keep the default to -1 if (id_n == -1) { if (input.containsKey("hc")) { id_n = input["hc"]; } else if (input.containsKey("wwc")) { id_n = input["wwc"]; id_n += DeviceValueTAG::TAG_WWC1 - DeviceValueTAG::TAG_HC1; // wwc1 has id 9 } else if (input.containsKey("id")) { id_n = input["id"]; } else if (input.containsKey("ahs")) { id_n = input["ahs"]; id_n += DeviceValueTAG::TAG_AHS1 - DeviceValueTAG::TAG_HC1; // ahs1 has id 19 } else if (input.containsKey("hs")) { id_n = input["hs"]; id_n += DeviceValueTAG::TAG_HS1 - DeviceValueTAG::TAG_HC1; // hs1 has id 20 } } // the value must always come from the input JSON. It's allowed to be empty. JsonVariant data; if (input.containsKey("data")) { data = input["data"]; } else if (input.containsKey("value")) { data = input["value"]; } // check if data is entity like device/hc/name/value if (data.is()) { const char * d = data.as(); if (strlen(d)) { char * device_end = (char *)strchr(d, '/'); if (device_end != nullptr) { char device_s[15] = {'\0'}; const char * device_p = device_s; const char * data_p = nullptr; strlcpy(device_s, d, device_end - d + 1); data_p = device_end + 1; int8_t id_d = -1; data_p = parse_command_string(data_p, id_d); if (data_p == nullptr) { return CommandRet::INVALID; } char data_s[40]; strlcpy(data_s, Helpers::toLower(data_p).c_str(), 30); if (strstr(data_s, "/value") == nullptr) { strcat(data_s, "/value"); } uint8_t device_type = EMSdevice::device_name_2_device_type(device_p); if (CommandRet::OK != Command::call(device_type, data_s, "", true, id_d, output)) { return CommandRet::INVALID; } if (!output.containsKey("api_data")) { return CommandRet::INVALID; } data = output["api_data"]; output.clear(); } } } // call the command based on the type uint8_t return_code = CommandRet::ERROR; if (data.is()) { return_code = Command::call(device_type, command_p, data.as(), is_admin, id_n, output); } else if (data.is()) { char data_str[10]; return_code = Command::call(device_type, command_p, Helpers::itoa((int16_t)data.as(), data_str), is_admin, id_n, output); } else if (data.is()) { char data_str[10]; return_code = Command::call(device_type, command_p, Helpers::render_value(data_str, data.as(), 2), is_admin, id_n, output); } else if (data.is()) { return_code = Command::call(device_type, command_p, data.as() ? "1" : "0", is_admin, id_n, output); } else if (data.isNull()) { return_code = Command::call(device_type, command_p, "", is_admin, id_n, output); // empty, will do a query instead } else { return message(CommandRet::ERROR, "cannot parse command", output); // can't process } return return_code; } std::string Command::return_code_string(const uint8_t return_code) { switch (return_code) { case CommandRet::ERROR: return "Error"; case CommandRet::OK: return "OK"; case CommandRet::NOT_FOUND: return "Not Found"; case CommandRet::NOT_ALLOWED: return "Not Authorized"; case CommandRet::FAIL: return "Failed"; case CommandRet::INVALID: return "Invalid"; default: break; } char s[4]; return Helpers::smallitoa(s, return_code); } // takes a string like "hc1/seltemp" or "seltemp" or "wwc2.seltemp" and tries to get the id and cmd // returns start position of the command string const char * Command::parse_command_string(const char * command, int8_t & id) { if (command == nullptr) { return nullptr; } // convert cmd to lowercase and compare char * lowerCmd = strdup(command); for (char * p = lowerCmd; *p; p++) { *p = tolower(*p); } // check prefix and valid number range, also check 'id' if (!strncmp(lowerCmd, "hc", 2) && command[2] >= '1' && command[2] <= '8') { id = command[2] - '0'; command += 3; } else if (!strncmp(lowerCmd, "wwc", 3) && command[3] == '1' && command[4] == '0') { id = DeviceValueTAG::TAG_WWC10 - DeviceValueTAG::TAG_HC1 + 1; //18; command += 5; } else if (!strncmp(lowerCmd, "wwc", 3) && command[3] >= '1' && command[3] <= '9') { id = command[3] - '1' + DeviceValueTAG::TAG_WWC1 - DeviceValueTAG::TAG_HC1 + 1; //9; command += 4; } else if (!strncmp(lowerCmd, "id", 2) && command[2] == '1' && command[3] >= '0' && command[3] <= '9') { id = command[3] - '0' + 10; command += 4; } else if (!strncmp(lowerCmd, "id", 2) && command[2] >= '1' && command[2] <= '9') { id = command[2] - '0'; command += 3; } else if (!strncmp(lowerCmd, "ahs", 3) && command[3] >= '1' && command[3] <= '1') { // only ahs1 for now id = command[3] - '1' + DeviceValueTAG::TAG_AHS1 - DeviceValueTAG::TAG_HC1 + 1; // 19; command += 4; } else if (!strncmp(lowerCmd, "hs", 2) && command[2] == '1' && command[3] >= '0' && command[3] <= '6') { id = command[3] - '0' + DeviceValueTAG::TAG_HS10 - DeviceValueTAG::TAG_HC1 + 1; //29; command += 4; } else if (!strncmp(lowerCmd, "hs", 2) && command[2] >= '1' && command[2] <= '9') { id = command[2] - '1' + DeviceValueTAG::TAG_HS1 - DeviceValueTAG::TAG_HC1 + 1; //20; command += 3; } // remove separator if (command[0] == '/' || command[0] == '.' || command[0] == '_') { command++; } free(lowerCmd); // return null for empty command if (command[0] == '\0') { return nullptr; } return command; } // calls a command directly uint8_t Command::call(const uint8_t device_type, const char * cmd, const char * value) { // create a temporary buffer StaticJsonDocument output_doc; JsonObject output = output_doc.to(); // authenticated is always true and ID is the default value return call(device_type, cmd, value, true, -1, output); } // calls a command. Takes a json object for output. // id may be used to represent a heating circuit for example // returns 0 if the command errored, 1 (TRUE) if ok, 2 if not found, 3 if error or 4 if not allowed uint8_t Command::call(const uint8_t device_type, const char * cmd, const char * value, const bool is_admin, const int8_t id, JsonObject & output) { if (cmd == nullptr) { return CommandRet::NOT_FOUND; } uint8_t return_code = CommandRet::OK; auto dname = EMSdevice::device_type_2_device_name(device_type); uint8_t device_id = EMSESP::device_id_from_cmd(device_type, cmd, id); // see if there is a command registered auto cf = find_command(device_type, device_id, cmd); // check if its a call to and end-point to a device // this is used to fetch the attributes of the device entity, or call a command directly bool single_command = (!value || !strlen(value)); if (single_command) { // exception 1: anything that is from System // exception 2: boiler coldshot command bool get_attributes = (!cf || !cf->cmdfunction_json_) && (device_type > EMSdevice::DeviceType::SYSTEM) && (strcmp(cmd, F_(coldshot)) != 0); if (get_attributes) { LOG_DEBUG("Calling %s command '%s' to retrieve attributes", dname, cmd); return EMSESP::get_device_value_info(output, cmd, id, device_type) ? CommandRet::OK : CommandRet::ERROR; // entity = cmd } } // check if we have a matching command if (cf) { // check permissions if (cf->has_flags(CommandFlag::ADMIN_ONLY) && !is_admin) { output["message"] = "authentication failed"; return CommandRet::NOT_ALLOWED; // command not allowed } auto description = Helpers::translated_word(cf->description_); char info_s[100]; if (strlen(description)) { snprintf(info_s, sizeof(info_s), "'%s/%s' (%s)", dname, cmd, description); } else { snprintf(info_s, sizeof(info_s), "'%s/%s'", dname, cmd); } std::string ro = EMSESP::system_.readonly_mode() ? "[readonly] " : ""; if ((value == nullptr) || (strlen(value) == 0)) { LOG_DEBUG(("%sCalling command %s"), ro.c_str(), info_s); } else { if (id > 0) { LOG_DEBUG(("%sCalling command %s with value %s and id %d on device 0x%02X"), ro.c_str(), info_s, value, id, device_id); } else { LOG_DEBUG(("%sCalling command %s with value %s"), ro.c_str(), info_s, value); } } // call the function based on type if (cf->cmdfunction_json_) { return_code = ((cf->cmdfunction_json_)(value, id, output)) ? CommandRet::OK : CommandRet::ERROR; } // check if read-only. This also checks for valid tags (e.g. heating circuits) if (cf->cmdfunction_) { if (!single_command && EMSESP::cmd_is_readonly(device_type, device_id, cmd, id)) { return_code = CommandRet::INVALID; // readonly or invalid hc } else { return_code = ((cf->cmdfunction_)(value, id)) ? CommandRet::OK : CommandRet::ERROR; } } // report back if (return_code != CommandRet::OK) { return message(return_code, "callback function failed", output); } return return_code; } // we didn't find the command and its not an endpoint, report error LOG_DEBUG("Command failed: invalid command '%s'", cmd ? cmd : ""); return message(CommandRet::NOT_FOUND, "invalid command", output); } // add a command to the list, which does not return json void Command::add(const uint8_t device_type, const uint8_t device_id, const char * cmd, const cmd_function_p cb, const char * const * description, uint8_t flags) { // if the command already exists for that device type don't add it if (find_command(device_type, device_id, cmd) != nullptr) { return; } // if the description is empty, it's hidden which means it will not show up in Web API or Console as an available command if (!description) { flags |= CommandFlag::HIDDEN; } cmdfunctions_.emplace_back(device_type, device_id, flags, cmd, cb, nullptr, description); // callback for json is nullptr } // add a command with no json output // system/temperature/analog devices uses device_id 0 void Command::add(const uint8_t device_type, const char * cmd, const cmd_function_p cb, const char * const * description, uint8_t flags) { add(device_type, 0, cmd, cb, description, flags); } // add a command to the list, which does return a json object as output void Command::add(const uint8_t device_type, const char * cmd, const cmd_json_function_p cb, const char * const * description, uint8_t flags) { // if the command already exists for that device type don't add it if (find_command(device_type, 0, cmd) != nullptr) { return; } cmdfunctions_.emplace_back(device_type, 0, flags, cmd, nullptr, cb, description); // callback for json is included } // see if a command exists for that device type // is not case sensitive Command::CmdFunction * Command::find_command(const uint8_t device_type, const uint8_t device_id, const char * cmd) { if ((cmd == nullptr) || (strlen(cmd) == 0) || (cmdfunctions_.empty())) { return nullptr; } for (auto & cf : cmdfunctions_) { if (Helpers::toLower(cmd) == Helpers::toLower(cf.cmd_) && (cf.device_type_ == device_type) && (!device_id || cf.device_id_ == device_id)) { return &cf; } } return nullptr; // command not found } void Command::erase_command(const uint8_t device_type, const char * cmd) { if ((cmd == nullptr) || (strlen(cmd) == 0) || (cmdfunctions_.empty())) { return; } auto it = cmdfunctions_.begin(); for (auto & cf : cmdfunctions_) { if (Helpers::toLower(cmd) == Helpers::toLower(cf.cmd_) && (cf.device_type_ == device_type)) { cmdfunctions_.erase(it); return; } it++; } } // list all commands for a specific device, output as json bool Command::list(const uint8_t device_type, JsonObject & output) { if (cmdfunctions_.empty()) { output["message"] = "no commands available"; return false; } // create a list of commands, sort them std::list sorted_cmds; for (const auto & cf : cmdfunctions_) { if ((cf.device_type_ == device_type) && !cf.has_flags(CommandFlag::HIDDEN)) { sorted_cmds.push_back((cf.cmd_)); } } sorted_cmds.sort(); for (const auto & cl : sorted_cmds) { for (const auto & cf : cmdfunctions_) { if ((cf.device_type_ == device_type) && !cf.has_flags(CommandFlag::HIDDEN) && cf.description_ && (cl == std::string(cf.cmd_))) { if (cf.has_flags(CommandFlag::MQTT_SUB_FLAG_WW)) { char s[100]; snprintf(s, sizeof(s), "%s %s", EMSdevice::tag_to_string(DeviceValueTAG::TAG_DEVICE_DATA_WW), Helpers::translated_word(cf.description_)); output[cl] = s; } else { output[cl] = Helpers::translated_word(cf.description_); } } } } return true; } // output list of all commands to console for a specific DeviceType void Command::show(uuid::console::Shell & shell, uint8_t device_type, bool verbose) { if (cmdfunctions_.empty()) { shell.println("No commands available"); return; } // create a list of commands, sort them std::list sorted_cmds; for (const auto & cf : cmdfunctions_) { if ((cf.device_type_ == device_type) && !cf.has_flags(CommandFlag::HIDDEN)) { sorted_cmds.push_back((cf.cmd_)); } } sorted_cmds.sort(); // if not in verbose mode, just print them on a single line if (!verbose) { for (const auto & cl : sorted_cmds) { shell.print(cl); shell.print(" "); } shell.println(); return; } // verbose mode shell.println(); for (const auto & cl : sorted_cmds) { // find and print the description for (const auto & cf : cmdfunctions_) { if ((cf.device_type_ == device_type) && !cf.has_flags(CommandFlag::HIDDEN) && cf.description_ && (cl == std::string(cf.cmd_))) { uint8_t i = cl.length(); shell.print(" "); if (cf.has_flags(MQTT_SUB_FLAG_HC)) { shell.print("[hc.]"); i += 8; } else if (cf.has_flags(MQTT_SUB_FLAG_WWC)) { shell.print("[wwc.]"); i += 9; } shell.print(cl); // pad with spaces while (i++ < 22) { shell.print(' '); } shell.print(COLOR_BRIGHT_CYAN); if (cf.has_flags(MQTT_SUB_FLAG_WW)) { shell.print(EMSdevice::tag_to_string(DeviceValueTAG::TAG_DEVICE_DATA_WW)); shell.print(' '); } shell.print(Helpers::translated_word(cf.description_)); if (!cf.has_flags(CommandFlag::ADMIN_ONLY)) { shell.print(' '); shell.print(COLOR_BRIGHT_RED); shell.print('*'); } shell.print(COLOR_RESET); } } shell.println(); } shell.println(); } // see if a device_type is active and has associated commands // returns false if the device has no commands bool Command::device_has_commands(const uint8_t device_type) { if (device_type == EMSdevice::DeviceType::UNKNOWN) { return false; } if (device_type == EMSdevice::DeviceType::SYSTEM) { return true; // we always have System } if (device_type == EMSdevice::DeviceType::SCHEDULER) { return EMSESP::webSchedulerService.has_commands(); } if (device_type == EMSdevice::DeviceType::CUSTOM) { return (EMSESP::webEntityService.count_entities() != 0); } if (device_type == EMSdevice::DeviceType::TEMPERATURESENSOR) { return (EMSESP::temperaturesensor_.have_sensors()); } if (device_type == EMSdevice::DeviceType::ANALOGSENSOR) { return (EMSESP::analogsensor_.have_sensors()); } for (const auto & emsdevice : EMSESP::emsdevices) { if (emsdevice && (emsdevice->device_type() == device_type)) { // device found, now see if it has any commands for (const auto & cf : cmdfunctions_) { if (cf.device_type_ == device_type) { return true; } } } } return false; } // list sensors and EMS devices void Command::show_devices(uuid::console::Shell & shell) { shell.printf("%s ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::SYSTEM)); if (EMSESP::webSchedulerService.has_commands()) { shell.printf("%s ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::SCHEDULER)); } if (EMSESP::temperaturesensor_.have_sensors()) { shell.printf("%s ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::TEMPERATURESENSOR)); } if (EMSESP::analogsensor_.have_sensors()) { shell.printf("%s ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::ANALOGSENSOR)); } for (const auto & device_class : EMSFactory::device_handlers()) { for (const auto & emsdevice : EMSESP::emsdevices) { if (emsdevice && (emsdevice->device_type() == device_class.first) && (device_has_commands(device_class.first))) { shell.printf("%s ", EMSdevice::device_type_2_device_name(device_class.first)); break; // we only want to show one (not multiple of the same device types) } } } shell.println(); } // output list of all commands to console // calls show with verbose mode set void Command::show_all(uuid::console::Shell & shell) { shell.println("Available commands (*=authorization not required): "); // show system first shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::SYSTEM)); shell.print(COLOR_RESET); show(shell, EMSdevice::DeviceType::SYSTEM, true); // show Custom if (EMSESP::webEntityService.has_commands()) { shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::CUSTOM)); shell.println(COLOR_RESET); shell.printf(" info: %slists all values %s*", COLOR_BRIGHT_CYAN, COLOR_BRIGHT_RED); shell.println(COLOR_RESET); shell.printf(" commands: %slists all commands %s*", COLOR_BRIGHT_CYAN, COLOR_BRIGHT_RED); shell.print(COLOR_RESET); show(shell, EMSdevice::DeviceType::CUSTOM, true); } // show scheduler if (EMSESP::webSchedulerService.has_commands()) { shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::SCHEDULER)); shell.println(COLOR_RESET); shell.printf(" info: %slists all values %s*", COLOR_BRIGHT_CYAN, COLOR_BRIGHT_RED); shell.println(COLOR_RESET); shell.printf(" commands: %slists all commands %s*", COLOR_BRIGHT_CYAN, COLOR_BRIGHT_RED); shell.print(COLOR_RESET); show(shell, EMSdevice::DeviceType::SCHEDULER, true); } // show sensors if (EMSESP::temperaturesensor_.have_sensors()) { shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::TEMPERATURESENSOR)); shell.print(COLOR_RESET); show(shell, EMSdevice::DeviceType::TEMPERATURESENSOR, true); } if (EMSESP::analogsensor_.have_sensors()) { shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(EMSdevice::DeviceType::ANALOGSENSOR)); shell.print(COLOR_RESET); show(shell, EMSdevice::DeviceType::ANALOGSENSOR, true); } // do this in the order of factory classes to keep a consistent order when displaying for (const auto & device_class : EMSFactory::device_handlers()) { if (Command::device_has_commands(device_class.first)) { shell.print(COLOR_BOLD_ON); shell.print(COLOR_YELLOW); shell.printf(" %s: ", EMSdevice::device_type_2_device_name(device_class.first)); shell.print(COLOR_RESET); show(shell, device_class.first, true); } } } // Extract only the path component from the passed URI and normalized it // e.g. //one/two////three/// becomes /one/two/three std::string SUrlParser::path() { std::string s = "/"; // set up the beginning slash for (const std::string & f : m_folders) { s += f; s += "/"; } s.pop_back(); // deleting last letter, that is slash '/' return std::string(s); } SUrlParser::SUrlParser(const char * uri) { parse(uri); } bool SUrlParser::parse(const char * uri) { if (uri == nullptr) { return false; } if (*uri == '\0') { return false; } m_folders.clear(); m_keysvalues.clear(); enum Type { begin, folder, param, value }; std::string s; const char * c = uri; enum Type t = Type::begin; std::string last_param; do { if (*c == '/') { if (s.length() > 0) { m_folders.push_back(s); s.clear(); } t = Type::folder; } else if (*c == '?' && (t == Type::folder || t == Type::begin)) { if (s.length() > 0) { m_folders.push_back(s); s.clear(); } t = Type::param; } else if (*c == '=' && (t == Type::param || t == Type::begin)) { m_keysvalues[s] = ""; last_param = s; s.clear(); t = Type::value; } else if (*c == '&' && (t == Type::value || t == Type::param || t == Type::begin)) { if (t == Type::value) { m_keysvalues[last_param] = s; } else if ((t == Type::param || t == Type::begin) && (s.length() > 0)) { m_keysvalues[s] = ""; last_param = s; } t = Type::param; s.clear(); } else if (*c == '\0' && s.length() > 0) { if (t == Type::value) { m_keysvalues[last_param] = s; } else if (t == Type::folder || t == Type::begin) { m_folders.push_back(s); } else if (t == Type::param) { m_keysvalues[s] = ""; last_param = s; } s.clear(); } else if (*c == '\0' && s.length() == 0) { if (t == Type::param && last_param.length() > 0) { m_keysvalues[last_param] = ""; } s.clear(); } else { s += *c; } } while (*c++ != '\0'); return true; } } // namespace emsesp