Skip to content

models

Module with database tables definitions.

Helper classes and utility functions for data management are defined here.

Classes:

Name Description
CommonTable

Abstract table with common methods.

Credentials

Table with users credentials, used only if basic authentication is active.

DatabaseConnector

Class for handling database connections and operations.

Encrypted

Allows storing and retrieving password hashes using PasswordHash.

Flags

Table with global flags used by Data-Lunch.

Menu

Table with menu items.

Orders

Table with items that belongs to an order.

Password

Allows storing and retrieving password hashes using PasswordHash.

PrivilegedUsers

Table with user that have privileges (normal users and admin).

Stats

Table with number of users that ate a lunch, grouped by guest type.

Users

Table with users that placed an order.

Functions:

Name Description
set_sqlite_pragma

Force foreign key constraints for sqlite connections.

Attributes:

Name Type Description
Data DeclarativeMeta

SQLAlchemy declarative base.

SCHEMA str

Schema name from environment (may be overridden by configuration files).

log Logger

Module logger.

metadata_obj MetaData

Database metadata (SQLAlchemy).

Data module-attribute

Data: DeclarativeMeta = declarative_base(
    metadata=metadata_obj
)

SQLAlchemy declarative base.

SCHEMA module-attribute

SCHEMA: str = get('DATA_LUNCH_DB_SCHEMA', None)

Schema name from environment (may be overridden by configuration files).

log module-attribute

log: Logger = getLogger(__name__)

Module logger.

metadata_obj module-attribute

metadata_obj: MetaData = MetaData(schema=SCHEMA)

Database metadata (SQLAlchemy).

CommonTable

Bases: Data

Abstract table with common methods.

Methods:

Name Description
clear

Clear table and return deleted rows.

read_as_df

Read table as pandas DataFrame.

write_from_df

Write table from pandas DataFrame.

Attributes:

Name Type Description
__abstract__

Abstract table flag.

Source code in dlunch/models.py
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
class CommonTable(Data):
    """Abstract table with common methods."""

    __abstract__ = True
    """Abstract table flag."""

    @classmethod
    def clear(self, config: DictConfig) -> int:
        """Clear table and return deleted rows.

        Args:
            config (DictConfig): Hydra configuration dictionary.

        Returns:
            int: deleted rows.
        """
        session = DatabaseConnector(config=config).create_session()
        with session:
            # Clean table
            num_rows_deleted = session.execute(delete(self))
            session.commit()
            log.info(
                f"deleted {num_rows_deleted.rowcount} rows from table '{self.__tablename__}'"
            )
        return num_rows_deleted.rowcount

    @classmethod
    def read_as_df(self, config: DictConfig, **kwargs) -> pd.DataFrame:
        """Read table as pandas DataFrame.

        Args:
            config (DictConfig): Hydra configuration dictionary.

        Returns:
            pd.DataFrame: dataframe with table content.
        """
        df = pd.read_sql_table(
            table_name=self.__tablename__,
            con=DatabaseConnector(config=config).create_engine(),
            schema=config.db.get("schema", SCHEMA),
            **kwargs,
        )
        return df

    @classmethod
    def write_from_df(
        self, config: DictConfig, df: pd.DataFrame, index: bool = True
    ) -> int:
        """Write table from pandas DataFrame.

        If a record already exists in the table, it will be updated.

        Args:
            config (DictConfig): Hydra configuration dictionary.
            df (pd.DataFrame): dataframe with table content.
            index (bool): write index as a column. Use False to ignore index. Defaults to True.

        Returns:
            int: number of rows written.
        """

        session = DatabaseConnector(config=config).create_session()
        # Convert the dataframe to a dictionary of records
        drop_index = not index
        records_dict = df.reset_index(drop=drop_index).to_dict(
            orient="records"
        )

        with session:
            # Add rows
            for record in records_dict:
                # Convert the tuple to dict and expand to avoid errors
                new_record = self(**record)
                DatabaseConnector.session_add_with_upsert(
                    session=session,
                    constraint=f"{self.__tablename__}_pkey",
                    new_record=new_record,
                )

            # Commit only at the end
            session.commit()

        return len(df)

__abstract__ class-attribute instance-attribute

__abstract__ = True

Abstract table flag.

clear classmethod

clear(config: DictConfig) -> int

Clear table and return deleted rows.

Parameters:

Name Type Description Default
config DictConfig

Hydra configuration dictionary.

required

Returns:

Type Description
int

deleted rows.

Source code in dlunch/models.py
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
@classmethod
def clear(self, config: DictConfig) -> int:
    """Clear table and return deleted rows.

    Args:
        config (DictConfig): Hydra configuration dictionary.

    Returns:
        int: deleted rows.
    """
    session = DatabaseConnector(config=config).create_session()
    with session:
        # Clean table
        num_rows_deleted = session.execute(delete(self))
        session.commit()
        log.info(
            f"deleted {num_rows_deleted.rowcount} rows from table '{self.__tablename__}'"
        )
    return num_rows_deleted.rowcount

read_as_df classmethod

read_as_df(config: DictConfig, **kwargs) -> DataFrame

Read table as pandas DataFrame.

Parameters:

Name Type Description Default
config DictConfig

Hydra configuration dictionary.

required

Returns:

Type Description
DataFrame

dataframe with table content.

Source code in dlunch/models.py
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
@classmethod
def read_as_df(self, config: DictConfig, **kwargs) -> pd.DataFrame:
    """Read table as pandas DataFrame.

    Args:
        config (DictConfig): Hydra configuration dictionary.

    Returns:
        pd.DataFrame: dataframe with table content.
    """
    df = pd.read_sql_table(
        table_name=self.__tablename__,
        con=DatabaseConnector(config=config).create_engine(),
        schema=config.db.get("schema", SCHEMA),
        **kwargs,
    )
    return df

write_from_df classmethod

write_from_df(
    config: DictConfig, df: DataFrame, index: bool = True
) -> int

Write table from pandas DataFrame.

If a record already exists in the table, it will be updated.

Parameters:

Name Type Description Default
config DictConfig

Hydra configuration dictionary.

required
df DataFrame

dataframe with table content.

required
index bool

write index as a column. Use False to ignore index. Defaults to True.

True

Returns:

Type Description
int

number of rows written.

