From a506e897b554b143661437f4411e06d7c0179096 Mon Sep 17 00:00:00 2001 From: Matthias Meulien Date: Wed, 11 Oct 2023 12:01:08 +0200 Subject: [PATCH] Dump/Restore most of model in application state Refs: #57 --- src/app.cc | 19 ++-- src/convert.cc | 281 ++++++++++++++++++++++++++++++++++++++++++++++++ src/convert.h | 21 ++++ src/meson.build | 1 + src/model.h | 5 +- src/state.cc | 56 ++-------- src/state.h | 4 +- 7 files changed, 327 insertions(+), 60 deletions(-) create mode 100644 src/convert.cc create mode 100644 src/convert.h diff --git a/src/app.cc b/src/app.cc index 7a3f47a..b7363a0 100644 --- a/src/app.cc +++ b/src/app.cc @@ -152,18 +152,19 @@ void App::load_config() { initialize_translations(); } - const bool is_data_obsolete = - not config_already_loaded or is_api_key_obsolete or - is_unit_system_obsolete or is_language_obsolete; + const bool is_data_obsolete = not config_already_loaded or + is_api_key_obsolete or + is_unit_system_obsolete or is_language_obsolete; // temperatures, wind speed and weather description are computed // by the backend thus unit system or language change implies that // data are obsolete const auto event_handler = GetEventHandler(); if (is_data_obsolete) { - const auto context = config_already_loaded ? - CustomEventParam::triggered_by_configuration_change : - CustomEventParam::triggered_by_application_startup; + const auto context = + config_already_loaded + ? CustomEventParam::triggered_by_configuration_change + : CustomEventParam::triggered_by_application_startup; SendEvent(event_handler, EVT_CUSTOM, CustomEvent::refresh_data, context); } else if (is_display_daily_forecast_obsolete) { SendEvent(event_handler, EVT_CUSTOM, @@ -196,8 +197,8 @@ int App::handle_custom_event(int param_one, int param_two) { } } else if (param_one == CustomEvent::refresh_data) { const bool force_refresh = - (param_two == CustomEventParam::triggered_by_configuration_change or - param_two == CustomEventParam::triggered_by_model_change); + (param_two == CustomEventParam::triggered_by_configuration_change or + param_two == CustomEventParam::triggered_by_model_change); // At application startup or when refresh timer triggers, one must // respect flight mode being enabled this->refresh_data(force_refresh); @@ -269,7 +270,7 @@ void App::clear_model_weather_conditions() { bool App::must_skip_data_refresh() const { if (IsFlightModeEnabled()) { BOOST_LOG_TRIVIAL(info) - << "Won't refresh data since device has flight mode enabled"; + << "Won't refresh data since device has flight mode enabled"; return true; } return false; diff --git a/src/convert.cc b/src/convert.cc new file mode 100644 index 0000000..c1a5b75 --- /dev/null +++ b/src/convert.cc @@ -0,0 +1,281 @@ +#include "convert.h" +#include "experimental/optional" +#include "model.h" + +namespace taranis { + +// NAN is written as null by jsoncpp, this will fix deserialization + +inline double deserialize_possible_null(const Json::Value &value) { + if (value.isNull()) { + return NAN; + } + return value.asDouble(); +} + +Json::Value to_json(const Location &location) { + Json::Value value; + value["longitude"] = location.longitude; + value["latitude"] = location.latitude; + value["name"] = location.name; + value["country"] = location.country; + value["state"] = location.state; + + return value; +} + +Json::Value to_json(const Condition &condition) { + Json::Value value; + value["date"] = static_cast(condition.date); + value["sunrise"] = static_cast(condition.sunrise); + value["sunset"] = static_cast(condition.sunset); + value["temperature"] = condition.temperature; + value["felt_temperature"] = condition.felt_temperature; + value["pressure"] = condition.pressure; + value["humidity"] = condition.humidity; + value["uv_index"] = condition.uv_index; + value["clouds"] = condition.clouds; + value["visibility"] = condition.visibility; + value["probability_of_precipitation"] = + condition.probability_of_precipitation; + value["wind_speed"] = condition.wind_speed; + value["wind_degree"] = condition.wind_degree; + value["wind_gust"] = condition.wind_gust; + value["weather"] = condition.weather; + value["weather_description"] = condition.weather_description; + value["weather_icon_name"] = condition.weather_icon_name; + value["rain"] = condition.rain; + value["snow"] = condition.snow; + + return value; +} + +Json::Value to_json(const DailyCondition &condition) { + Json::Value value; + value["date"] = static_cast(condition.date); + value["sunrise"] = static_cast(condition.sunrise); + value["sunset"] = static_cast(condition.sunset); + value["moonrise"] = static_cast(condition.moonrise); + value["moonset"] = static_cast(condition.moonset); + value["moon_phase"] = condition.moon_phase; + value["pressure"] = condition.pressure; + value["humidity"] = condition.humidity; + value["dew_point"] = condition.dew_point; + value["wind_speed"] = condition.wind_speed; + value["wind_degree"] = condition.wind_degree; + value["wind_gust"] = condition.wind_gust; + value["clouds"] = condition.clouds; + value["probability_of_precipitation"] = + condition.probability_of_precipitation; + value["uv_index"] = condition.uv_index; + value["rain"] = condition.rain; + value["snow"] = condition.snow; + value["weather"] = condition.weather; + value["weather_description"] = condition.weather_description; + value["weather_icon_name"] = condition.weather_icon_name; + value["temperature_day"] = condition.temperature_day; + value["temperature_min"] = condition.temperature_min; + value["temperature_max"] = condition.temperature_max; + value["temperature_night"] = condition.temperature_night; + value["temperature_evening"] = condition.temperature_evening; + value["temperature_morning"] = condition.temperature_morning; + value["felt_temperature_day"] = condition.felt_temperature_day; + value["felt_temperature_night"] = condition.felt_temperature_night; + value["felt_temperature_evening"] = condition.felt_temperature_evening; + value["felt_temperature_morning"] = condition.felt_temperature_morning; + + return value; +} + +Json::Value to_json(const Alert &alert) { + Json::Value value; + value["sender"] = alert.sender; + value["event"] = alert.event; + value["start_date"] = static_cast(alert.start_date); + value["end_date"] = static_cast(alert.end_date); + value["description"] = alert.description; + return value; +} + +Json::Value to_json(const HistoryItem &item) { + Json::Value value; + value["location"] = to_json(item.location); + value["favorite"] = item.favorite; + + return value; +} + +Json::Value to_json(const Model &model) { + Json::Value value; + value["source"] = model.source; + value["unit_system"] = model.unit_system; + value["refresh_date"] = static_cast(model.refresh_date); + value["location"] = to_json(model.location); + + if (model.current_condition) { + value["current_condition"] = to_json(*(model.current_condition)); + } else { + value["current_condition"] = Json::Value::null; + } + for (const auto &forecast : model.hourly_forecast) { + value["hourly_forecast"].append(to_json(forecast)); + } + for (const auto &forecast : model.daily_forecast) { + value["daily_forecast"].append(to_json(forecast)); + } + for (const auto &alert : model.alerts) { + value["alerts"].append(to_json(alert)); + } + for (const auto &item : model.location_history) { + value["location_history"].append(to_json(item)); + } + return value; +} + +void update_from_json(Location &location, const Json::Value &value) { + location.longitude = deserialize_possible_null(value["longitude"]); + location.latitude = deserialize_possible_null(value["latitude"]); + location.name = value.get("name", "").asString(); + location.country = value.get("country", "").asString(); + location.state = value.get("state", "").asString(); +} + +void update_from_json(Condition &condition, const Json::Value &value) { + condition.date = + static_cast(value.get("date", 0).asLargestInt()); + condition.sunrise = + static_cast(value.get("sunrise", 0).asLargestInt()); + condition.sunset = + static_cast(value.get("sunset", 0).asLargestInt()); + condition.temperature = deserialize_possible_null(value["temperature"]); + condition.felt_temperature = + deserialize_possible_null(value["felt_temperature"]); + condition.pressure = value.get("pressure", 0).asInt(); + condition.humidity = value.get("humidity", 0).asInt(); + condition.uv_index = deserialize_possible_null(value["uv_index"]); + condition.clouds = value.get("clouds", 0).asInt(); + condition.visibility = value.get("visibility", 0).asInt(); + condition.probability_of_precipitation = + deserialize_possible_null(value["probability_of_precipitation"]); + condition.wind_speed = deserialize_possible_null(value["wind_speed"]); + condition.wind_degree = value.get("wind_degree", 0).asInt(); + condition.wind_gust = deserialize_possible_null(value["wind_gust"]); + condition.weather = + static_cast(value.get("weather", CLEAR_SKY).asInt()); + condition.weather_description = + value.get("weather_description", "").asString(); + condition.weather_icon_name = value.get("weather_icon_name", "").asString(); + condition.rain = deserialize_possible_null(value["rain"]); + condition.snow = deserialize_possible_null(value["snow"]); +} + +void update_from_json(DailyCondition &condition, const Json::Value &value) { + condition.date = + static_cast(value.get("date", 0).asLargestInt()); + condition.sunrise = + static_cast(value.get("sunrise", 0).asLargestInt()); + condition.sunset = + static_cast(value.get("sunset", 0).asLargestInt()); + condition.moonrise = + static_cast(value.get("moonrise", 0).asLargestInt()); + condition.moonset = + static_cast(value.get("moonset", 0).asLargestInt()); + condition.moon_phase = deserialize_possible_null(value["moon_phase"]); + condition.pressure = value.get("pressure", 0).asInt(); + condition.humidity = value.get("humidity", 0).asInt(); + condition.dew_point = deserialize_possible_null(value["dew_point"]); + condition.wind_speed = deserialize_possible_null(value["wind_speed"]); + condition.wind_degree = value.get("wind_degree", 0).asInt(); + condition.wind_gust = deserialize_possible_null(value["wind_gust"]); + condition.clouds = value.get("clouds", 0).asInt(); + condition.probability_of_precipitation = + deserialize_possible_null(value["probability_of_precipitation"]); + condition.uv_index = deserialize_possible_null(value["uv_index"]); + condition.rain = deserialize_possible_null(value["rain"]); + condition.snow = deserialize_possible_null(value["snow"]); + condition.weather = + static_cast(value.get("weather", CLEAR_SKY).asInt()); + condition.weather_description = + value.get("weather_description", "").asString(); + condition.weather_icon_name = value.get("weather_icon_name", "").asString(); + condition.temperature_day = + deserialize_possible_null(value["temperature_day"]); + condition.temperature_min = + deserialize_possible_null(value["temperature_min"]); + condition.temperature_max = + deserialize_possible_null(value["temperature_max"]); + condition.temperature_night = + deserialize_possible_null(value["temperature_night"]); + condition.temperature_evening = + deserialize_possible_null(value["temperature_evening"]); + condition.temperature_morning = + deserialize_possible_null(value["temperature_morning"]); + condition.felt_temperature_day = + deserialize_possible_null(value["felt_temperature_day"]); + condition.felt_temperature_night = + deserialize_possible_null(value["felt_temperature_night"]); + condition.felt_temperature_evening = + deserialize_possible_null(value["felt_temperature_evening"]); + condition.felt_temperature_morning = + deserialize_possible_null(value["felt_temperature_morning"]); +} + +void update_from_json(Alert &alert, const Json::Value &value) { + alert.sender = value.get("sender", "").asString(); + alert.event = value.get("event", "").asString(); + alert.start_date = + static_cast(value.get("start_date", 0).asLargestInt()); + alert.end_date = + static_cast(value.get("end_date", 0).asLargestInt()); + alert.description = value.get("description", "").asString(); +} + +void update_from_json(HistoryItem &item, const Json::Value &value) { + auto &location = item.location; + const auto &location_value = value["location"]; + update_from_json(location, location_value); + + item.favorite = value.get("favorite", false).asBool(); +} + +void update_from_json(Model &model, const Json::Value &value) { + model.source = value.get("source", "OpenWeather").asString(); + model.unit_system = + static_cast(value.get("unit_system", standard).asInt()); + model.refresh_date = + static_cast(value.get("refresh_date", 0).asLargestInt()); + + auto &location = model.location; + const auto &location_value = value["location"]; + update_from_json(location, location_value); + + if (value["current_condition"].isNull()) { + model.current_condition = std::experimental::nullopt; + } else { + Condition current_condition{}; + const auto ¤t_condition_value = value["current_condition"]; + update_from_json(current_condition, current_condition_value); + model.current_condition = current_condition; + } + for (const auto &condition_value : value["hourly_forecast"]) { + Condition condition{}; + update_from_json(condition, condition_value); + model.hourly_forecast.push_back(condition); + } + for (const auto &condition_value : value["daily_forecast"]) { + DailyCondition condition{}; + update_from_json(condition, condition_value); + model.daily_forecast.push_back(condition); + } + for (const auto &alert_value : value["alerts"]) { + Alert alert{}; + update_from_json(alert, alert_value); + model.alerts.push_back(alert); + } + for (const auto &item_value : value["location_history"]) { + HistoryItem item{}; + update_from_json(item, item_value); + model.location_history.push_back(item); + } +} +} // namespace taranis diff --git a/src/convert.h b/src/convert.h new file mode 100644 index 0000000..5879176 --- /dev/null +++ b/src/convert.h @@ -0,0 +1,21 @@ +#pragma once + +#include + +#include "model.h" + +namespace taranis { +Json::Value to_json(const Location &location); +Json::Value to_json(const Condition &condition); +Json::Value to_json(const DailyCondition &condition); +Json::Value to_json(const Alert &alert); +Json::Value to_json(const HistoryItem &item); +Json::Value to_json(const Model &model); + +void update_from_json(Location &location, const Json::Value &value); +void update_from_json(Condition &condition, const Json::Value &value); +void update_from_json(DailyCondition &condition, const Json::Value &value); +void update_from_json(Alert &alert, const Json::Value &value); +void update_from_json(HistoryItem &item, const Json::Value &value); +void update_from_json(Model &model, const Json::Value &value); +} // namespace taranis diff --git a/src/meson.build b/src/meson.build index d9589ea..faaacdf 100644 --- a/src/meson.build +++ b/src/meson.build @@ -22,6 +22,7 @@ sources = [about_cc] + files( 'alerts.cc', 'app.cc', 'config.cc', + 'convert.cc', 'currentconditionbox.cc', 'events.cc', 'http.cc', diff --git a/src/model.h b/src/model.h index dfbab13..55e7acf 100644 --- a/src/model.h +++ b/src/model.h @@ -180,11 +180,8 @@ struct Alert { }; struct HistoryItem { - const Location location; + Location location; bool favorite; - - HistoryItem(const Location &location, bool favorite) - : location{location}, favorite{favorite} {} }; struct Model { diff --git a/src/state.cc b/src/state.cc index 58e68d9..fd52a05 100644 --- a/src/state.cc +++ b/src/state.cc @@ -6,10 +6,12 @@ #include #include -constexpr char LOCATION_HISTORY_KEY[]{"location_history"}; +#include "convert.h" namespace taranis { +constexpr char MODEL_KEY[]{"model"}; + void ApplicationState::restore() { BOOST_LOG_TRIVIAL(debug) << "Restoring application state"; @@ -27,7 +29,7 @@ void ApplicationState::restore() { BOOST_LOG_TRIVIAL(error) << "Unexpected application state"; return; } - this->restore_location_history(root); + this->restore_model(root); } void ApplicationState::dump() { @@ -41,7 +43,7 @@ void ApplicationState::dump() { return; } Json::Value root; - this->dump_location_history(root); + this->dump_model(root); output << root << std::endl; if (not output) { @@ -63,50 +65,14 @@ std::string ApplicationState::get_application_state_path() { return application_state_path; } -void ApplicationState::restore_location_history(const Json::Value &root) { - this->model->location_history.clear(); - - for (const auto &value : root[LOCATION_HISTORY_KEY]) { - const auto location_value = value["location"]; - if (!location_value.isObject()) { - continue; - } - - const auto longitude{location_value.get("longitude", NAN).asDouble()}; - const auto latitude{location_value.get("latitude", NAN).asDouble()}; - const auto name{location_value.get("name", "").asString()}; - const auto country{location_value.get("country", "").asString()}; - const auto state{location_value.get("state", "").asString()}; - if (std::isnan(longitude) or std::isnan(latitude) or name.empty() or - country.empty()) { - continue; - } - const Location location{longitude, latitude, name, country, state}; - const HistoryItem item{location, value.get("favorite", false).asBool()}; - this->model->location_history.push_back(item); - } - if (this->model->location_history.size() > 0) { - this->model->location = this->model->location_history.begin()->location; - } +void ApplicationState::restore_model(const Json::Value &root) { + const auto &model_value = root[MODEL_KEY]; + update_from_json(*this->model, model_value); } -void ApplicationState::dump_location_history(Json::Value &root) { - auto &history_value = root[LOCATION_HISTORY_KEY]; - const auto &history = this->model->location_history; - int location_index = 0; - for (const auto &item : history) { - history_value[location_index]["location"]["longitude"] = - Json::Value{item.location.longitude}; - history_value[location_index]["location"]["latitude"] = - Json::Value{item.location.latitude}; - history_value[location_index]["location"]["name"] = item.location.name; - history_value[location_index]["location"]["country"] = - item.location.country; - history_value[location_index]["location"]["state"] = item.location.state; - history_value[location_index]["favorite"] = item.favorite; - - ++location_index; - } +void ApplicationState::dump_model(Json::Value &root) { + auto &model_value = root[MODEL_KEY]; + model_value = to_json(*this->model); } } // namespace taranis diff --git a/src/state.h b/src/state.h index f5e5aba..4dc9b02 100644 --- a/src/state.h +++ b/src/state.h @@ -25,8 +25,8 @@ class ApplicationState { static std::string get_application_state_path(); - void restore_location_history(const Json::Value &root); + void restore_model(const Json::Value &root); - void dump_location_history(Json::Value &root); + void dump_model(Json::Value &root); }; } // namespace taranis