aboutsummaryrefslogtreecommitdiffstats
path: root/main.py
blob: 88bb81fcfb161d2a97fbeb1489773cb4ec5d4e00 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
import argparse
import sys
from io import StringIO
from textwrap import dedent
from typing import Tuple

import pandas as pd
import requests
from geopy.distance import geodesic
from geopy.geocoders import Nominatim
from geopy.location import Location
from tabulate import tabulate

ENDPOINT = "https://www.fuel-finder.service.gov.uk/internal/v1.0.2/csv/get-latest-fuel-prices-csv"

SORT_KV = {
    "e10": "e10_price",
    "e5": "e5_price",
    "b7s": "diesel_price",
    "distance": "distance",
}

HEADERS = {
    "User-Agent": "Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10_6_4; en-US) AppleWebKit/533.45 (KHTML, like Gecko) Chrome/48.0.2094.221 Safari/602"
}


def parse_args() -> argparse.Namespace:
    parser = argparse.ArgumentParser()
    parser.add_argument("-a", "--address", type=str, required=True)
    parser.add_argument("-r", "--radius", type=int, default=5)
    parser.add_argument("-s", "--sort", type=str, default="e10")
    return parser.parse_args()


def get_location(address: str) -> tuple[float, float]:
    geolocator = Nominatim(user_agent="FuelNearMe")
    result = geolocator.geocode(address)
    if not isinstance(result, Location):
        print("[*] Failed to get location. Please check if the address is valid.")
        sys.exit(1)
    return (result.latitude, result.longitude)


def get_latest_data() -> Tuple[pd.DataFrame, str]:
    try:
        response = requests.get(ENDPOINT, headers=HEADERS, timeout=10)
        response.raise_for_status()
    except Exception as e:
        raise e
    return pd.read_csv(StringIO(response.text)), response.headers.get("Last-Modified")


def process_data(dframe):
    price_cols = [c for c in dframe.columns if "fuel_price" in c]
    dframe[price_cols] = dframe[price_cols].fillna(0.0)
    return dframe.fillna("N/A")


def filter_df(dframe, arguments, loc):
    near_stations = []
    for station, latitude, longitude, e5_price, e10_price, diesel_price in zip(
        dframe["forecourts.trading_name"],
        dframe["forecourts.location.latitude"],
        dframe["forecourts.location.longitude"],
        dframe["forecourts.fuel_price.E5"],
        dframe["forecourts.fuel_price.E10"],
        dframe["forecourts.fuel_price.B7S"],
    ):
        distance_from_current_location = geodesic((latitude, longitude), loc).miles
        if distance_from_current_location < arguments.radius:
            station_dict = {
                "station_name": station,
                "distance": round(distance_from_current_location, 1),
                "e5_price": round(e5_price / 100, 2),
                "e10_price": round(e10_price / 100, 2),
                "diesel_price": round(diesel_price / 100, 2),
            }
            na_dict = {
                k: (v if v != 0.00 else "N/A") for (k, v) in station_dict.items()
            }
            near_stations.append(na_dict)
    return near_stations


def sort_stations(stations: list[dict], sort: str) -> list[dict]:
    sort_key = SORT_KV.get(sort)
    return sorted(stations, key=lambda d: d[sort_key] if d[sort_key] != "N/A" else 999)


def output_stations(stations):
    print(
        tabulate(
            stations,
            headers={
                "station_name": "Station Name",
                "distance": "Distance (miles)",
                "e5_price": "E5 (£/L)",
                "e10_price": "E10 (£/L)",
                "diesel_price": "B7S (£/L)",
            },
            floatfmt=".2f",
        )
    )


def main():
    args = parse_args()
    location = get_location(args.address)
    df, last_modified = get_latest_data()

    print(f"Last modified: {last_modified}")

    df_processed = process_data(df)

    df_filtered = filter_df(df_processed, args, location)

    sorted_stations_list = sort_stations(df_filtered, args.sort)

    output_stations(sorted_stations_list)


if __name__ == "__main__":
    main()
git.ajschof.me — hosted by ajschofield — powered by cgit