Source code in dlunch/models.py
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
@classmethod
def write_from_df(
    self, config: DictConfig, df: pd.DataFrame, index: bool = True
) -> int:
    """Write table from pandas DataFrame.

    If a record already exists in the table, it will be updated.

    Args:
        config (DictConfig): Hydra configuration dictionary.
        df (pd.DataFrame): dataframe with table content.
        index (bool): write index as a column. Use False to ignore index. Defaults to True.

    Returns:
        int: number of rows written.
    """

    session = DatabaseConnector(config=config).create_session()
    # Convert the dataframe to a dictionary of records
    drop_index = not index
    records_dict = df.reset_index(drop=drop_index).to_dict(
        orient="records"
    )

    with session:
        # Add rows
        for record in records_dict:
            # Convert the tuple to dict and expand to avoid errors
            new_record = self(**record)
            DatabaseConnector.session_add_with_upsert(
                session=session,
                constraint=f"{self.__tablename__}_pkey",
                new_record=new_record,
            )

        # Commit only at the end
        session.commit()

    return len(df)

Credentials

Bases: CommonTable

Table with users credentials, used only if basic authentication is active.

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__tablename__

Name of the table.

password_encrypted

Encryped password.

password_hash

Hashed password.

user

Username.

Source code in dlunch/models.py
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
class Credentials(CommonTable):
    """Table with users credentials, used only if basic authentication is active."""

    __tablename__ = "credentials"
    """Name of the table."""
    user = Column(
        String(100),
        primary_key=True,
        sqlite_on_conflict_primary_key="REPLACE",
    )
    """Username."""
    password_hash = Column(Password(150), unique=False, nullable=False)
    """Hashed password."""
    password_encrypted = Column(
        Encrypted(150),
        unique=False,
        nullable=True,
        default=None,
        server_default=None,
    )
    """Encryped password.

    Used only if basic authentication and guest users are both active."""

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<CREDENTIAL:{self.user}>"

    @validates("password_hash")
    def _validate_password(
        self, key: str, password: str
    ) -> auth.PasswordHash | None:
        """Function that validate password input.

        It converts string to auth.PasswordHash if necessary.

        Args:
            key (str): validated attribute name.
            password (str): hashed password to be validated.

        Returns:
            auth.PasswordHash | None: validated hashed password.
        """
        return getattr(type(self), key).type.validator(password)

    @validates("password_encrypted")
    def _validate_encrypted(
        self, key: str, password: str
    ) -> auth.PasswordEncrypt | None:
        """Function that validate encrypted input.

        It converts string to auth.PasswordEncrypt if necessary.

        Args:
            key (str): validated attribute name.
            password (str): encrypted password to be validated.

        Returns:
            auth.PasswordEncrypt | None: validated encrypted password.
        """
        return getattr(type(self), key).type.validator(password)

__tablename__ class-attribute instance-attribute

__tablename__ = 'credentials'

Name of the table.

password_encrypted class-attribute instance-attribute

password_encrypted = Column(
    Encrypted(150),
    unique=False,
    nullable=True,
    default=None,
    server_default=None,
)

Encryped password.

Used only if basic authentication and guest users are both active.

password_hash class-attribute instance-attribute

password_hash = Column(
    Password(150), unique=False, nullable=False
)

Hashed password.

user class-attribute instance-attribute

user = Column(
    String(100),
    primary_key=True,
    sqlite_on_conflict_primary_key="REPLACE",
)

Username.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
596
597
598
599
600
601
602
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<CREDENTIAL:{self.user}>"

DatabaseConnector

Class for handling database connections and operations.

Methods:

Name Description
__init__
create_database

Function to create the database through SQLAlchemy models.

create_engine

Factory function for SQLAlchemy engine.

create_session

Factory function for database session.

get_db_dialect

Return database type (postgresql, sqlite, etc.) based on the database object passed as input.

get_flag

Get the value of a flag.

read_sql_query

Read a SQL query as pandas DataFrame.

session_add_with_upsert

Use an upsert statement for postgresql to add a new record to a table,

set_flag

Set a key,value pair inside flag table.

Attributes:

Name Type Description
config DictConfig

Hydra configuration dictionary.

