diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 8be315f..bc9c3d1 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,6 +1,6 @@ repos: - repo: https://github.com/psf/black - rev: 24.2.0 + rev: 24.3.0 hooks: - id: black language_version: python3 @@ -9,7 +9,7 @@ repos: hooks: - id: flake8 - repo: https://github.com/commitizen-tools/commitizen - rev: v3.18.3 + rev: v3.21.3 hooks: - id: commitizen stages: [commit-msg] diff --git a/dlunch/__init__.py b/dlunch/__init__.py index c2118c4..32d7747 100755 --- a/dlunch/__init__.py +++ b/dlunch/__init__.py @@ -102,7 +102,6 @@ def create_app(config: DictConfig) -> pn.Template: app.main.append(gi.results_divider) app.main.append(gi.res_col) app.modal.append(gi.error_message) - app.modal.append(gi.confirm_message) # Set components visibility based on no_more_order_button state # and reload menu diff --git a/dlunch/conf/db/postgresql.yaml b/dlunch/conf/db/postgresql.yaml index a2bd9f4..87bdd52 100644 --- a/dlunch/conf/db/postgresql.yaml +++ b/dlunch/conf/db/postgresql.yaml @@ -14,10 +14,12 @@ url: ${db.dialect}+${db.driver}://${db.username}:${db.password}@${db.host}:${db. # QUERIES # Orders orders_query: |- - SELECT o.user, o.lunch_time, m.item, o.note + SELECT o.user, u.lunch_time, m.item, o.note FROM {schema}.orders o LEFT JOIN {schema}.menu m - ON m.id = o.menu_item_id; + ON m.id = o.menu_item_id + LEFT JOIN {schema}.users u + ON u.id = o.user; # Stats stats_query: |- SELECT EXTRACT(YEAR FROM date)::varchar(4) AS "Year", diff --git a/dlunch/conf/db/sqlite.yaml b/dlunch/conf/db/sqlite.yaml index 606f192..e4a597f 100644 --- a/dlunch/conf/db/sqlite.yaml +++ b/dlunch/conf/db/sqlite.yaml @@ -9,10 +9,12 @@ url: ${db.dialect}:///${db.db_path} # QUERIES # Orders orders_query: |- - SELECT o.user, o.lunch_time, m.item, o.note + SELECT o.user, u.lunch_time, m.item, o.note FROM orders o LEFT JOIN menu m - ON m.id = o.menu_item_id; + ON m.id = o.menu_item_id + LEFT JOIN users u + ON u.id = o.user; # Stats stats_query: |- SELECT STRFTIME('%Y', date) AS "Year", diff --git a/dlunch/core.py b/dlunch/core.py index be69363..2db08ab 100644 --- a/dlunch/core.py +++ b/dlunch/core.py @@ -15,7 +15,7 @@ from pytesseract import pytesseract import random import re -from sqlalchemy import func, select, delete +from sqlalchemy import func, select, delete, update from sqlalchemy.sql.expression import true as sql_true from time import sleep @@ -166,7 +166,6 @@ def build_menu( ) -> pd.DataFrame: # Hide messages gi.error_message.visible = False - gi.confirm_message.visible = False # Build image path menu_filename = str( @@ -316,17 +315,24 @@ def reload_menu( value_if_missing=False, ) - # Set no more orders toggle button visibility and activation + # Set no more orders toggle button and the change order time button + # visibility and activation if auth.is_guest( user=pn_user(config), config=config, allow_override=False ): - # Deactivate the no_more_orders button for guest users + # Deactivate the no_more_orders_button for guest users gi.toggle_no_more_order_button.disabled = True gi.toggle_no_more_order_button.visible = False + # Deactivate the change_order_time_button for guest users + gi.change_order_time_takeaway_button.disabled = True + gi.change_order_time_takeaway_button.visible = False else: - # Activate the no_more_orders button for privileged users + # Activate the no_more_orders_button for privileged users gi.toggle_no_more_order_button.disabled = False gi.toggle_no_more_order_button.visible = True + # Show the change_order_time_button for privileged users + # It is disabled by the no more order button if necessary + gi.change_order_time_takeaway_button.visible = True # Guest graphic configuration if auth.is_guest(user=pn_user(config), config=config): @@ -586,7 +592,6 @@ def send_order( # Hide messages gi.error_message.visible = False - gi.confirm_message.visible = False # Create session session = models.create_session(config) @@ -677,11 +682,13 @@ def send_order( new_user = models.Users( id=username_key_press, guest=person.guest, + lunch_time=person.lunch_time, takeaway=person.takeaway, ) else: new_user = models.Users( id=username_key_press, + lunch_time=person.lunch_time, takeaway=person.takeaway, ) session.add(new_user) @@ -691,7 +698,6 @@ def send_order( # Order new_order = models.Orders( user=username_key_press, - lunch_time=person.lunch_time, menu_item_id=row.Index, note=getattr( row, config.panel.gui.note_column_name @@ -722,7 +728,7 @@ def send_order( f"DATABASE ERROR

