// Copyright Pacificao. All Rights Reserved. #include "AgrarianWeatherProviderSubsystem.h" #include "AgrarianGameState.h" #include "Dom/JsonObject.h" #include "Engine/World.h" #include "HttpModule.h" #include "Interfaces/IHttpResponse.h" #include "Kismet/GameplayStatics.h" #include "Serialization/JsonReader.h" #include "Serialization/JsonSerializer.h" bool UAgrarianWeatherProviderSubsystem::RequestWeatherForActiveGameState() { UWorld* World = GetWorld(); AAgrarianGameState* GameState = World ? World->GetGameState() : nullptr; if (!GameState || !GameState->HasAuthority()) { return false; } return RequestWeatherForTile(GameState->ActiveSolarTileId, GameState->ActiveTileLatitude, GameState->ActiveTileLongitude); } bool UAgrarianWeatherProviderSubsystem::RequestWeatherForTile(FName TileId, float Latitude, float Longitude) { if (!bEnableLiveWeatherRequests || TileId == NAME_None) { return false; } UWorld* World = GetWorld(); if (!World || !World->GetAuthGameMode()) { return false; } const FString Url = BuildOpenMeteoForecastUrl(TileId, Latitude, Longitude); TSharedRef Request = FHttpModule::Get().CreateRequest(); Request->SetURL(Url); Request->SetVerb(TEXT("GET")); Request->SetHeader(TEXT("Accept"), TEXT("application/json")); Request->SetHeader(TEXT("User-Agent"), TEXT("AgrarianGameMVP/0.1")); Request->OnProcessRequestComplete().BindUObject(this, &UAgrarianWeatherProviderSubsystem::OnOpenMeteoResponse, TileId, Latitude, Longitude); return Request->ProcessRequest(); } FString UAgrarianWeatherProviderSubsystem::BuildOpenMeteoForecastUrl(FName TileId, float Latitude, float Longitude) const { const float ClampedLatitude = FMath::Clamp(Latitude, -90.0f, 90.0f); const float ClampedLongitude = FMath::Clamp(Longitude, -180.0f, 180.0f); return FString::Printf( TEXT("%s?latitude=%.6f&longitude=%.6f¤t=temperature_2m,relative_humidity_2m,precipitation,rain,showers,snowfall,weather_code,cloud_cover,pressure_msl,wind_speed_10m&daily=temperature_2m_max,temperature_2m_min,weather_code,precipitation_sum,wind_speed_10m_max&forecast_days=1&timezone=auto"), *OpenMeteoForecastEndpoint, ClampedLatitude, ClampedLongitude); } bool UAgrarianWeatherProviderSubsystem::RequestNoaaNwsFallbackForTile(FName TileId, float Latitude, float Longitude) { if (!bEnableNoaaNwsFallbackForUSTiles || TileId == NAME_None || !IsNoaaNwsEligibleCoordinate(Latitude, Longitude)) { return false; } UWorld* World = GetWorld(); if (!World || !World->GetAuthGameMode()) { return false; } TSharedRef Request = FHttpModule::Get().CreateRequest(); Request->SetURL(BuildNoaaNwsPointsUrl(Latitude, Longitude)); Request->SetVerb(TEXT("GET")); Request->SetHeader(TEXT("Accept"), TEXT("application/geo+json")); Request->SetHeader(TEXT("User-Agent"), NoaaNwsUserAgent); Request->OnProcessRequestComplete().BindUObject(this, &UAgrarianWeatherProviderSubsystem::OnNoaaNwsPointsResponse, TileId, Latitude, Longitude); return Request->ProcessRequest(); } bool UAgrarianWeatherProviderSubsystem::IsNoaaNwsEligibleCoordinate(float Latitude, float Longitude) const { return Latitude >= 18.0f && Latitude <= 72.0f && Longitude >= -180.0f && Longitude <= -64.0f; } FString UAgrarianWeatherProviderSubsystem::BuildNoaaNwsPointsUrl(float Latitude, float Longitude) const { return FString::Printf( TEXT("%s/%.4f,%.4f"), *NoaaNwsPointsEndpoint, FMath::Clamp(Latitude, -90.0f, 90.0f), FMath::Clamp(Longitude, -180.0f, 180.0f)); } bool UAgrarianWeatherProviderSubsystem::ApplySnapshotToGameState(const FAgrarianWeatherProviderSnapshot& Snapshot, AAgrarianGameState* GameState) const { if (!Snapshot.bIsValid || !GameState || !GameState->HasAuthority()) { return false; } GameState->SetRegionalTemperatureProfile(Snapshot.DailyLowTemperatureC, Snapshot.DailyHighTemperatureC); GameState->SetRegionalObservedTemperature( Snapshot.CurrentTemperatureC, 1.0f, FString::Printf(TEXT("%s:%s"), *Snapshot.Provider, *Snapshot.ProviderTimestamp)); GameState->SetWeather(Snapshot.MappedWeather); return true; } EAgrarianWeatherType UAgrarianWeatherProviderSubsystem::MapOpenMeteoWeatherCode(int32 WeatherCode, float PrecipitationMm, float WindSpeedKmh) { if (WeatherCode >= 95 || WindSpeedKmh >= 55.0f) { return EAgrarianWeatherType::Storm; } if (WeatherCode == 71 || WeatherCode == 73 || WeatherCode == 75 || WeatherCode == 77 || WeatherCode == 85 || WeatherCode == 86) { return EAgrarianWeatherType::ColdWind; } if ((WeatherCode >= 51 && WeatherCode <= 67) || (WeatherCode >= 80 && WeatherCode <= 82) || PrecipitationMm > 0.05f) { return EAgrarianWeatherType::Rain; } if (WindSpeedKmh >= 35.0f) { return EAgrarianWeatherType::ColdWind; } return EAgrarianWeatherType::Clear; } void UAgrarianWeatherProviderSubsystem::OnOpenMeteoResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful, FName TileId, float Latitude, float Longitude) { if (!bWasSuccessful || !Response.IsValid() || Response->GetResponseCode() < 200 || Response->GetResponseCode() >= 300) { return; } FAgrarianWeatherProviderSnapshot Snapshot; if (!ParseOpenMeteoForecast(Response->GetContentAsString(), TileId, Latitude, Longitude, Snapshot)) { return; } LastSnapshot = Snapshot; UWorld* World = GetWorld(); AAgrarianGameState* GameState = World ? World->GetGameState() : nullptr; ApplySnapshotToGameState(Snapshot, GameState); } void UAgrarianWeatherProviderSubsystem::OnNoaaNwsPointsResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful, FName TileId, float Latitude, float Longitude) { if (!bWasSuccessful || !Response.IsValid() || Response->GetResponseCode() < 200 || Response->GetResponseCode() >= 300) { return; } TSharedPtr RootObject; const TSharedRef> Reader = TJsonReaderFactory<>::Create(Response->GetContentAsString()); if (!FJsonSerializer::Deserialize(Reader, RootObject) || !RootObject.IsValid()) { return; } const TSharedPtr* PropertiesObject = nullptr; if (!RootObject->TryGetObjectField(TEXT("properties"), PropertiesObject) || !PropertiesObject || !PropertiesObject->IsValid()) { return; } FString ForecastGridDataUrl; if (!(*PropertiesObject)->TryGetStringField(TEXT("forecastGridData"), ForecastGridDataUrl) || ForecastGridDataUrl.IsEmpty()) { return; } LastNoaaNwsForecastGridDataUrl = ForecastGridDataUrl; TSharedRef GridRequest = FHttpModule::Get().CreateRequest(); GridRequest->SetURL(ForecastGridDataUrl); GridRequest->SetVerb(TEXT("GET")); GridRequest->SetHeader(TEXT("Accept"), TEXT("application/geo+json")); GridRequest->SetHeader(TEXT("User-Agent"), NoaaNwsUserAgent); GridRequest->OnProcessRequestComplete().BindUObject(this, &UAgrarianWeatherProviderSubsystem::OnNoaaNwsGridDataResponse, TileId, Latitude, Longitude); GridRequest->ProcessRequest(); } void UAgrarianWeatherProviderSubsystem::OnNoaaNwsGridDataResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful, FName TileId, float Latitude, float Longitude) { if (!bWasSuccessful || !Response.IsValid() || Response->GetResponseCode() < 200 || Response->GetResponseCode() >= 300) { return; } FAgrarianWeatherProviderSnapshot Snapshot; if (!ParseNoaaNwsGridData(Response->GetContentAsString(), TileId, Latitude, Longitude, Snapshot)) { return; } LastSnapshot = Snapshot; UWorld* World = GetWorld(); AAgrarianGameState* GameState = World ? World->GetGameState() : nullptr; ApplySnapshotToGameState(Snapshot, GameState); } bool UAgrarianWeatherProviderSubsystem::ParseOpenMeteoForecast(const FString& ResponseContent, FName TileId, float Latitude, float Longitude, FAgrarianWeatherProviderSnapshot& OutSnapshot) const { TSharedPtr RootObject; const TSharedRef> Reader = TJsonReaderFactory<>::Create(ResponseContent); if (!FJsonSerializer::Deserialize(Reader, RootObject) || !RootObject.IsValid()) { return false; } const TSharedPtr* CurrentObject = nullptr; if (!RootObject->TryGetObjectField(TEXT("current"), CurrentObject) || !CurrentObject || !CurrentObject->IsValid()) { return false; } const TSharedPtr* DailyObject = nullptr; if (!RootObject->TryGetObjectField(TEXT("daily"), DailyObject) || !DailyObject || !DailyObject->IsValid()) { return false; } const TArray>* DailyLowValues = nullptr; const TArray>* DailyHighValues = nullptr; if (!(*DailyObject)->TryGetArrayField(TEXT("temperature_2m_min"), DailyLowValues) || !(*DailyObject)->TryGetArrayField(TEXT("temperature_2m_max"), DailyHighValues) || !DailyLowValues || !DailyHighValues || DailyLowValues->Num() == 0 || DailyHighValues->Num() == 0) { return false; } double CurrentTemperature = 0.0; double RelativeHumidity = 0.0; double Precipitation = 0.0; double Rain = 0.0; double Showers = 0.0; double Snowfall = 0.0; double WeatherCode = 0.0; double CloudCover = 0.0; double Pressure = 0.0; double WindSpeed = 0.0; (*CurrentObject)->TryGetNumberField(TEXT("temperature_2m"), CurrentTemperature); (*CurrentObject)->TryGetNumberField(TEXT("relative_humidity_2m"), RelativeHumidity); (*CurrentObject)->TryGetNumberField(TEXT("precipitation"), Precipitation); (*CurrentObject)->TryGetNumberField(TEXT("rain"), Rain); (*CurrentObject)->TryGetNumberField(TEXT("showers"), Showers); (*CurrentObject)->TryGetNumberField(TEXT("snowfall"), Snowfall); (*CurrentObject)->TryGetNumberField(TEXT("weather_code"), WeatherCode); (*CurrentObject)->TryGetNumberField(TEXT("cloud_cover"), CloudCover); (*CurrentObject)->TryGetNumberField(TEXT("pressure_msl"), Pressure); (*CurrentObject)->TryGetNumberField(TEXT("wind_speed_10m"), WindSpeed); OutSnapshot.TileId = TileId; OutSnapshot.Latitude = FMath::Clamp(Latitude, -90.0f, 90.0f); OutSnapshot.Longitude = FMath::Clamp(Longitude, -180.0f, 180.0f); OutSnapshot.Provider = TEXT("open-meteo"); (*CurrentObject)->TryGetStringField(TEXT("time"), OutSnapshot.ProviderTimestamp); OutSnapshot.CurrentTemperatureC = static_cast(CurrentTemperature); OutSnapshot.DailyLowTemperatureC = static_cast((*DailyLowValues)[0]->AsNumber()); OutSnapshot.DailyHighTemperatureC = static_cast((*DailyHighValues)[0]->AsNumber()); OutSnapshot.PrecipitationMm = static_cast(FMath::Max(Precipitation, FMath::Max(Rain + Showers, Snowfall))); OutSnapshot.WindSpeedKmh = static_cast(WindSpeed); OutSnapshot.CloudCoverPercent = static_cast(CloudCover); OutSnapshot.RelativeHumidityPercent = static_cast(RelativeHumidity); OutSnapshot.PressureMslHpa = static_cast(Pressure); OutSnapshot.WeatherCode = FMath::RoundToInt(WeatherCode); OutSnapshot.MappedWeather = MapOpenMeteoWeatherCode(OutSnapshot.WeatherCode, OutSnapshot.PrecipitationMm, OutSnapshot.WindSpeedKmh); OutSnapshot.bIsValid = true; return true; } bool UAgrarianWeatherProviderSubsystem::ParseNoaaNwsGridData(const FString& ResponseContent, FName TileId, float Latitude, float Longitude, FAgrarianWeatherProviderSnapshot& OutSnapshot) const { TSharedPtr RootObject; const TSharedRef> Reader = TJsonReaderFactory<>::Create(ResponseContent); if (!FJsonSerializer::Deserialize(Reader, RootObject) || !RootObject.IsValid()) { return false; } const TSharedPtr* PropertiesObject = nullptr; if (!RootObject->TryGetObjectField(TEXT("properties"), PropertiesObject) || !PropertiesObject || !PropertiesObject->IsValid()) { return false; } auto ReadFirstGridValue = [](const TSharedPtr& Properties, const TCHAR* FieldName, double& OutValue) -> bool { const TSharedPtr* FieldObject = nullptr; if (!Properties->TryGetObjectField(FieldName, FieldObject) || !FieldObject || !FieldObject->IsValid()) { return false; } const TArray>* Values = nullptr; if (!(*FieldObject)->TryGetArrayField(TEXT("values"), Values) || !Values || Values->Num() == 0) { return false; } const TSharedPtr FirstValue = (*Values)[0]->AsObject(); return FirstValue.IsValid() && FirstValue->TryGetNumberField(TEXT("value"), OutValue); }; double TemperatureC = 0.0; if (!ReadFirstGridValue(*PropertiesObject, TEXT("temperature"), TemperatureC)) { return false; } double PrecipitationPercent = 0.0; double WindSpeedKmh = 0.0; ReadFirstGridValue(*PropertiesObject, TEXT("probabilityOfPrecipitation"), PrecipitationPercent); ReadFirstGridValue(*PropertiesObject, TEXT("windSpeed"), WindSpeedKmh); OutSnapshot.TileId = TileId; OutSnapshot.Latitude = FMath::Clamp(Latitude, -90.0f, 90.0f); OutSnapshot.Longitude = FMath::Clamp(Longitude, -180.0f, 180.0f); OutSnapshot.Provider = TEXT("noaa-nws"); OutSnapshot.ProviderTimestamp = FDateTime::UtcNow().ToIso8601(); OutSnapshot.CurrentTemperatureC = static_cast(TemperatureC); OutSnapshot.DailyLowTemperatureC = static_cast(TemperatureC - 4.0); OutSnapshot.DailyHighTemperatureC = static_cast(TemperatureC + 4.0); OutSnapshot.PrecipitationMm = PrecipitationPercent > 30.0 ? 0.1f : 0.0f; OutSnapshot.WindSpeedKmh = static_cast(WindSpeedKmh); OutSnapshot.WeatherCode = PrecipitationPercent > 30.0 ? 61 : 0; OutSnapshot.MappedWeather = MapOpenMeteoWeatherCode(OutSnapshot.WeatherCode, OutSnapshot.PrecipitationMm, OutSnapshot.WindSpeedKmh); OutSnapshot.bIsValid = true; return true; }