Source code in dlunch/models.py
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
class DatabaseConnector:
    """Class for handling database connections and operations."""

    def __init__(self, config: DictConfig):
        self.config: DictConfig = config
        """Hydra configuration dictionary."""

    @staticmethod
    def get_db_dialect(
        db_obj: Session | ConnectionSqlite | ConnectionPostgresql,
    ) -> str:
        """Return database type (postgresql, sqlite, etc.) based on the database object passed as input.
        If a session is passed, the database type is set based on an internal map (see `models._DBTYPE_MAP`).

        Args:
            db_obj (Session | ConnectionSqlite | ConnectionPostgresql): session or connection object.

        Raises:
            TypeError: db_obj shall be a session or a connection object.

        Returns:
            str: database dialect.
        """
        if isinstance(db_obj, Session):
            dialect = db_obj.bind.dialect.name
        elif isinstance(db_obj, ConnectionSqlite) or isinstance(
            db_obj, ConnectionPostgresql
        ):
            module = db_obj.__class__.__module__.split(".", 1)[0]
            dialect = _MODULE_TO_DIALECT_MAP[module]
        else:
            raise TypeError("db_obj should be a session or connection object")

        return dialect

    @staticmethod
    def session_add_with_upsert(
        session: Session, constraint: str, new_record: Stats | Flags
    ) -> None:
        """Use an upsert statement for postgresql to add a new record to a table,
        a simple session add otherwise.

        Args:
            session (Session): SQLAlchemy session object.
            constraint (str): constraint used for upsert (usually the primary key)
            new_record (Stats | Flags): table resord (valid tables are `stats` or `flags`)
        """
        # Use an upsert for postgresql (for sqlite an 'on conflict replace' is set
        # on table, so session.add is fine)
        if DatabaseConnector.get_db_dialect(session) == "postgresql":
            insert_statement = postgresql_upsert(new_record.__table__).values(
                {
                    column.name: getattr(new_record, column.name)
                    for column in new_record.__table__.c
                    if getattr(new_record, column.name, None) is not None
                }
            )
            upsert_statement = insert_statement.on_conflict_do_update(
                constraint=constraint,
                set_={
                    column.name: getattr(
                        insert_statement.excluded, column.name
                    )
                    for column in insert_statement.excluded
                },
            )
            session.execute(upsert_statement)
        else:
            session.add(new_record)

    @staticmethod
    def read_sql_query(session: Session, query: str) -> pd.DataFrame:
        """Read a SQL query as pandas DataFrame.

        Args:
            session (Session): SQLAlchemy session object.
            query (str): SQL query.

        Returns:
            pd.DataFrame: dataframe with query result.
        """
        results = session.execute(text(query))
        columns = results.keys()
        # Pass columns to build dataframe with correct columns names even if
        # the query returns no rows
        df = pd.DataFrame(results.all(), columns=columns)

        return df

    def create_engine(self) -> Engine:
        """Factory function for SQLAlchemy engine.

        Returns:
            Engine: SQLAlchemy engine.
        """
        engine = hydra.utils.instantiate(self.config.db.engine)

        # Change schema with change_execution_options
        # If schema exist in config.db it will override the schema selected through
        # the environment variable
        if "schema" in self.config.db:
            engine.update_execution_options(
                schema_translate_map={SCHEMA: self.config.db.schema}
            )

        return engine

    def create_session(self) -> Session:
        """Factory function for database session.

        Returns:
            Session: SQLAlchemy session.
        """
        engine = self.create_engine()
        session = Session(engine)

        return session

    def create_database(self, add_basic_auth_users=False) -> None:
        """Function to create the database through SQLAlchemy models.

        Args:
            add_basic_auth_users (bool, optional): set to true to add admin and guest users.
                These users are of interest only if 'basic authentication' is selected.
                Defaults to False.
        """
        # Create directory if missing
        log.debug("create 'shared_data' folder")
        pathlib.Path(self.config.db.shared_data_folder).mkdir(exist_ok=True)

        # In case the database is not ready use a retry mechanism
        @tenacity.retry(
            retry=tenacity.retry_if_exception_type(OperationalError),
            wait=tenacity.wait_fixed(self.config.db.create_retries.wait),
            stop=(
                tenacity.stop_after_delay(
                    self.config.db.create_retries.stop.delay
                )
                | tenacity.stop_after_attempt(
                    self.config.db.create_retries.stop.attempts
                )
            ),
        )
        def _create_database_with_retries(config: DictConfig) -> None:
            engine = self.create_engine()
            Data.metadata.create_all(engine)

        # Create tables
        log.debug(
            f"attempt database creation: {self.config.db.attempt_creation}"
        )
        if self.config.db.attempt_creation:
            _create_database_with_retries(self.config)

            # Retries stats
            log.debug(
                f"create database attempts: {_create_database_with_retries.retry.statistics}"
            )

        # If requested add users for basic auth (admin and guest)
        if add_basic_auth_users:
            log.debug("add basic auth standard users (if missing)")
            # If no user exist create the default admin
            session = self.create_session()

            with session:
                # Check if admin exists
                if session.get(Credentials, "admin") is None:
                    # Add authorization and credentials for admin
                    auth_user = auth.AuthUser(config=self.config, name="admin")
                    auth_user.add_privileged_user(is_admin=True)
                    auth_user.add_user_hashed_password(password="admin")
                    log.warning(
                        "admin user created, remember to change the default password"
                    )
                # Check if guest exists
                if (
                    session.get(Credentials, "guest") is None
                ) and self.config.basic_auth.guest_user:
                    # Add only credentials for guest (guest users are not included
                    # in privileged_users table)
                    auth_user = auth.AuthUser(config=self.config, name="guest")
                    auth_user.add_user_hashed_password(password="guest")
                    log.warning(
                        "guest user created, remember to change the default password"
                    )

    def set_flag(self, id: str, value: bool) -> None:
        """Set a key,value pair inside `flag` table.

        Args:
            id (str): flag id (name).
            value (bool): flag value.
        """

        session = self.create_session()

        with session:
            # Write the selected flag (it will be overwritten if exists)
            new_flag = Flags(id=id, value=value)

            # Use an upsert for postgresql, a simple session add otherwise
            DatabaseConnector.session_add_with_upsert(
                session=session, constraint="flags_pkey", new_record=new_flag
            )

            session.commit()

    def get_flag(
        self, id: str, value_if_missing: bool | None = None
    ) -> bool | None:
        """Get the value of a flag.
        Optionally select the values to return if the flag is missing (default to None).

        Args:
            id (str): flag id (name).
            value_if_missing (bool | None, optional): value to return if the flag does not exist. Defaults to None.

        Returns:
            bool | None: flag value.
        """

        session = self.create_session()
        flag = session.get(Flags, id)
        if flag is None:
            value = value_if_missing
        else:
            value = flag.value
        return value

config instance-attribute

config: DictConfig = config

Hydra configuration dictionary.

__init__

__init__(config: DictConfig)
Source code in dlunch/models.py
643
644
645
def __init__(self, config: DictConfig):
    self.config: DictConfig = config
    """Hydra configuration dictionary."""

create_database

create_database(add_basic_auth_users=False) -> None

Function to create the database through SQLAlchemy models.

Parameters:

Name Type Description Default
add_basic_auth_users bool

set to true to add admin and guest users. These users are of interest only if 'basic authentication' is selected. Defaults to False.

False
Source code in dlunch/models.py
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
def create_database(self, add_basic_auth_users=False) -> None:
    """Function to create the database through SQLAlchemy models.

    Args:
        add_basic_auth_users (bool, optional): set to true to add admin and guest users.
            These users are of interest only if 'basic authentication' is selected.
            Defaults to False.
    """
    # Create directory if missing
    log.debug("create 'shared_data' folder")
    pathlib.Path(self.config.db.shared_data_folder).mkdir(exist_ok=True)

    # In case the database is not ready use a retry mechanism
    @tenacity.retry(
        retry=tenacity.retry_if_exception_type(OperationalError),
        wait=tenacity.wait_fixed(self.config.db.create_retries.wait),
        stop=(
            tenacity.stop_after_delay(
                self.config.db.create_retries.stop.delay
            )
            | tenacity.stop_after_attempt(
                self.config.db.create_retries.stop.attempts
            )
        ),
    )
    def _create_database_with_retries(config: DictConfig) -> None:
        engine = self.create_engine()
        Data.metadata.create_all(engine)

    # Create tables
    log.debug(
        f"attempt database creation: {self.config.db.attempt_creation}"
    )
    if self.config.db.attempt_creation:
        _create_database_with_retries(self.config)

        # Retries stats
        log.debug(
            f"create database attempts: {_create_database_with_retries.retry.statistics}"
        )

    # If requested add users for basic auth (admin and guest)
    if add_basic_auth_users:
        log.debug("add basic auth standard users (if missing)")
        # If no user exist create the default admin
        session = self.create_session()

        with session:
            # Check if admin exists
            if session.get(Credentials, "admin") is None:
                # Add authorization and credentials for admin
                auth_user = auth.AuthUser(config=self.config, name="admin")
                auth_user.add_privileged_user(is_admin=True)
                auth_user.add_user_hashed_password(password="admin")
                log.warning(
                    "admin user created, remember to change the default password"
                )
            # Check if guest exists
            if (
                session.get(Credentials, "guest") is None
            ) and self.config.basic_auth.guest_user:
                # Add only credentials for guest (guest users are not included
                # in privileged_users table)
                auth_user = auth.AuthUser(config=self.config, name="guest")
                auth_user.add_user_hashed_password(password="guest")
                log.warning(
                    "guest user created, remember to change the default password"
                )

