Skip to content

Alembic

The tool to check the availability or syntax of domain, IP or URL.

::

██████╗ ██╗   ██╗███████╗██╗   ██╗███╗   ██╗ ██████╗███████╗██████╗ ██╗     ███████╗
██╔══██╗╚██╗ ██╔╝██╔════╝██║   ██║████╗  ██║██╔════╝██╔════╝██╔══██╗██║     ██╔════╝
██████╔╝ ╚████╔╝ █████╗  ██║   ██║██╔██╗ ██║██║     █████╗  ██████╔╝██║     █████╗
██╔═══╝   ╚██╔╝  ██╔══╝  ██║   ██║██║╚██╗██║██║     ██╔══╝  ██╔══██╗██║     ██╔══╝
██║        ██║   ██║     ╚██████╔╝██║ ╚████║╚██████╗███████╗██████╔╝███████╗███████╗
╚═╝        ╚═╝   ╚═╝      ╚═════╝ ╚═╝  ╚═══╝ ╚═════╝╚══════╝╚═════╝ ╚══════╝╚══════╝

Provides our very own alembic interface.

Author: Nissar Chababy, @funilrys, contactTATAfunilrysTODTODcom

Special thanks: https://pyfunceble.github.io/special-thanks.html

Contributors: https://pyfunceble.github.io/contributors.html

Project link: https://github.com/funilrys/PyFunceble

Project documentation: https://docs.pyfunceble.com

Project homepage: https://pyfunceble.github.io/

License: ::

Copyright 2017, 2018, 2019, 2020, 2022, 2023, 2024 Nissar Chababy

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    https://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

Alembic

Provides our very own alambic handler.

Source code in PyFunceble/cli/migrators/alembic.py
class Alembic:
    """
    Provides our very own alambic handler.
    """

    db_session: Optional[Session] = None
    migrator_base: Optional[DBMigratorBase] = None

    alembic_config: Optional[alembic.config.Config] = None

    def __init__(self, db_session: Session) -> None:
        self.db_session = db_session

        self.migrator_base = DBMigratorBase()
        self.migrator_base.db_session = db_session

    def execute_if_authorized(default: Any = None):  # pylint: disable=no-self-argument
        """
        Executes the decorated method only if we are authorized to process.
        Otherwise, apply the given :code:`default`.
        """

        def inner_metdhod(func):
            @functools.wraps(func)
            def wrapper(self, *args, **kwargs):
                if self.authorized:
                    return func(self, *args, **kwargs)  # pylint: disable=not-callable
                return self if default is None else default

            return wrapper

        return inner_metdhod

    @property
    def authorized(self):
        """
        Provides the authorization to process.
        """

        # Here we explicitly start because the usage of alembic may be out
        # of our running context.
        return PyFunceble.cli.facility.CredentialLoader.is_already_loaded()

    @property
    def migration_directory(self) -> str:
        """
        Provides the location of our migration directory.
        """

        with package_resources.path(
            f"PyFunceble.data.{PyFunceble.cli.storage.ALEMBIC_DIRECTORY_NAME}",
            "__init__.py",
        ) as file_path:
            result = os.path.split(file_path)[0]

        if PyFunceble.storage.CONFIGURATION.cli_testing.db_type == "postgresql":
            return os.path.join(result, "postgresql")
        return os.path.join(result, "mysql")

    @execute_if_authorized(None)
    def configure(self) -> "Alembic":
        """
        Configure our alembic configuration based on what we need.
        """

        if self.alembic_config is None:
            self.alembic_config = alembic.config.Config()

        self.alembic_config.set_main_option("script_location", self.migration_directory)
        self.alembic_config.set_main_option(
            "sqlalchemy.url",
            PyFunceble.cli.facility.CredentialLoader.get_uri(),
        )

        return self

    def is_revision_different(self, revision: str) -> bool:
        """
        Checks if the given revision is already set.

        :param revision:
            The revision to check
        """

        revision_id = (
            ScriptDirectory.from_config(self.alembic_config)
            .get_revision(revision)
            .revision
        )

        statement = text(
            "SELECT * from alembic_version WHERE version_num = :db_revision"
        )

        result = self.db_session.execute(statement, {"db_revision": revision_id})

        return result.fetchone() is None

    @execute_if_authorized(None)
    def upgrade(self, revision: str = "head") -> "Alembic":
        """
        Upgrades the database structure.

        :param revision:
            The revision to apply.
        """

        self.configure()

        if not self.migrator_base.does_table_exists(
            "alembic_version"
        ) or self.is_revision_different(revision):
            PyFunceble.facility.Logger.info(
                "Started update (%r) of the database schema(s).", revision
            )

            alembic_command.upgrade(self.alembic_config, revision)

            PyFunceble.facility.Logger.info(
                "Finished update (%r) of the database schema(s).", revision
            )

    @execute_if_authorized(None)
    def downgrade(self, revision: str = "head") -> "Alembic":
        """
        Upgrades the database structure.

        :param revision:
            The revision to apply.
        """

        self.configure()

        if not self.migrator_base.does_table_exists(
            "alembic_version"
        ) or self.is_revision_different(revision):
            PyFunceble.facility.Logger.info(
                "Started downgrade (%r) of the database schema(s).", revision
            )

            alembic_command.downgrade(self.alembic_config, revision)

            PyFunceble.facility.Logger.info(
                "Finished downgrade (%r) of the database schema(s).", revision
            )

