2024-04-27 22:10:24 +03:00
|
|
|
package feed
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"slices"
|
2024-05-02 21:54:20 +03:00
|
|
|
"strings"
|
2024-04-27 22:10:24 +03:00
|
|
|
"time"
|
|
|
|
|
|
|
|
_ "time/tzdata"
|
|
|
|
)
|
|
|
|
|
|
|
|
type PlacesResponseJson struct {
|
|
|
|
Results []PlaceJson
|
|
|
|
}
|
|
|
|
|
|
|
|
type PlaceJson struct {
|
|
|
|
Name string
|
2024-05-02 21:54:20 +03:00
|
|
|
Area string `json:"admin1"`
|
2024-04-27 22:10:24 +03:00
|
|
|
Latitude float64
|
|
|
|
Longitude float64
|
|
|
|
Timezone string
|
|
|
|
Country string
|
|
|
|
location *time.Location
|
|
|
|
}
|
|
|
|
|
|
|
|
type WeatherResponseJson struct {
|
|
|
|
Daily struct {
|
|
|
|
Sunrise []int64 `json:"sunrise"`
|
|
|
|
Sunset []int64 `json:"sunset"`
|
|
|
|
} `json:"daily"`
|
|
|
|
|
|
|
|
Hourly struct {
|
|
|
|
Temperature []float64 `json:"temperature_2m"`
|
|
|
|
PrecipitationProbability []int `json:"precipitation_probability"`
|
|
|
|
} `json:"hourly"`
|
|
|
|
|
|
|
|
Current struct {
|
|
|
|
Temperature float64 `json:"temperature_2m"`
|
|
|
|
ApparentTemperature float64 `json:"apparent_temperature"`
|
|
|
|
WeatherCode int `json:"weather_code"`
|
|
|
|
} `json:"current"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type weatherColumn struct {
|
|
|
|
Temperature int
|
|
|
|
Scale float64
|
|
|
|
HasPrecipitation bool
|
|
|
|
}
|
|
|
|
|
2024-05-02 21:54:20 +03:00
|
|
|
var commonCountryAbbreviations = map[string]string{
|
|
|
|
"US": "United States",
|
|
|
|
"USA": "United States",
|
|
|
|
"UK": "United Kingdom",
|
|
|
|
}
|
|
|
|
|
|
|
|
func expandCountryAbbreviations(name string) string {
|
|
|
|
if expanded, ok := commonCountryAbbreviations[strings.TrimSpace(name)]; ok {
|
|
|
|
return expanded
|
|
|
|
}
|
|
|
|
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
// Separates the location that Open Meteo accepts from the administrative area
|
|
|
|
// which can then be used to filter to the correct place after the list of places
|
|
|
|
// has been retrieved. Also expands abbreviations since Open Meteo does not accept
|
|
|
|
// country names like "US", "USA" and "UK"
|
|
|
|
func parsePlaceName(name string) (string, string) {
|
|
|
|
parts := strings.Split(name, ",")
|
|
|
|
|
|
|
|
if len(parts) == 1 {
|
|
|
|
return name, ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(parts) == 2 {
|
|
|
|
return parts[0] + ", " + expandCountryAbbreviations(parts[1]), ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return parts[0] + ", " + expandCountryAbbreviations(parts[2]), strings.TrimSpace(parts[1])
|
|
|
|
}
|
|
|
|
|
2024-04-27 22:10:24 +03:00
|
|
|
func FetchPlaceFromName(location string) (*PlaceJson, error) {
|
2024-05-02 21:54:20 +03:00
|
|
|
location, area := parsePlaceName(location)
|
|
|
|
requestUrl := fmt.Sprintf("https://geocoding-api.open-meteo.com/v1/search?name=%s&count=10&language=en&format=json", url.QueryEscape(location))
|
2024-04-27 22:10:24 +03:00
|
|
|
request, _ := http.NewRequest("GET", requestUrl, nil)
|
|
|
|
responseJson, err := decodeJsonFromRequest[PlacesResponseJson](defaultClient, request)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not fetch places data: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(responseJson.Results) == 0 {
|
|
|
|
return nil, fmt.Errorf("no places found for %s", location)
|
|
|
|
}
|
|
|
|
|
2024-05-02 21:54:20 +03:00
|
|
|
var place *PlaceJson
|
|
|
|
|
|
|
|
if area != "" {
|
|
|
|
area = strings.ToLower(area)
|
|
|
|
|
|
|
|
for i := range responseJson.Results {
|
|
|
|
if strings.ToLower(responseJson.Results[i].Area) == area {
|
|
|
|
place = &responseJson.Results[i]
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if place == nil {
|
|
|
|
return nil, fmt.Errorf("no place found for %s in %s", location, area)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
place = &responseJson.Results[0]
|
|
|
|
}
|
2024-04-27 22:10:24 +03:00
|
|
|
|
|
|
|
loc, err := time.LoadLocation(place.Timezone)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not load location: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
place.location = loc
|
|
|
|
|
|
|
|
return place, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func barIndexFromHour(h int) int {
|
|
|
|
return h / 2
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: bunch of spaget, refactor
|
2024-04-29 18:29:11 +03:00
|
|
|
func FetchWeatherForPlace(place *PlaceJson, units string) (*Weather, error) {
|
2024-04-27 22:10:24 +03:00
|
|
|
query := url.Values{}
|
2024-04-29 18:29:11 +03:00
|
|
|
var temperatureUnit string
|
|
|
|
|
|
|
|
if units == "imperial" {
|
|
|
|
temperatureUnit = "fahrenheit"
|
|
|
|
} else {
|
|
|
|
temperatureUnit = "celsius"
|
|
|
|
}
|
2024-04-27 22:10:24 +03:00
|
|
|
|
|
|
|
query.Add("latitude", fmt.Sprintf("%f", place.Latitude))
|
|
|
|
query.Add("longitude", fmt.Sprintf("%f", place.Longitude))
|
|
|
|
query.Add("timeformat", "unixtime")
|
|
|
|
query.Add("timezone", place.Timezone)
|
|
|
|
query.Add("forecast_days", "1")
|
2024-05-02 19:14:27 +03:00
|
|
|
query.Add("current", "temperature_2m,apparent_temperature,weather_code")
|
2024-04-27 22:10:24 +03:00
|
|
|
query.Add("hourly", "temperature_2m,precipitation_probability")
|
|
|
|
query.Add("daily", "sunrise,sunset")
|
2024-04-29 18:29:11 +03:00
|
|
|
query.Add("temperature_unit", temperatureUnit)
|
2024-04-27 22:10:24 +03:00
|
|
|
|
|
|
|
requestUrl := "https://api.open-meteo.com/v1/forecast?" + query.Encode()
|
|
|
|
request, _ := http.NewRequest("GET", requestUrl, nil)
|
|
|
|
responseJson, err := decodeJsonFromRequest[WeatherResponseJson](defaultClient, request)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%w: %v", ErrNoContent, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
now := time.Now().In(place.location)
|
|
|
|
bars := make([]weatherColumn, 0, 24)
|
|
|
|
currentBar := barIndexFromHour(now.Hour())
|
|
|
|
sunriseBar := barIndexFromHour(time.Unix(int64(responseJson.Daily.Sunrise[0]), 0).In(place.location).Hour())
|
|
|
|
sunsetBar := barIndexFromHour(time.Unix(int64(responseJson.Daily.Sunset[0]), 0).In(place.location).Hour()) - 1
|
|
|
|
|
|
|
|
if sunsetBar < 0 {
|
|
|
|
sunsetBar = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(responseJson.Hourly.Temperature) == 24 {
|
|
|
|
temperatures := make([]int, 12)
|
|
|
|
precipitations := make([]bool, 12)
|
|
|
|
|
|
|
|
t := responseJson.Hourly.Temperature
|
|
|
|
p := responseJson.Hourly.PrecipitationProbability
|
|
|
|
|
|
|
|
for i := 0; i < 24; i += 2 {
|
|
|
|
if i/2 == currentBar {
|
|
|
|
temperatures[i/2] = int(responseJson.Current.Temperature)
|
|
|
|
} else {
|
|
|
|
temperatures[i/2] = int(math.Round((t[i] + t[i+1]) / 2))
|
|
|
|
}
|
|
|
|
|
|
|
|
precipitations[i/2] = (p[i]+p[i+1])/2 > 75
|
|
|
|
}
|
|
|
|
|
|
|
|
minT := slices.Min(temperatures)
|
|
|
|
maxT := slices.Max(temperatures)
|
|
|
|
|
|
|
|
for i := 0; i < 12; i++ {
|
|
|
|
bars = append(bars, weatherColumn{
|
|
|
|
Temperature: temperatures[i],
|
|
|
|
Scale: float64(temperatures[i]-minT) / float64(maxT-minT),
|
|
|
|
HasPrecipitation: precipitations[i],
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Weather{
|
|
|
|
Temperature: int(responseJson.Current.Temperature),
|
|
|
|
ApparentTemperature: int(responseJson.Current.ApparentTemperature),
|
|
|
|
WeatherCode: responseJson.Current.WeatherCode,
|
|
|
|
CurrentColumn: currentBar,
|
|
|
|
SunriseColumn: sunriseBar,
|
|
|
|
SunsetColumn: sunsetBar,
|
|
|
|
Columns: bars,
|
|
|
|
}, nil
|
|
|
|
}
|