create_engine

create_engine() -> Engine

Factory function for SQLAlchemy engine.

Returns:

Type Description
Engine

SQLAlchemy engine.

Source code in dlunch/models.py
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
def create_engine(self) -> Engine:
    """Factory function for SQLAlchemy engine.

    Returns:
        Engine: SQLAlchemy engine.
    """
    engine = hydra.utils.instantiate(self.config.db.engine)

    # Change schema with change_execution_options
    # If schema exist in config.db it will override the schema selected through
    # the environment variable
    if "schema" in self.config.db:
        engine.update_execution_options(
            schema_translate_map={SCHEMA: self.config.db.schema}
        )

    return engine

create_session

create_session() -> Session

Factory function for database session.

Returns:

Type Description
Session

SQLAlchemy session.

Source code in dlunch/models.py
747
748
749
750
751
752
753
754
755
756
def create_session(self) -> Session:
    """Factory function for database session.

    Returns:
        Session: SQLAlchemy session.
    """
    engine = self.create_engine()
    session = Session(engine)

    return session

get_db_dialect staticmethod

get_db_dialect(
    db_obj: Session | Connection | Connection,
) -> str

Return database type (postgresql, sqlite, etc.) based on the database object passed as input. If a session is passed, the database type is set based on an internal map (see models._DBTYPE_MAP).

Parameters:

Name Type Description Default
db_obj Session | Connection | Connection

session or connection object.

required

Raises:

Type Description
TypeError

db_obj shall be a session or a connection object.

Returns:

Type Description
str

database dialect.

Source code in dlunch/models.py
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
@staticmethod
def get_db_dialect(
    db_obj: Session | ConnectionSqlite | ConnectionPostgresql,
) -> str:
    """Return database type (postgresql, sqlite, etc.) based on the database object passed as input.
    If a session is passed, the database type is set based on an internal map (see `models._DBTYPE_MAP`).

    Args:
        db_obj (Session | ConnectionSqlite | ConnectionPostgresql): session or connection object.

    Raises:
        TypeError: db_obj shall be a session or a connection object.

    Returns:
        str: database dialect.
    """
    if isinstance(db_obj, Session):
        dialect = db_obj.bind.dialect.name
    elif isinstance(db_obj, ConnectionSqlite) or isinstance(
        db_obj, ConnectionPostgresql
    ):
        module = db_obj.__class__.__module__.split(".", 1)[0]
        dialect = _MODULE_TO_DIALECT_MAP[module]
    else:
        raise TypeError("db_obj should be a session or connection object")

    return dialect

get_flag

get_flag(
    id: str, value_if_missing: bool | None = None
) -> bool | None

Get the value of a flag. Optionally select the values to return if the flag is missing (default to None).

Parameters:

Name Type Description Default
id str

flag id (name).

required
value_if_missing bool | None

value to return if the flag does not exist. Defaults to None.

None

Returns:

Type Description
bool | None

flag value.

Source code in dlunch/models.py
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
def get_flag(
    self, id: str, value_if_missing: bool | None = None
) -> bool | None:
    """Get the value of a flag.
    Optionally select the values to return if the flag is missing (default to None).

    Args:
        id (str): flag id (name).
        value_if_missing (bool | None, optional): value to return if the flag does not exist. Defaults to None.

    Returns:
        bool | None: flag value.
    """

    session = self.create_session()
    flag = session.get(Flags, id)
    if flag is None:
        value = value_if_missing
    else:
        value = flag.value
    return value

read_sql_query staticmethod

read_sql_query(session: Session, query: str) -> DataFrame

Read a SQL query as pandas DataFrame.

Parameters:

Name Type Description Default
session Session

SQLAlchemy session object.

required
query str

SQL query.

required

Returns:

Type Description
DataFrame

dataframe with query result.

Source code in dlunch/models.py
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
@staticmethod
def read_sql_query(session: Session, query: str) -> pd.DataFrame:
    """Read a SQL query as pandas DataFrame.

    Args:
        session (Session): SQLAlchemy session object.
        query (str): SQL query.

    Returns:
        pd.DataFrame: dataframe with query result.
    """
    results = session.execute(text(query))
    columns = results.keys()
    # Pass columns to build dataframe with correct columns names even if
    # the query returns no rows
    df = pd.DataFrame(results.all(), columns=columns)

    return df

session_add_with_upsert staticmethod

session_add_with_upsert(
    session: Session,
    constraint: str,
    new_record: Stats | Flags,
) -> None

Use an upsert statement for postgresql to add a new record to a table, a simple session add otherwise.

Parameters:

Name Type Description Default
session Session

SQLAlchemy session object.

required
constraint str

constraint used for upsert (usually the primary key)

required
new_record Stats | Flags

table resord (valid tables are stats or flags)

required
Source code in dlunch/models.py
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
@staticmethod
def session_add_with_upsert(
    session: Session, constraint: str, new_record: Stats | Flags
) -> None:
    """Use an upsert statement for postgresql to add a new record to a table,
    a simple session add otherwise.

    Args:
        session (Session): SQLAlchemy session object.
        constraint (str): constraint used for upsert (usually the primary key)
        new_record (Stats | Flags): table resord (valid tables are `stats` or `flags`)
    """
    # Use an upsert for postgresql (for sqlite an 'on conflict replace' is set
    # on table, so session.add is fine)
    if DatabaseConnector.get_db_dialect(session) == "postgresql":
        insert_statement = postgresql_upsert(new_record.__table__).values(
            {
                column.name: getattr(new_record, column.name)
                for column in new_record.__table__.c
                if getattr(new_record, column.name, None) is not None
            }
        )
        upsert_statement = insert_statement.on_conflict_do_update(
            constraint=constraint,
            set_={
                column.name: getattr(
                    insert_statement.excluded, column.name
                )
                for column in insert_statement.excluded
            },
        )
        session.execute(upsert_statement)
    else:
        session.add(new_record)

set_flag

set_flag(id: str, value: bool) -> None

Set a key,value pair inside flag table.

