haldis/app/models/order.py

116 lines
3.8 KiB
Python
Raw Permalink Normal View History

2019-09-10 01:06:11 +02:00
"Script for everything Order related in the database"
2019-09-08 00:41:50 +02:00
import typing
from datetime import datetime
2021-06-21 02:05:11 +02:00
from collections import defaultdict
2022-04-19 21:29:53 +02:00
import secrets
import string
from utils import first
from hlds.definitions import location_definitions
from .database import db
from .user import User
class Order(db.Model):
2019-09-10 01:06:11 +02:00
"Class used for configuring the Order model in the database"
id = db.Column(db.Integer, primary_key=True)
courier_id = db.Column(db.Integer, nullable=True)
2020-01-26 02:39:58 +01:00
location_id = db.Column(db.String(64))
location_name = db.Column(db.String(128))
starttime = db.Column(db.DateTime)
stoptime = db.Column(db.DateTime)
public = db.Column(db.Boolean, default=True)
2022-04-19 21:29:53 +02:00
slug = db.Column(db.String(7))
2020-01-26 02:39:58 +01:00
2019-09-05 03:33:29 +02:00
items = db.relationship("OrderItem", backref="order", lazy="dynamic")
2022-04-19 21:29:53 +02:00
def __init__(self):
super().__init__()
alphabet = string.ascii_letters + string.digits
self.slug = ''.join(secrets.choice(alphabet) for i in range(7))
def __getattr__(self, name):
if name == "location":
2020-07-17 11:40:15 +02:00
return first(
filter(lambda l: l.id == self.location_id, location_definitions)
)
raise AttributeError()
2019-09-08 00:41:50 +02:00
def __repr__(self) -> str:
2019-09-10 01:06:11 +02:00
# pylint: disable=R1705
if self.location:
2019-09-05 03:33:29 +02:00
return "Order %d @ %s" % (self.id, self.location.name or "None")
else:
2019-09-05 03:33:29 +02:00
return "Order %d" % (self.id)
def update_from_hlds(self) -> None:
"""
Update the location name from the HLDS definition.
User should commit after running this to make the change persistent.
"""
2020-07-17 11:40:15 +02:00
assert (
self.location_id
), "location_id must be configured before updating from HLDS"
self.location_name = self.location.name
2020-08-14 04:57:02 +02:00
def for_user(self, anon=None, user=None) -> typing.List:
return list(
filter(
(lambda i: i.user == user)
if user is not None
else (lambda i: i.user_name == anon),
self.items
)
)
def group_by_user(self) -> typing.List[typing.Tuple[str, typing.List]]:
2019-09-10 01:06:11 +02:00
"Group items of an Order by user"
2020-08-14 04:57:02 +02:00
group: typing.Dict[str, typing.List] = dict()
for item in self.items:
2020-08-14 04:57:02 +02:00
if item.for_name not in group:
group[item.for_name] = []
group[item.for_name].append(item)
for _user_name, order_items in group.items():
order_items.sort(key=lambda order_item: order_item.comment or "")
2020-08-19 13:02:14 +02:00
return list(sorted(group.items(), key=lambda t: (t[0] or "", t[1] or "")))
2020-08-14 04:57:02 +02:00
2021-06-21 02:05:11 +02:00
def group_by_dish(self) \
-> typing.List[typing.Tuple[str, int, typing.List[typing.Tuple[str, typing.List]]]]:
"Group items of an Order by dish"
2021-06-21 02:05:11 +02:00
group: typing.Dict[str, typing.Dict[str, typing.List]] = \
defaultdict(lambda: defaultdict(list))
2020-08-14 04:57:02 +02:00
for item in self.items:
2021-06-21 02:05:11 +02:00
group[item.dish_name][item.comment].append(item)
return sorted(
(
dish_name,
# Amount of items of this dish
sum(map(len, comment_group.values())),
sorted(
(comment, sorted(items, key=lambda x: (x.for_name or "")))
2021-06-21 02:05:11 +02:00
for comment, items in comment_group.items()
)
)
for dish_name, comment_group in group.items()
)
def is_closed(self) -> bool:
2020-01-27 03:52:29 +01:00
return self.stoptime and datetime.now() > self.stoptime
2019-09-08 00:41:50 +02:00
def can_close(self, user_id: int) -> bool:
2019-09-10 01:06:11 +02:00
"Check if a user can close the Order"
if self.stoptime and self.stoptime < datetime.now():
return False
user = None
if user_id:
user = User.query.filter_by(id=user_id).first()
if self.courier_id == user_id or (user and user.is_admin()):
return True
return False