ERROR:
{str(e)}" ) gi.error_message.visible = True - log.warning("database error") + log.error("database error") # Open modal window app.open_modal() else: @@ -744,7 +750,6 @@ def delete_order( event, config: DictConfig, app: pn.Template, - person: gui.Person, gi: gui.GraphicInterface, ) -> None: # Get username, updated on every keypress @@ -752,7 +757,6 @@ def delete_order( # Hide messages gi.error_message.visible = False - gi.confirm_message.visible = False # Create session session = models.create_session(config) @@ -797,21 +801,114 @@ def delete_order( return # Delete user - num_rows_deleted_users = session.execute( - delete(models.Users).where( - models.Users.id == username_key_press + try: + num_rows_deleted_users = session.execute( + delete(models.Users).where( + models.Users.id == username_key_press + ) ) - ) - # Delete also orders (hotfix for Debian) - num_rows_deleted_orders = session.execute( - delete(models.Orders).where( - models.Orders.user == username_key_press + # Delete also orders (hotfix for Debian) + num_rows_deleted_orders = session.execute( + delete(models.Orders).where( + models.Orders.user == username_key_press + ) ) + session.commit() + if (num_rows_deleted_users.rowcount > 0) or ( + num_rows_deleted_orders.rowcount > 0 + ): + # Update dataframe widget + reload_menu( + None, + config, + gi, + ) + + pn.state.notifications.success( + "Order canceled", + duration=config.panel.notifications.duration, + ) + log.info(f"{username_key_press}'s order canceled") + else: + pn.state.notifications.warning( + f'No order for user named
"{username_key_press}"', + duration=config.panel.notifications.duration, + ) + log.info(f"no order for user named {username_key_press}") + except Exception as e: + # Any exception here is a database fault + pn.state.notifications.error( + "Database error", + duration=config.panel.notifications.duration, + ) + gi.error_message.object = ( + f"DATABASE ERROR