Parameters:

Name Type Description Default
id str

flag id (name).

required
value bool

flag value.

required
Source code in dlunch/models.py
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
def set_flag(self, id: str, value: bool) -> None:
    """Set a key,value pair inside `flag` table.

    Args:
        id (str): flag id (name).
        value (bool): flag value.
    """

    session = self.create_session()

    with session:
        # Write the selected flag (it will be overwritten if exists)
        new_flag = Flags(id=id, value=value)

        # Use an upsert for postgresql, a simple session add otherwise
        DatabaseConnector.session_add_with_upsert(
            session=session, constraint="flags_pkey", new_record=new_flag
        )

        session.commit()

Encrypted

Bases: TypeDecorator

Allows storing and retrieving password hashes using PasswordHash.

Methods:

Name Description
process_bind_param

Ensure the value is a PasswordEncrypt and then return the encrypted password.

process_result_value

Convert the hash to a PasswordEncrypt, if it's non-NULL.

validator

Provides a validator/converter used by @validates.

Attributes:

Name Type Description
impl String

Base column implementation.

Source code in dlunch/models.py
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
class Encrypted(TypeDecorator):
    """Allows storing and retrieving password hashes using PasswordHash."""

    impl: String = String
    """Base column implementation."""

    def process_bind_param(
        self, value: auth.PasswordEncrypt | str | None, dialect
    ) -> str | None:
        """Ensure the value is a PasswordEncrypt and then return the encrypted password.

        Args:
            value (auth.PasswordEncrypt | str | None): input value (plain password or encrypted or `None` if empty)
            dialect (Any): dialect (not used).

        Returns:
            str | None: encrypted password or `None` if empty.
        """
        converted_value = self._convert(value)
        if converted_value:
            return converted_value.encrypted_password
        else:
            return None

    def process_result_value(
        self, value: str | None, dialect
    ) -> auth.PasswordEncrypt | None:
        """Convert the hash to a PasswordEncrypt, if it's non-NULL.

        Args:
            value (str | None): input value (plain password or encrypted or `None` if empty)
            dialect (Any): dialect (not used).

        Returns:
            auth.PasswordEncrypt | None: encrypted password as object or `None` (if nothing is passed as value).
        """
        if value is not None:
            return auth.PasswordEncrypt(value)

    def validator(
        self, password: auth.PasswordEncrypt | str | None
    ) -> auth.PasswordEncrypt | None:
        """Provides a validator/converter used by @validates.

        Args:
            password (auth.PasswordEncrypt | str | None): input value (plain password or encrypted or `None` if empty)

        Returns:
            auth.PasswordEncrypt | None: encrypted password as object or `None` (if nothing is passed as value).
        """
        return self._convert(password)

    def _convert(
        self, value: auth.PasswordEncrypt | str | None
    ) -> auth.PasswordEncrypt | None:
        """Returns a PasswordEncrypt from the given string.

        PasswordEncrypt instances or None values will return unchanged.
        Strings will be encrypted and the resulting PasswordEncrypt returned.
        Any other input will result in a TypeError.

        Args:
            value (auth.PasswordEncrypt | str | None): input value (plain password or encrypted or `None` if empty)

        Raises:
            TypeError: unknown type.

        Returns:
            auth.PasswordEncrypt | None: encrypted password as object or `None` (if nothing is passed as value).
        """
        if isinstance(value, auth.PasswordEncrypt):
            return value
        elif isinstance(value, str):
            return auth.PasswordEncrypt.from_str(value)
        elif value is not None:
            raise TypeError(
                f"Cannot initialize PasswordEncrypt from type '{type(value)}'"
            )

        # Reached only if value is None
        return None

impl class-attribute instance-attribute

impl: String = String

Base column implementation.

process_bind_param

process_bind_param(
    value: PasswordEncrypt | str | None, dialect
) -> str | None

Ensure the value is a PasswordEncrypt and then return the encrypted password.

Parameters:

Name Type Description Default
value PasswordEncrypt | str | None

input value (plain password or encrypted or None if empty)

required
dialect Any

dialect (not used).

required

Returns:

Type Description
str | None

encrypted password or None if empty.

Source code in dlunch/models.py
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
def process_bind_param(
    self, value: auth.PasswordEncrypt | str | None, dialect
) -> str | None:
    """Ensure the value is a PasswordEncrypt and then return the encrypted password.

    Args:
        value (auth.PasswordEncrypt | str | None): input value (plain password or encrypted or `None` if empty)
        dialect (Any): dialect (not used).

    Returns:
        str | None: encrypted password or `None` if empty.
    """
    converted_value = self._convert(value)
    if converted_value:
        return converted_value.encrypted_password
    else:
        return None

process_result_value

process_result_value(
    value: str | None, dialect
) -> PasswordEncrypt | None

Convert the hash to a PasswordEncrypt, if it's non-NULL.

Parameters:

Name Type Description Default
value str | None

input value (plain password or encrypted or None if empty)

required
dialect Any

dialect (not used).

required

Returns:

Type Description
PasswordEncrypt | None

encrypted password as object or None (if nothing is passed as value).

Source code in dlunch/models.py
195
196
197
198
199
200
201
202
203
204
205
206
207
208
def process_result_value(
    self, value: str | None, dialect
) -> auth.PasswordEncrypt | None:
    """Convert the hash to a PasswordEncrypt, if it's non-NULL.

    Args:
        value (str | None): input value (plain password or encrypted or `None` if empty)
        dialect (Any): dialect (not used).

    Returns:
        auth.PasswordEncrypt | None: encrypted password as object or `None` (if nothing is passed as value).
    """
    if value is not None:
        return auth.PasswordEncrypt(value)

validator

validator(
    password: PasswordEncrypt | str | None,
) -> PasswordEncrypt | None

Provides a validator/converter used by @validates.

Parameters:

Name Type Description Default
password PasswordEncrypt | str | None

input value (plain password or encrypted or None if empty)

required

Returns:

Type Description
PasswordEncrypt | None

encrypted password as object or None (if nothing is passed as value).

Source code in dlunch/models.py
210
211
212
213
214
215
216
217
218
219
220
221
def validator(
    self, password: auth.PasswordEncrypt | str | None
) -> auth.PasswordEncrypt | None:
    """Provides a validator/converter used by @validates.

    Args:
        password (auth.PasswordEncrypt | str | None): input value (plain password or encrypted or `None` if empty)

    Returns:
        auth.PasswordEncrypt | None: encrypted password as object or `None` (if nothing is passed as value).
    """
    return self._convert(password)

