mirror of
https://codeberg.org/andyscott/weather-cli.git
synced 2024-12-21 12:43:09 -05:00
Merge branch 'dev'
This commit is contained in:
commit
4e545f6383
9 changed files with 185 additions and 64 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -27,3 +27,6 @@ go.work
|
||||||
|
|
||||||
# Partner project
|
# Partner project
|
||||||
SimplrWeather/
|
SimplrWeather/
|
||||||
|
|
||||||
|
# Protobuf definitions
|
||||||
|
proto/
|
|
@ -1 +1 @@
|
||||||
Subproject commit 8c4b03510c4a3b89e1adb28d5ee36ad091240f0a
|
Subproject commit da7e32450045132e40ab6c09ff2dbc199f450f9d
|
|
@ -8,7 +8,7 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
func advancedMenu(app *application) {
|
func advancedMenu(app *Application) {
|
||||||
|
|
||||||
var option string
|
var option string
|
||||||
|
|
||||||
|
@ -19,6 +19,7 @@ func advancedMenu(app *application) {
|
||||||
|
|
||||||
fmt.Printf("1. Change units (Default: imperial; Current: %s)\n", app.Config.Units)
|
fmt.Printf("1. Change units (Default: imperial; Current: %s)\n", app.Config.Units)
|
||||||
fmt.Println("2. Enter precise location")
|
fmt.Println("2. Enter precise location")
|
||||||
|
fmt.Println("3. Get historical data")
|
||||||
fmt.Print("0. Back\n\n")
|
fmt.Print("0. Back\n\n")
|
||||||
|
|
||||||
// Read user input
|
// Read user input
|
||||||
|
@ -27,7 +28,6 @@ func advancedMenu(app *application) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
option = strings.TrimSuffix(input, "\n")
|
option = strings.TrimSuffix(input, "\n")
|
||||||
|
|
||||||
// Check user input
|
// Check user input
|
||||||
|
@ -54,7 +54,23 @@ func advancedMenu(app *application) {
|
||||||
fmt.Println("Are you sure the coordinates are valid?")
|
fmt.Println("Are you sure the coordinates are valid?")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
printWeather(app)
|
||||||
|
app.Forecast.Main.Temp = 0.00
|
||||||
|
|
||||||
|
} else if option == "3" {
|
||||||
|
var validLoc bool
|
||||||
|
for !validLoc {
|
||||||
|
getPreciseLocation(app)
|
||||||
|
getDate(app)
|
||||||
|
getHistoricalData(app.Client, app)
|
||||||
|
|
||||||
|
if app.Forecast.Main.Temp != 0.00 {
|
||||||
|
validLoc = true
|
||||||
|
} else {
|
||||||
|
fmt.Println("I couldn't get any information for that location.")
|
||||||
|
fmt.Println("Are you sure the coordinates are valid?")
|
||||||
|
}
|
||||||
|
}
|
||||||
printWeather(app)
|
printWeather(app)
|
||||||
app.Forecast.Main.Temp = 0.00
|
app.Forecast.Main.Temp = 0.00
|
||||||
}
|
}
|
||||||
|
|
72
cmd/history.go
Normal file
72
cmd/history.go
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
pb "codeberg.org/andcscott/weather-cli/proto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Get a date from the user and save it to the config
|
||||||
|
func getDate(app *Application) {
|
||||||
|
|
||||||
|
fmt.Print("Enter 4-digit year: ")
|
||||||
|
_, err := fmt.Scanf("%d", &app.Config.Date.Year)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Invalid year")
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Print("Enter month (1-12): ")
|
||||||
|
_, err = fmt.Scanf("%d", &app.Config.Date.Month)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Invalid year")
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Print("Enter day (1-31): ")
|
||||||
|
_, err = fmt.Scanf("%d", &app.Config.Date.Day)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Invalid year")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query SimplrWeather for date in Unix time
|
||||||
|
func getUnixTime(c pb.RouteGuideClient) int32 {
|
||||||
|
|
||||||
|
var year, month, day int32
|
||||||
|
|
||||||
|
res, err := c.GetUnixTime(context.Background(), &pb.Date{
|
||||||
|
Year: year,
|
||||||
|
Month: month,
|
||||||
|
Day: day,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Error getting Unix time: %v\n", err)
|
||||||
|
}
|
||||||
|
return res.Unixtime
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query SimplrWeather for historical data
|
||||||
|
func getHistoricalData(c pb.RouteGuideClient, app *Application) {
|
||||||
|
|
||||||
|
var lat, lon int
|
||||||
|
lat, _ = strconv.Atoi(app.Config.Latitude)
|
||||||
|
lon, _ = strconv.Atoi(app.Config.Longitude)
|
||||||
|
|
||||||
|
res, err := c.GetHistoricalData(context.Background(), &pb.LocationDate{
|
||||||
|
Latitude: int32(lat),
|
||||||
|
Longitude: int32(lon),
|
||||||
|
Year: app.Config.Date.Year,
|
||||||
|
Month: app.Config.Date.Month,
|
||||||
|
Day: app.Config.Date.Day,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Error getting historical data: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.Unmarshal([]byte(res.Data), &app.Forecast)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Error reading data from server: %v", err)
|
||||||
|
}
|
||||||
|
}
|
|
@ -12,7 +12,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
// Use WAN address to obtain location data
|
// Use WAN address to obtain location data
|
||||||
func getLocation(app *application) {
|
func getLocation(app *Application) {
|
||||||
|
|
||||||
res, err := http.Get("https://ipinfo.io/json")
|
res, err := http.Get("https://ipinfo.io/json")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -37,7 +37,7 @@ func getLocation(app *application) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prompt user for location
|
// Prompt user for location
|
||||||
func getPreciseLocation(app *application) {
|
func getPreciseLocation(app *Application) {
|
||||||
|
|
||||||
fmt.Print("\nEnter latitude: ")
|
fmt.Print("\nEnter latitude: ")
|
||||||
|
|
||||||
|
|
89
cmd/main.go
89
cmd/main.go
|
@ -1,18 +1,18 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
|
||||||
"fmt"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
|
||||||
|
|
||||||
|
pb "codeberg.org/andcscott/weather-cli/proto"
|
||||||
"github.com/joho/godotenv"
|
"github.com/joho/godotenv"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/credentials/insecure"
|
||||||
)
|
)
|
||||||
|
|
||||||
const version string = "0.2.0"
|
const version string = "0.2.0"
|
||||||
|
|
||||||
type weatherMain struct {
|
type Weather struct {
|
||||||
Temp float32 `json:"temp"`
|
Temp float32 `json:"temp"`
|
||||||
FeelsLike float32 `json:"feels_like"`
|
FeelsLike float32 `json:"feels_like"`
|
||||||
HighTemp float32 `json:"temp_max"`
|
HighTemp float32 `json:"temp_max"`
|
||||||
|
@ -21,93 +21,68 @@ type weatherMain struct {
|
||||||
Humidity uint `json:"humidity"`
|
Humidity uint `json:"humidity"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type weatherWind struct {
|
type Wind struct {
|
||||||
Speed float32 `json:"speed"`
|
Speed float32 `json:"speed"`
|
||||||
Gust float32 `json:"gust"`
|
Gust float32 `json:"gust"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type forecast struct {
|
type Forecast struct {
|
||||||
//Overview weatherOverview `json:"weather"`
|
//Overview weatherOverview `json:"weather"`
|
||||||
Main weatherMain `json:"main"`
|
Main Weather `json:"main"`
|
||||||
Wind weatherWind `json:"wind"`
|
Wind Wind `json:"wind"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type config struct {
|
type Date struct {
|
||||||
|
Year int32
|
||||||
|
Month int32
|
||||||
|
Day int32
|
||||||
|
}
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
Units string `json:"units"`
|
Units string `json:"units"`
|
||||||
Location string `json:"loc"`
|
Location string `json:"loc"`
|
||||||
Longitude string `json:"lat"`
|
Longitude string `json:"lat"`
|
||||||
Latitude string `json:"lon"`
|
Latitude string `json:"lon"`
|
||||||
|
Date Date
|
||||||
ApiKey string `json:"appid"`
|
ApiKey string `json:"appid"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type application struct {
|
type Application struct {
|
||||||
Forecast forecast
|
Forecast Forecast
|
||||||
Config config
|
Config Config
|
||||||
Version string
|
Version string
|
||||||
|
Client pb.RouteGuideClient
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
|
||||||
|
conn, err := grpc.Dial("localhost:50051", grpc.WithTransportCredentials(insecure.NewCredentials()))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalln(err)
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
|
||||||
// Load .env
|
// Load .env
|
||||||
err := godotenv.Load()
|
err = godotenv.Load()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalln(err)
|
log.Fatalln(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read API_KEY from .env and create app
|
// Read API_KEY from .env and create app
|
||||||
cfg := config{
|
cfg := Config{
|
||||||
Units: "imperial",
|
Units: "imperial",
|
||||||
ApiKey: os.Getenv("API_KEY"),
|
ApiKey: os.Getenv("API_KEY"),
|
||||||
}
|
}
|
||||||
|
|
||||||
fcst := forecast{}
|
fcst := Forecast{}
|
||||||
|
|
||||||
app := application{
|
app := Application{
|
||||||
Config: cfg,
|
Config: cfg,
|
||||||
Forecast: fcst,
|
Forecast: fcst,
|
||||||
Version: version,
|
Version: version,
|
||||||
|
Client: pb.NewRouteGuideClient(conn),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display main menu
|
mainMenu(&app)
|
||||||
fmt.Println("\n=====================================================")
|
|
||||||
fmt.Printf("| Welcome to the OpenWeatherMap-gRPC Client! v%s |\n", app.Version)
|
|
||||||
fmt.Println("=====================================================")
|
|
||||||
|
|
||||||
fmt.Printf("New in version %s:\n", app.Version)
|
|
||||||
fmt.Println(" - Advanced option: Get historical data back to 1972")
|
|
||||||
fmt.Println("New in version 0.1.0")
|
|
||||||
fmt.Println(" - Default option: Automatically determine location")
|
|
||||||
fmt.Println(" - Advanced option: Enter exact location for anywhere in the world")
|
|
||||||
fmt.Println(" - Advanced option: Change back and forth between imperial and metric units of measurement")
|
|
||||||
|
|
||||||
var option string
|
|
||||||
// Menu loop
|
|
||||||
for option != "0" {
|
|
||||||
fmt.Print("\nMain Menu\n---------\n\n")
|
|
||||||
fmt.Println("1. Today's forecast (use current location, default)")
|
|
||||||
fmt.Println("2. Advanced options (Change units, precise location, etc.)")
|
|
||||||
fmt.Print("0. Exit\n\n")
|
|
||||||
|
|
||||||
// Read user input
|
|
||||||
reader := bufio.NewReader(os.Stdin)
|
|
||||||
input, err := reader.ReadString('\n')
|
|
||||||
if err != nil {
|
|
||||||
log.Println(err)
|
|
||||||
}
|
|
||||||
option = strings.TrimSuffix(input, "\n")
|
|
||||||
|
|
||||||
// Check user input
|
|
||||||
if option == "1" || option == "" {
|
|
||||||
getLocation(&app)
|
|
||||||
getCurrent(&app)
|
|
||||||
printWeather(&app)
|
|
||||||
} else if option == "2" {
|
|
||||||
advancedMenu(&app)
|
|
||||||
} else if option == "0" {
|
|
||||||
return
|
|
||||||
} else {
|
|
||||||
fmt.Print("\nOops! An error occurred, please choose a valid option.\n\n")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
55
cmd/menu.go
Normal file
55
cmd/menu.go
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func mainMenu(app *Application) {
|
||||||
|
|
||||||
|
// Display main menu
|
||||||
|
fmt.Println("\n=====================================================")
|
||||||
|
fmt.Printf("| Welcome to the OpenWeatherMap-gRPC Client! v%s |\n", app.Version)
|
||||||
|
fmt.Println("=====================================================")
|
||||||
|
|
||||||
|
fmt.Printf("New in version %s:\n", app.Version)
|
||||||
|
fmt.Println(" - Advanced option: Get historical data back to 1972")
|
||||||
|
fmt.Println("New in version 0.1.0")
|
||||||
|
fmt.Println(" - Default option: Automatically determine location")
|
||||||
|
fmt.Println(" - Advanced option: Enter exact location for anywhere in the world")
|
||||||
|
fmt.Println(" - Advanced option: Change back and forth between imperial and metric units of measurement")
|
||||||
|
|
||||||
|
var option string
|
||||||
|
// Menu loop
|
||||||
|
for option != "0" {
|
||||||
|
fmt.Print("\nMain Menu\n---------\n\n")
|
||||||
|
fmt.Println("1. Today's forecast (use current location, default)")
|
||||||
|
fmt.Println("2. Advanced options (Change units, precise location, etc.)")
|
||||||
|
fmt.Print("0. Exit\n\n")
|
||||||
|
|
||||||
|
// Read user input
|
||||||
|
reader := bufio.NewReader(os.Stdin)
|
||||||
|
input, err := reader.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
option = strings.TrimSuffix(input, "\n")
|
||||||
|
|
||||||
|
// Check user input
|
||||||
|
if option == "1" || option == "" {
|
||||||
|
getLocation(app)
|
||||||
|
getCurrent(app)
|
||||||
|
printWeather(app)
|
||||||
|
} else if option == "2" {
|
||||||
|
advancedMenu(app)
|
||||||
|
} else if option == "0" {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
fmt.Print("\nOops! An error occurred, please choose a valid option.\n\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -3,7 +3,7 @@ package main
|
||||||
import "fmt"
|
import "fmt"
|
||||||
|
|
||||||
// Prints saved weather data to the terminal
|
// Prints saved weather data to the terminal
|
||||||
func printWeather(app *application) {
|
func printWeather(app *Application) {
|
||||||
|
|
||||||
var unitStr string
|
var unitStr string
|
||||||
if app.Config.Units == "imperial" {
|
if app.Config.Units == "imperial" {
|
||||||
|
|
|
@ -8,7 +8,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
// Get and store the current forecast
|
// Get and store the current forecast
|
||||||
func getCurrent(app *application) {
|
func getCurrent(app *Application) {
|
||||||
|
|
||||||
lat := "lat=" + app.Config.Latitude
|
lat := "lat=" + app.Config.Latitude
|
||||||
lon := "&lon=" + app.Config.Longitude
|
lon := "&lon=" + app.Config.Longitude
|
||||||
|
|
Loading…
Reference in a new issue