166 lines
3.4 KiB
Python
166 lines
3.4 KiB
Python
import bottle
|
|
|
|
from . import database, parcelsapp
|
|
|
|
PARCELSAPP_CLIENT: parcelsapp.ParcelsAppClient | None = None
|
|
|
|
TEMPLATE = """
|
|
<doctype HTML>
|
|
<html>
|
|
<head>
|
|
<style type="text/css">
|
|
|
|
.tracking-grid {
|
|
display: grid;
|
|
grid-template-columns: 1fr;
|
|
grid-gap: 10px;
|
|
}
|
|
|
|
.tracking-item {
|
|
display: grid;
|
|
grid-template-columns: 150px 1fr auto;
|
|
grid-gap: 1em;
|
|
padding: 0.5em 0.2em;
|
|
border-radius: 5px;
|
|
border: black solid 1px;
|
|
}
|
|
|
|
|
|
.tracking-name {
|
|
grid-column: 1;
|
|
}
|
|
|
|
.tracking-name-inner {
|
|
border: solid black 1px;
|
|
padding: 0.2em;
|
|
border-radius: 3px;
|
|
background-color: #FFF8;
|
|
}
|
|
|
|
a {
|
|
text-decoration: none;
|
|
}
|
|
|
|
.tracking-status {
|
|
grid-column: 2;
|
|
}
|
|
|
|
.tracking-update-date {
|
|
grid-column: 3;
|
|
}
|
|
|
|
.tracking-item.delivered {
|
|
background: #ABFBD6;
|
|
}
|
|
|
|
.tracking-item.transit {
|
|
background: #AEABFB;
|
|
}
|
|
|
|
h1 {
|
|
text-align: center;
|
|
}
|
|
main {
|
|
max-width: 800px;
|
|
margin:auto;
|
|
}
|
|
|
|
body {
|
|
font-family: sans;
|
|
}
|
|
|
|
a {
|
|
color:black;
|
|
}
|
|
|
|
.add-number-form {
|
|
display: grid;
|
|
grid-template: auto 1fr;
|
|
max-width: 300px;
|
|
grid-gap: 2px;
|
|
margin: auto;
|
|
}
|
|
|
|
.add-number-form .add {
|
|
grid-column: 1 / 3;
|
|
}
|
|
|
|
footer {
|
|
margin-top: 10em;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
|
|
<h1>Parcels</h1>
|
|
|
|
<main>
|
|
|
|
<div class="tracking-grid">
|
|
% for entry, tracking_data in tracking_results:
|
|
<a href="{{tracking_data.tracking_url}}">
|
|
<div class="tracking-item {{tracking_data.status}}">
|
|
<div class="tracking-name" title="{{entry.number}}"><span class="tracking-name-inner">{{ entry.name }}</span></div>
|
|
<div class="tracking-status">{{ tracking_data.latest_state().status }}</div>
|
|
<div class="tracking-update-date">{{ tracking_data.latest_state().date.date() }}</div>
|
|
</div>
|
|
</a>
|
|
% end
|
|
</div>
|
|
|
|
</main>
|
|
|
|
% if with_form:
|
|
<footer>
|
|
<form action="/add" method="POST" class="add-number-form">
|
|
<label for="name">Name</label>
|
|
<input type="text" name="name" id="name" required />
|
|
<label for="tracking-number">Number</label>
|
|
<input type="text" name="tracking-number" id="tracking-number" required />
|
|
<input class="add" type="submit" value="Add Tracking Number!" />
|
|
</form>
|
|
</footer>
|
|
% end
|
|
|
|
</body>
|
|
</html>
|
|
"""
|
|
|
|
|
|
def render_tracking(with_form: bool):
|
|
tracking_entries = database.get_tracking_numbers()
|
|
tracking_numbers = [e.number for e in tracking_entries]
|
|
|
|
tracking_results = PARCELSAPP_CLIENT.get_tracking_status(tracking_numbers)
|
|
|
|
tracking_results_by_id = {
|
|
result.tracking_number: result for result in tracking_results
|
|
}
|
|
|
|
tracking_results_with_name = [(e, tracking_results_by_id.get(e.number)) for e in tracking_entries]
|
|
tracking_results_with_name.sort(key=lambda x: x[1].latest_state().date, reverse=True)
|
|
|
|
return bottle.template(TEMPLATE, tracking_results=tracking_results_with_name, with_form=with_form)
|
|
|
|
@bottle.route('/')
|
|
def index():
|
|
return render_tracking(with_form=True)
|
|
|
|
@bottle.route('/noninteraction')
|
|
def inner():
|
|
return render_tracking(with_form=False)
|
|
|
|
|
|
@bottle.route('/add', method='POST')
|
|
def add():
|
|
number: str = bottle.request.forms['tracking-number']
|
|
name: str = bottle.request.forms['name']
|
|
database.add_tracking_number(database.TrackingNumberEntry(number, name))
|
|
return bottle.redirect('/')
|
|
|
|
|
|
def initialize_server(parcelsapp_client: parcelsapp.ParcelsAppClient):
|
|
global PARCELSAPP_CLIENT
|
|
PARCELSAPP_CLIENT = parcelsapp_client
|
|
bottle.run(host='localhost', port=8080, debug=True)
|