Flags

Bases: CommonTable

Table with global flags used by Data-Lunch.

'No more orders' flag and guest override flags are stored here.

Methods:

Name Description
__repr__

Simple object representation.

clear_guest_override

Clear 'guest_override' flags and return deleted rows

Attributes:

Name Type Description
__tablename__

Name of the table.

id

Flag ID (name).

value

Flag value.

Source code in dlunch/models.py
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
class Flags(CommonTable):
    """Table with global flags used by Data-Lunch.

    'No more orders' flag and guest override flags are stored here.
    """

    __tablename__ = "flags"
    """Name of the table."""
    id = Column(
        String(50),
        primary_key=True,
        nullable=False,
        sqlite_on_conflict_primary_key="REPLACE",
    )
    """Flag ID (name)."""
    value = Column(Boolean, nullable=False)
    """Flag value."""

    @classmethod
    def clear_guest_override(self, config: DictConfig) -> int:
        """Clear 'guest_override' flags and return deleted rows

        Args:
            config (DictConfig): Hydra configuration dictionary.

        Returns:
            int: deleted rows.
        """

        session = DatabaseConnector(config=config).create_session()
        with session:
            # Clean menu
            num_rows_deleted = session.execute(
                delete(self).where(self.id.like("%_guest_override"))
            )
            session.commit()
            log.info(
                f"deleted {num_rows_deleted.rowcount} rows (guest override) from table '{self.__tablename__}'"
            )

        return num_rows_deleted.rowcount

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<FLAG:{self.id} - value:{self.value}>"

__tablename__ class-attribute instance-attribute

__tablename__ = 'flags'

Name of the table.

id class-attribute instance-attribute

id = Column(
    String(50),
    primary_key=True,
    nullable=False,
    sqlite_on_conflict_primary_key="REPLACE",
)

Flag ID (name).

value class-attribute instance-attribute

value = Column(Boolean, nullable=False)

Flag value.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
533
534
535
536
537
538
539
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<FLAG:{self.id} - value:{self.value}>"

clear_guest_override classmethod

clear_guest_override(config: DictConfig) -> int

Clear 'guest_override' flags and return deleted rows

Parameters:

Name Type Description Default
config DictConfig

Hydra configuration dictionary.

required

Returns:

Type Description
int

deleted rows.

Source code in dlunch/models.py
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
@classmethod
def clear_guest_override(self, config: DictConfig) -> int:
    """Clear 'guest_override' flags and return deleted rows

    Args:
        config (DictConfig): Hydra configuration dictionary.

    Returns:
        int: deleted rows.
    """

    session = DatabaseConnector(config=config).create_session()
    with session:
        # Clean menu
        num_rows_deleted = session.execute(
            delete(self).where(self.id.like("%_guest_override"))
        )
        session.commit()
        log.info(
            f"deleted {num_rows_deleted.rowcount} rows (guest override) from table '{self.__tablename__}'"
        )

    return num_rows_deleted.rowcount

Menu

Bases: CommonTable

Table with menu items.

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__tablename__

Name of the table.

id

Menu item ID.

item

Item name.

orders

Orders connected to each menu item.

Source code in dlunch/models.py
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
class Menu(CommonTable):
    """Table with menu items."""

    __tablename__ = "menu"
    """Name of the table."""
    id = Column(Integer, Identity(start=1, cycle=True), primary_key=True)
    """Menu item ID."""
    item = Column(String(250), unique=False, nullable=False)
    """Item name."""
    orders = relationship(
        "Orders",
        back_populates="menu_item",
        cascade="all, delete-orphan",
        passive_deletes=True,
    )
    """Orders connected to each menu item."""

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<MENU_ITEM:{self.id} - {self.item}>"

__tablename__ class-attribute instance-attribute

__tablename__ = 'menu'

Name of the table.

id class-attribute instance-attribute

id = Column(
    Integer, Identity(start=1, cycle=True), primary_key=True
)

Menu item ID.

item class-attribute instance-attribute

item = Column(String(250), unique=False, nullable=False)

Item name.

orders class-attribute instance-attribute

orders = relationship(
    "Orders",
    back_populates="menu_item",
    cascade="all, delete-orphan",
    passive_deletes=True,
)

Orders connected to each menu item.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
359
360
361
362
363
364
365
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<MENU_ITEM:{self.id} - {self.item}>"

Orders

Bases: CommonTable

Table with items that belongs to an order.

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__tablename__

Name of the table.

id

Order ID.

menu_item

Menu items connected to each order (see menu table).

menu_item_id

ID of the menu item included in the order.

note

Note field attached to the order.

user

User that placed the order.

user_record

User connected to this order.

Source code in dlunch/models.py
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
class Orders(CommonTable):
    """Table with items that belongs to an order."""

    __tablename__ = "orders"
    """Name of the table."""
    id = Column(Integer, Identity(start=1, cycle=True), primary_key=True)
    """Order ID."""
    user = Column(
        String(100),
        ForeignKey("users.id", ondelete="CASCADE"),
        index=True,
        nullable=False,
    )
    """User that placed the order."""
    user_record = relationship("Users", back_populates="orders", uselist=False)
    """User connected to this order."""
    menu_item_id = Column(
        Integer,
        ForeignKey("menu.id", ondelete="CASCADE"),
        nullable=False,
    )
    """ID of the menu item included in the order."""
    menu_item = relationship("Menu", back_populates="orders")
    """Menu items connected to each order (see `menu` table)."""
    note = Column(String(300), unique=False, nullable=True)
    """Note field attached to the order."""

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<ORDER:{self.user}, {self.menu_item.item}>"

__tablename__ class-attribute instance-attribute

__tablename__ = 'orders'

Name of the table.

id class-attribute instance-attribute

id = Column(
    Integer, Identity(start=1, cycle=True), primary_key=True
)

Order ID.

menu_item class-attribute instance-attribute

menu_item = relationship('Menu', back_populates='orders')

Menu items connected to each order (see menu table).

menu_item_id class-attribute instance-attribute

menu_item_id = Column(
    Integer,
    ForeignKey("menu.id", ondelete="CASCADE"),
    nullable=False,
)

ID of the menu item included in the order.

note class-attribute instance-attribute

note = Column(String(300), unique=False, nullable=True)

Note field attached to the order.

user class-attribute instance-attribute

user = Column(
    String(100),
    ForeignKey("users.id", ondelete="CASCADE"),
    index=True,
    nullable=False,
)

User that placed the order.

user_record class-attribute instance-attribute