ERROR:
{str(e)}" + ) + gi.error_message.visible = True + log.error("database error") + # Open modal window + app.open_modal() + else: + pn.state.notifications.warning( + "Please insert user name", + duration=config.panel.notifications.duration, + ) + log.warning("missing username") + + +def change_order_time_takeaway( + event, + config: DictConfig, + person: gui.Person, + gi: gui.GraphicInterface, +) -> None: + # Get username, updated on every keypress + username_key_press = gi.person_widget._widgets["username"].value_input + + # Create session + session = models.create_session(config) + + with session: + # Check if the "no more order" toggle button is pressed + if models.get_flag(config=config, id="no_more_orders"): + pn.state.notifications.error( + "It is not possible to update orders (time)", + duration=config.panel.notifications.duration, ) + + # Reload the menu + reload_menu( + None, + config, + gi, + ) + + return + + if username_key_press: + # Build and execute the update statement + update_statement = ( + update(models.Users) + .where(models.Users.id == username_key_press) + .values(lunch_time=person.lunch_time, takeaway=person.takeaway) + .returning(models.Users) + ) + + updated_user = session.scalars(update_statement).one_or_none() + session.commit() - if (num_rows_deleted_users.rowcount > 0) or ( - num_rows_deleted_orders.rowcount > 0 - ): + + if updated_user: + # Find updated values + updated_time = updated_user.lunch_time + updated_takeaway = ( + (" " + config.panel.gui.takeaway_id) + if updated_user.takeaway + else "" + ) + updated_items_names = [ + order.menu_item.item for order in updated_user.orders + ] # Update dataframe widget reload_menu( None, @@ -820,10 +917,10 @@ def delete_order( ) pn.state.notifications.success( - "Order canceled", + f"{username_key_press}'s
lunch time changed to
{updated_time}{updated_takeaway}
({', '.join(updated_items_names)})", duration=config.panel.notifications.duration, ) - log.info(f"{username_key_press}'s order canceled") + log.info(f"{username_key_press}'s order updated") else: pn.state.notifications.warning( f'No order for user named
"{username_key_press}"', @@ -952,12 +1049,8 @@ def clean_up_table( def download_dataframe( config: DictConfig, - app: pn.Template, gi: gui.GraphicInterface, ) -> None: - # Hide messages - gi.error_message.visible = False - gi.confirm_message.visible = False # Build a dict of dataframes, one for each lunch time (the key contains # a lunch time) diff --git a/dlunch/gui.py b/dlunch/gui.py index cb111fd..4be808f 100644 --- a/dlunch/gui.py +++ b/dlunch/gui.py @@ -383,7 +383,17 @@ def reload_on_guest_override_callback( button_style="outline", button_type="warning", height=generic_button_height, - icon="alarm", + icon="hand-stop", + icon_size="2em", + sizing_mode="stretch_width", + ) + # Create change time + self.change_order_time_takeaway_button = pnw.Button( + name="Change Time/Takeaway", + button_type="primary", + button_style="outline", + height=generic_button_height, + icon="clock-edit", icon_size="2em", sizing_mode="stretch_width", ) @@ -432,6 +442,7 @@ def reload_on_guest_override_callback( *[ self.send_order_button, self.toggle_no_more_order_button, + self.change_order_time_takeaway_button, self.delete_order_button, ], flex_wrap="nowrap", @@ -454,9 +465,10 @@ def reload_on_no_more_order_callback( # Show "no more order" text self.no_more_order_alert.visible = toggle - # Deactivate send order and delete order buttons + # Deactivate send, delete and change order buttons self.send_order_button.disabled = toggle self.delete_order_button.disabled = toggle + self.change_order_time_takeaway_button.disabled = toggle # Simply reload the menu when the toggle button value changes if reload: @@ -493,6 +505,14 @@ def reload_on_no_more_order_callback( e, config, app, + self, + ) + ) + # Change order time button callback + self.change_order_time_takeaway_button.on_click( + lambda e: core.change_order_time_takeaway( + e, + config, person, self, ) @@ -505,12 +525,6 @@ def reload_on_no_more_order_callback( sizing_mode="stretch_width", ) self.error_message.visible = False - # Confirm message - self.confirm_message = pn.pane.HTML( - styles={"color": "green", "font-weight": "bold"}, - sizing_mode="stretch_width", - ) - self.confirm_message.visible = False # SIDEBAR ------------------------------------------------------------- # TEXTS @@ -610,7 +624,7 @@ def reload_on_no_more_order_callback( ) # Download button and callback self.download_button = pn.widgets.FileDownload( - callback=lambda: core.download_dataframe(config, app, self), + callback=lambda: core.download_dataframe(config, self), filename=config.panel.file_name + ".xlsx", sizing_mode="stretch_width", icon="download", diff --git a/dlunch/models.py b/dlunch/models.py index 9be9be3..8b61979 100755 --- a/dlunch/models.py +++ b/dlunch/models.py @@ -212,7 +212,7 @@ class Orders(Data): index=True, nullable=False, ) - lunch_time = Column(String(7), index=True, nullable=False) + user_record = relationship("Users", back_populates="orders", uselist=False) menu_item_id = Column( Integer, ForeignKey("menu.id", ondelete="CASCADE"), @@ -265,9 +265,16 @@ class Users(Data): default="NotAGuest", server_default="NotAGuest", ) + lunch_time = Column(String(7), index=True, nullable=False) takeaway = Column( Boolean, nullable=False, default=False, server_default=sql_false() ) + orders = relationship( + "Orders", + back_populates="user_record", + cascade="all, delete-orphan", + passive_deletes=True, + ) @classmethod def clear(self, config: DictConfig) -> int: diff --git a/requirements/environment.yml b/requirements/environment.yml index eaaf998..e9b13bc 100755 --- a/requirements/environment.yml +++ b/requirements/environment.yml @@ -5,7 +5,7 @@ channels: - pyviz dependencies: - python=3.11.7 - - setuptools=69.1.1 + - setuptools=69.2.0 - click=8.1.7 - cryptography=42.0.5 - ipykernel=6.29.3 @@ -15,10 +15,10 @@ dependencies: - passlib=1.7.4 - tenacity=8.2.3 - tqdm=4.66.2 - - panel=1.3.8 - - sqlalchemy=2.0.28 + - panel=1.4.0 + - sqlalchemy=2.0.29 - psycopg=3.1.18 - sqlite=3.41.2 - hydra-core=1.3.2 - - google-cloud-storage=2.14.0 + - google-cloud-storage=2.16.0 - pytesseract=0.3.10 diff --git a/requirements/requirements.txt b/requirements/requirements.txt index a2e97ef..6181cec 100644 --- a/requirements/requirements.txt +++ b/requirements/requirements.txt @@ -1,4 +1,4 @@ -setuptools==69.1.1 +setuptools==69.2.0 click==8.1.7 cryptography==42.0.5 ipykernel==6.29.3 @@ -8,9 +8,9 @@ pandas==2.2.1 passlib==1.7.4 tenacity==8.2.3 tqdm==4.66.2 -panel==1.3.8 -sqlalchemy==2.0.28 +panel==1.4.0 +sqlalchemy==2.0.29 psycopg==3.1.18 hydra-core==1.3.2 -google-cloud-storage==2.14.0 +google-cloud-storage==2.16.0 pytesseract==0.3.10