Сделайте ошибки SQLAlchemy более удобными и подробными - PullRequest
2 голосов
/ 13 марта 2019

У меня есть такая модель:

class Company(db.Model):
    __tablename__ = "my_table"
    id = db.Column(db.Integer(), primary_key=True)
    name = db.Column(db.String(128), unique=True, nullable=False)
    slug = db.Column(db.String(128), unique=True, nullable=False)

Как вы можете видеть, я использую классы и методы Flask-SQLAlchemy, но это не главное, скажем, у меня есть представлениекоторый выполняет следующие строки:

c = Company("Test", "test")
try:
    db.session.add(c)
    db.session.commit()
    return "Added!"
except Exception as e:
    db.session.rollback()
    return f"{e}"

Приведенный выше код создает объект Company, пытается добавить его в базу данных, выполняет откат транзакции при исключении,

Проблема здесь,Поскольку данные жестко закодированы, они всегда должны возвращать исключение, SQLAlchemy вызывает IntegrityError.

IntegrityError настолько уродлив и бесполезен для пользователя, пример:

(sqlite3.IntegrityError) UNIQUE constraint failed: my_table.name [SQL: 'INSERT INTO my_table (name, slug) VALUES (?, ?)'] [parameters: ('Test', 'tests')] (Background on this error at: http://sqlalche.me/e/gkpj)

Я ищу способ сделать его более привлекательным и удобным для пользователя , до этого я использовал декоратор db.validates и проверял дублирующиеся данные при проверке, но мне это кажется неправильным

Меньше всего мне нужно выяснить, какое поле вызывает проблему без жесткого кодирования

Ответы [ 3 ]

3 голосов
/ 13 марта 2019

SQLAlchemy включает механизм, позволяющий настраивать ошибки DBAPI с помощью перехвата событий handle_error . Я использовал этот API в Openstack oslo.db , который можно увидеть в этом файле: https://github.com/openstack/oslo.db/blob/master/oslo_db/sqlalchemy/exc_filters.py.

Поскольку stackoverflow ненавидит ссылки на код, вот POC, основанный на вышеупомянутом связанном подходе:

import collections
from sqlalchemy import event
from sqlalchemy import exc as sqla_exc
import re


class DuplicateKeyError(Exception):
    """Duplicate entry at unique column error."""

    def __init__(self, columns=None, inner_exception=None, value=None):
        self.columns = columns or []
        self.value = value
        self.inner_exception = inner_exception

    def __str__(self):
        return "Duplicate key for columns %s" % (
            self.columns,
        )


_registry = collections.defaultdict(lambda: collections.defaultdict(list))


def filters(ame, exception_type, regex):
    """Mark a function as receiving a filtered exception."""

    def _receive(fn):
        _registry[ame][exception_type].extend(
            (fn, re.compile(reg))
            for reg in ((regex,) if not isinstance(regex, tuple) else regex)
        )
        return fn

    return _receive


# each @filters() lists a database name, a SQLAlchemy exception to catch,
# and a list of regular expressions that will be matched.  If all the
# conditions match, the handler is called which then raises a nicer
# error message.

@filters(
    "sqlite",
    sqla_exc.IntegrityError,
    (
        r"^.*columns?(?P<columns>[^)]+)(is|are)\s+not\s+unique$",
        r"^.*UNIQUE\s+constraint\s+failed:\s+(?P<columns>.+)$",
        r"^.*PRIMARY\s+KEY\s+must\s+be\s+unique.*$",
    ),
)
def _sqlite_dupe_key_error(integrity_error, match, engine_name, is_disconnect):
    columns = []
    try:
        columns = match.group("columns")
        columns = [c.split(".")[-1] for c in columns.strip().split(", ")]
    except IndexError:
        pass

    raise DuplicateKeyError(columns, integrity_error)


@filters(
    "mysql",
    sqla_exc.IntegrityError,
    r"^.*\b1062\b.*Duplicate entry '(?P<value>.*)'"
    r" for key '(?P<columns>[^']+)'.*$",
)
@filters(
    "postgresql",
    sqla_exc.IntegrityError,
    (
        r'^.*duplicate\s+key.*"(?P<columns>[^"]+)"\s*\n.*'
        r"Key\s+\((?P<key>.*)\)=\((?P<value>.*)\)\s+already\s+exists.*$",
        r"^.*duplicate\s+key.*\"(?P<columns>[^\"]+)\"\s*\n.*$",
    ),
)
def _default_dupe_key_error(
    integrity_error, match, engine_name, is_disconnect
):
    columns = match.group("columns")
    uniqbase = "uniq_"
    if not columns.startswith(uniqbase):
        if engine_name == "postgresql":
            columns = [columns[columns.index("_") + 1 : columns.rindex("_")]]
        else:
            columns = [columns]
    else:
        columns = columns[len(uniqbase) :].split("0")[1:]

    value = match.groupdict().get("value")

    raise DuplicateKeyError(columns, integrity_error, value)


def handler(context):
    """Iterate through available filters and invoke those which match.
    The first one which raises wins.
    """

    def _dialect_registries(engine):
        if engine.dialect.name in _registry:
            yield _registry[engine.dialect.name]
        if "*" in _registry:
            yield _registry["*"]

    for per_dialect in _dialect_registries(context.engine):
        for exc in (context.sqlalchemy_exception, context.original_exception):
            for super_ in exc.__class__.__mro__:
                if super_ in per_dialect:
                    regexp_reg = per_dialect[super_]
                    for fn, regexp in regexp_reg:
                        match = regexp.match(exc.args[0])
                        if match:
                            fn(
                                exc,
                                match,
                                context.engine.dialect.name,
                                context.is_disconnect,
                            )


if __name__ == '__main__':
    from sqlalchemy import Column, Integer, String, create_engine
    from sqlalchemy.orm import Session
    from sqlalchemy.ext.declarative import declarative_base

    Base = declarative_base()


    class Company(Base):
        __tablename__ = "my_table"
        id = Column(Integer(), primary_key=True)
        name = Column(String(128), unique=True, nullable=False)
        slug = Column(String(128), unique=True, nullable=False)

        def __init__(self, name, slug):
            self.name = name
            self.slug = slug

    e = create_engine("sqlite://", echo=True)
    Base.metadata.create_all(e)
    event.listen(e, "handle_error", handler)

    s = Session(e)
    s.add(Company("Test", "test"))
    s.commit()


    s.add(Company("Test", "test"))
    s.commit()

Запустив его, мы видим:

2019-03-13 09:44:51,701 INFO sqlalchemy.engine.base.Engine INSERT INTO my_table (name, slug) VALUES (?, ?)
2019-03-13 09:44:51,701 INFO sqlalchemy.engine.base.Engine ('Test', 'test')

2019-03-13 09:44:53,387 INFO sqlalchemy.engine.base.Engine ROLLBACK
Traceback (most recent call last):
# ...
sqlite3.IntegrityError: UNIQUE constraint failed: my_table.slug

The above exception was the direct cause of the following exception:

Traceback (most recent call last):
 # ...
__main__.DuplicateKeyError: Duplicate key for columns ['slug']
1 голос
/ 13 марта 2019

Вы можете импортировать exception, а затем обработать его самостоятельно:

from sqlite3.__init__ import IntegrityError

это даст вам имя исключения, тогда вы можете сделать что-то вроде:

except IntegrityError :
    db.session.rollback()
    return f"duplicate data has been used!"

или, в противном случае, вам нужно будет обработать это исключение.

помните, что эта ошибка будет обнаружена только в том случае, если вы используете пакет sqlite, а не sqlalchemy, поэтому, если вы изменили db engine где-то по пути, вы не сможете справиться с этим exception.

from sqlalchemy.exc import IntegrityError

является исключением class, которое вам необходимо повысить для sqlalchemy повышенных исключений.

1 голос
/ 13 марта 2019

Вообще говоря, вы можете try/except сообщить об ошибке, "поймать" ее, записать в журнал и затем вернуть пользовательскую ошибку. Как это:

c = Company("Test", "test")
try:
    db.session.add(c)
    db.session.commit()
    return "Added!"
except Exception as e:
    db.session.rollback()
    return f"failed to insert company: {e.__class__.__name__}"

Это похоже на быстрый ответ, а не правильный. Вместо этого я бы добавил некоторую проверку перед попыткой вставки:

c = Company("Test", "test")
# note this is pseudo code
if Company.find.get("Test"):
    try:
        db.session.add(c)
        db.session.commit()
        return "Added!"
    except Exception as e:
        db.session.rollback()
        return f"failed to insert company: {e.__class__.__name__}"
else:
return f"company {c.id} already exists"

Таким образом, вы не выдаете ошибку, вместо этого ваше приложение обрабатывает свои данные, а не вставляет.

...