user_record = relationship(
    "Users", back_populates="orders", uselist=False
)

User connected to this order.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
395
396
397
398
399
400
401
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<ORDER:{self.user}, {self.menu_item.item}>"

Password

Bases: TypeDecorator

Allows storing and retrieving password hashes using PasswordHash.

Methods:

Name Description
process_bind_param

Ensure the value is a PasswordHash and then return its hash.

process_result_value

Convert the hash to a PasswordHash, if it's non-NULL.

validator

Provides a validator/converter used by @validates.

Attributes:

Name Type Description
impl String

Base column implementation.

Source code in dlunch/models.py
 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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
class Password(TypeDecorator):
    """Allows storing and retrieving password hashes using PasswordHash."""

    impl: String = String
    """Base column implementation."""

    def process_bind_param(
        self, value: auth.PasswordHash | str | None, dialect
    ) -> str:
        """Ensure the value is a PasswordHash and then return its hash.

        Args:
            value (auth.PasswordHash | str): input value (plain password or hash, or `None` if empty).
            dialect (Any): dialect (not used).

        Returns:
            str: password hash.
        """
        return self._convert(value).hashed_password

    def process_result_value(
        self, value: str | None, dialect
    ) -> auth.PasswordHash | None:
        """Convert the hash to a PasswordHash, if it's non-NULL.

        Args:
            value (str | None): password hash (or `None` if empty).
            dialect (Any): dialect (not used).

        Returns:
            auth.PasswordHash | None: hashed password as object or `None` (if nothing is passed as value).
        """
        if value is not None:
            return auth.PasswordHash(value)

    def validator(
        self, password: auth.PasswordHash | str | None
    ) -> auth.PasswordHash | None:
        """Provides a validator/converter used by @validates.

        Args:
            password (auth.PasswordHash | str | None): input value (plain password or hash or `None` if empty).

        Returns:
            auth.PasswordHash | None: hashed password as object or `None` (if nothing is passed as value).
        """
        return self._convert(password)

    def _convert(
        self, value: auth.PasswordHash | str | None
    ) -> auth.PasswordHash | None:
        """Returns a PasswordHash from the given string.

        PasswordHash instances or None values will return unchanged.
        Strings will be hashed and the resulting PasswordHash returned.
        Any other input will result in a TypeError.

        Args:
            value (auth.PasswordHash | str | None): input value (plain password or hash or `None` if empty).

        Raises:
            TypeError: unknown type.

        Returns:
            auth.PasswordHash | None: hashed password as object or `None` (if nothing is passed as value).
        """
        if isinstance(value, auth.PasswordHash):
            return value
        elif isinstance(value, str):
            return auth.PasswordHash.from_str(value)
        elif value is not None:
            raise TypeError(
                f"Cannot initialize PasswordHash from type '{type(value)}'"
            )

        # Reached only if value is None
        return None

impl class-attribute instance-attribute

impl: String = String

Base column implementation.

process_bind_param

process_bind_param(
    value: PasswordHash | str | None, dialect
) -> str

Ensure the value is a PasswordHash and then return its hash.

Parameters:

Name Type Description Default
value PasswordHash | str

input value (plain password or hash, or None if empty).

required
dialect Any

dialect (not used).

required

Returns:

Type Description
str

password hash.

Source code in dlunch/models.py
 98
 99
100
101
102
103
104
105
106
107
108
109
110
def process_bind_param(
    self, value: auth.PasswordHash | str | None, dialect
) -> str:
    """Ensure the value is a PasswordHash and then return its hash.

    Args:
        value (auth.PasswordHash | str): input value (plain password or hash, or `None` if empty).
        dialect (Any): dialect (not used).

    Returns:
        str: password hash.
    """
    return self._convert(value).hashed_password

process_result_value

process_result_value(
    value: str | None, dialect
) -> PasswordHash | None

Convert the hash to a PasswordHash, if it's non-NULL.

Parameters:

Name Type Description Default
value str | None

password hash (or None if empty).

required
dialect Any

dialect (not used).

required

Returns:

Type Description
PasswordHash | None

hashed password as object or None (if nothing is passed as value).

Source code in dlunch/models.py
112
113
114
115
116
117
118
119
120
121
122
123
124
125
def process_result_value(
    self, value: str | None, dialect
) -> auth.PasswordHash | None:
    """Convert the hash to a PasswordHash, if it's non-NULL.

    Args:
        value (str | None): password hash (or `None` if empty).
        dialect (Any): dialect (not used).

    Returns:
        auth.PasswordHash | None: hashed password as object or `None` (if nothing is passed as value).
    """
    if value is not None:
        return auth.PasswordHash(value)

validator

validator(
    password: PasswordHash | str | None,
) -> PasswordHash | None

Provides a validator/converter used by @validates.

Parameters:

Name Type Description Default
password PasswordHash | str | None

input value (plain password or hash or None if empty).

required

Returns:

Type Description
PasswordHash | None

hashed password as object or None (if nothing is passed as value).

Source code in dlunch/models.py
127
128
129
130
131
132
133
134
135
136
137
138
def validator(
    self, password: auth.PasswordHash | str | None
) -> auth.PasswordHash | None:
    """Provides a validator/converter used by @validates.

    Args:
        password (auth.PasswordHash | str | None): input value (plain password or hash or `None` if empty).

    Returns:
        auth.PasswordHash | None: hashed password as object or `None` (if nothing is passed as value).
    """
    return self._convert(password)

PrivilegedUsers

Bases: CommonTable

Table with user that have privileges (normal users and admin).

If enabled guests are all the authenticated users that do not belong to this table (see config key auth.authorize_guest_users and basic_auth.guest_user)

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__tablename__

Name of the table.

admin

Admin flag (true if admin).

user

User name.

Source code in dlunch/models.py
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
class PrivilegedUsers(CommonTable):
    """Table with user that have privileges (normal users and admin).

    If enabled guests are all the authenticated users that do not belong to this table
    (see config key `auth.authorize_guest_users` and `basic_auth.guest_user`)
    """

    __tablename__ = "privileged_users"
    """Name of the table."""
    user = Column(
        String(100),
        primary_key=True,
        sqlite_on_conflict_primary_key="REPLACE",
    )
    """User name."""
    admin = Column(
        Boolean, nullable=False, default=False, server_default=sql_false()
    )
    """Admin flag (true if admin)."""

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<PRIVILEGED_USER:{self.id}>"

__tablename__ class-attribute instance-attribute

__tablename__ = 'privileged_users'

Name of the table.

admin class-attribute instance-attribute