authorized property

Provides the authorization to process.

migration_directory: str property

Provides the location of our migration directory.

configure()

Configure our alembic configuration based on what we need.

Source code in PyFunceble/cli/migrators/alembic.py
@execute_if_authorized(None)
def configure(self) -> "Alembic":
    """
    Configure our alembic configuration based on what we need.
    """

    if self.alembic_config is None:
        self.alembic_config = alembic.config.Config()

    self.alembic_config.set_main_option("script_location", self.migration_directory)
    self.alembic_config.set_main_option(
        "sqlalchemy.url",
        PyFunceble.cli.facility.CredentialLoader.get_uri(),
    )

    return self

downgrade(revision='head')

Upgrades the database structure.

Parameters:

Name Type Description Default
revision str

The revision to apply.

'head'
Source code in PyFunceble/cli/migrators/alembic.py
@execute_if_authorized(None)
def downgrade(self, revision: str = "head") -> "Alembic":
    """
    Upgrades the database structure.

    :param revision:
        The revision to apply.
    """

    self.configure()

    if not self.migrator_base.does_table_exists(
        "alembic_version"
    ) or self.is_revision_different(revision):
        PyFunceble.facility.Logger.info(
            "Started downgrade (%r) of the database schema(s).", revision
        )

        alembic_command.downgrade(self.alembic_config, revision)

        PyFunceble.facility.Logger.info(
            "Finished downgrade (%r) of the database schema(s).", revision
        )

execute_if_authorized(default=None)

Executes the decorated method only if we are authorized to process. Otherwise, apply the given :code:default.

Source code in PyFunceble/cli/migrators/alembic.py
def execute_if_authorized(default: Any = None):  # pylint: disable=no-self-argument
    """
    Executes the decorated method only if we are authorized to process.
    Otherwise, apply the given :code:`default`.
    """

    def inner_metdhod(func):
        @functools.wraps(func)
        def wrapper(self, *args, **kwargs):
            if self.authorized:
                return func(self, *args, **kwargs)  # pylint: disable=not-callable
            return self if default is None else default

        return wrapper

    return inner_metdhod

is_revision_different(revision)

Checks if the given revision is already set.

Parameters:

Name Type Description Default
revision str

The revision to check

required
Source code in PyFunceble/cli/migrators/alembic.py
def is_revision_different(self, revision: str) -> bool:
    """
    Checks if the given revision is already set.

    :param revision:
        The revision to check
    """

    revision_id = (
        ScriptDirectory.from_config(self.alembic_config)
        .get_revision(revision)
        .revision
    )

    statement = text(
        "SELECT * from alembic_version WHERE version_num = :db_revision"
    )

    result = self.db_session.execute(statement, {"db_revision": revision_id})

    return result.fetchone() is None

upgrade(revision='head')

Upgrades the database structure.

Parameters:

Name Type Description Default
revision str

The revision to apply.

'head'
Source code in PyFunceble/cli/migrators/alembic.py
@execute_if_authorized(None)
def upgrade(self, revision: str = "head") -> "Alembic":
    """
    Upgrades the database structure.

    :param revision:
        The revision to apply.
    """

    self.configure()

    if not self.migrator_base.does_table_exists(
        "alembic_version"
    ) or self.is_revision_different(revision):
        PyFunceble.facility.Logger.info(
            "Started update (%r) of the database schema(s).", revision
        )

        alembic_command.upgrade(self.alembic_config, revision)

        PyFunceble.facility.Logger.info(
            "Finished update (%r) of the database schema(s).", revision
        )