admin = Column(
    Boolean,
    nullable=False,
    default=False,
    server_default=false(),
)

Admin flag (true if admin).

user class-attribute instance-attribute

user = Column(
    String(100),
    primary_key=True,
    sqlite_on_conflict_primary_key="REPLACE",
)

User name.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
563
564
565
566
567
568
569
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<PRIVILEGED_USER:{self.id}>"

Stats

Bases: CommonTable

Table with number of users that ate a lunch, grouped by guest type.

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__table_args__

Table arguments, used to create primary key (ON CONFLICT options for composite

__tablename__

Name of the table.

date

Day to which the statistics refers to.

guest

Different kind of guests are identified by the value defined in config files

hungry_people

Number of people that ate in a certain day.

Source code in dlunch/models.py
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
class Stats(CommonTable):
    """Table with number of users that ate a lunch, grouped by guest type."""

    # Primary key handled with __table_args__ because ON CONFLICT for composite
    # primary key is available only with __table_args__
    __tablename__ = "stats"
    """Name of the table."""
    __table_args__ = (
        PrimaryKeyConstraint(
            "date", "guest", name="stats_pkey", sqlite_on_conflict="REPLACE"
        ),
    )
    """Table arguments, used to create primary key (ON CONFLICT options for composite
    primary key is available only with __table_args__)."""
    date = Column(
        Date,
        nullable=False,
        server_default=func.current_date(),
    )
    """Day to which the statistics refers to."""
    guest = Column(
        String(20),
        nullable=True,
        default="NotAGuest",
        server_default="NotAGuest",
    )
    """Different kind of guests are identified by the value defined in config files
    (see config key `panel.guest_types`).
    'NotAGuest' is the value used for locals.
    """
    hungry_people = Column(
        Integer, nullable=False, default=0, server_default="0"
    )
    """Number of people that ate in a certain day.
    different kind of guests are identified by the value in guest column.
    """

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<STAT:{self.id} - HP:{self.hungry_people} - HG:{self.hungry_guests}>"

__table_args__ class-attribute instance-attribute

__table_args__ = PrimaryKeyConstraint(
    "date",
    "guest",
    name="stats_pkey",
    sqlite_on_conflict="REPLACE",
)

Table arguments, used to create primary key (ON CONFLICT options for composite primary key is available only with table_args).

__tablename__ class-attribute instance-attribute

__tablename__ = 'stats'

Name of the table.

date class-attribute instance-attribute

date = Column(
    Date, nullable=False, server_default=current_date()
)

Day to which the statistics refers to.

guest class-attribute instance-attribute

guest = Column(
    String(20),
    nullable=True,
    default="NotAGuest",
    server_default="NotAGuest",
)

Different kind of guests are identified by the value defined in config files (see config key panel.guest_types). 'NotAGuest' is the value used for locals.

hungry_people class-attribute instance-attribute

hungry_people = Column(
    Integer, nullable=False, default=0, server_default="0"
)

Number of people that ate in a certain day. different kind of guests are identified by the value in guest column.

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
482
483
484
485
486
487
488
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<STAT:{self.id} - HP:{self.hungry_people} - HG:{self.hungry_guests}>"

Users

Bases: CommonTable

Table with users that placed an order.

Methods:

Name Description
__repr__

Simple object representation.

Attributes:

Name Type Description
__tablename__

Name of the table.

guest

Guest flag (true if guest).

id

User ID.

lunch_time

User selected lunch time.

orders

Orders connected to each user.

takeaway

Takeaway flag (true if takeaway).

Source code in dlunch/models.py
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
class Users(CommonTable):
    """Table with users that placed an order."""

    __tablename__ = "users"
    """Name of the table."""
    id = Column(
        String(100),
        primary_key=True,
        nullable=False,
    )
    """User ID."""
    guest = Column(
        String(20),
        nullable=False,
        default="NotAGuest",
        server_default="NotAGuest",
    )
    """Guest flag (true if guest)."""
    lunch_time = Column(String(7), index=True, nullable=False)
    """User selected lunch time."""
    takeaway = Column(
        Boolean, nullable=False, default=False, server_default=sql_false()
    )
    """Takeaway flag (true if takeaway)."""
    orders = relationship(
        "Orders",
        back_populates="user_record",
        cascade="all, delete-orphan",
        passive_deletes=True,
    )
    """Orders connected to each user."""

    def __repr__(self) -> str:
        """Simple object representation.

        Returns:
            str: string representation.
        """
        return f"<USER:{self.id}>"

__tablename__ class-attribute instance-attribute

__tablename__ = 'users'

Name of the table.

guest class-attribute instance-attribute

guest = Column(
    String(20),
    nullable=False,
    default="NotAGuest",
    server_default="NotAGuest",
)

Guest flag (true if guest).

id class-attribute instance-attribute

id = Column(String(100), primary_key=True, nullable=False)

User ID.

lunch_time class-attribute instance-attribute

lunch_time = Column(String(7), index=True, nullable=False)

User selected lunch time.

orders class-attribute instance-attribute

orders = relationship(
    "Orders",
    back_populates="user_record",
    cascade="all, delete-orphan",
    passive_deletes=True,
)

Orders connected to each user.

takeaway class-attribute instance-attribute

takeaway = Column(
    Boolean,
    nullable=False,
    default=False,
    server_default=false(),
)

Takeaway flag (true if takeaway).

__repr__

__repr__() -> str

Simple object representation.

Returns:

Type Description
str

string representation.

Source code in dlunch/models.py
436
437
438
439
440
441
442
def __repr__(self) -> str:
    """Simple object representation.

    Returns:
        str: string representation.
    """
    return f"<USER:{self.id}>"

set_sqlite_pragma

set_sqlite_pragma(dbapi_connection, connection_record)

Force foreign key constraints for sqlite connections.

Parameters:

Name Type Description Default
dbapi_connection Any

connection to database. Shall have a cursor method.

required
connection_record Any

connection record (not used).

required
Source code in dlunch/models.py
77
78
79
80
81
82
83
84
85
86
87
88
@event.listens_for(Engine, "connect")
def set_sqlite_pragma(dbapi_connection, connection_record):
    """Force foreign key constraints for sqlite connections.

    Args:
        dbapi_connection (Any): connection to database. Shall have a `cursor` method.
        connection_record (Any): connection record (not used).
    """
    if DatabaseConnector.get_db_dialect(dbapi_connection) == "sqlite":
        cursor = dbapi_connection.cursor()
        cursor.execute("PRAGMA foreign_keys=ON;")
        cursor.close()