Хранимая процедура с Alembic: синтаксическая ошибка MySQL

Я использую Alembic для загрузки хранимой процедуры в базу данных MySQL. Следуя поваренной книге, я смог создать необходимые объекты для создания и удаления процедура. Теперь, когда я хочу обновить версию, чтобы фактически загрузить процедуру, я получаю синтаксис SQL рядом с DELIMITER $$, который мне нужен для определения процедуры. Я также пытался удалить DELIMITER и заменить AS для запуска процедуры, но это тоже не сработало. Я даже попробовал простой пример функции из кулинарной книги, но это также дало мне ту же синтаксическую ошибку рядом с $.

class ReplaceableObject:
    def __init__(self, name, sqltext):
        self.name = name
        self.sqltext = sqltext


class ReversibleOp(MigrateOperation):
    def __init__(self, target):
        self.target = target

    @classmethod
    def invoke_for_target(cls, operations, target):
        op = cls(target)
        return operations.invoke(op)

    def reverse(self):
        raise NotImplementedError()

    @classmethod
    def _get_object_from_version(cls, operations, ident):
        version, objname = ident.split(".")

        module = operations.get_context().script.get_revision(version).module
        obj = getattr(module, objname)
        return obj

    @classmethod
    def replace(cls, operations, target, replaces=None, replace_with=None):

        if replaces:
            old_obj = cls._get_object_from_version(operations, replaces)
            drop_old = cls(old_obj).reverse()
            create_new = cls(target)
        elif replace_with:
            old_obj = cls._get_object_from_version(operations, replace_with)
            drop_old = cls(target).reverse()
            create_new = cls(old_obj)
        else:
            raise TypeError("replaces or replace_with is required")

        operations.invoke(drop_old)
        operations.invoke(create_new)


@Operations.register_operation("create_sp", "invoke_for_target")
@Operations.register_operation("replace_sp", "replace")
class CreateSPOp(ReversibleOp):
    def reverse(self):
        return DropSPOp(self.target)


@Operations.register_operation("drop_sp", "invoke_for_target")
class DropSPOp(ReversibleOp):
    def reverse(self):
        return CreateSPOp(self.target)


@Operations.implementation_for(CreateSPOp)
def create_sp(operations, operation):
    operations.execute(
        """
        DELIMITER $$
        CREATE PROCEDURE %s %s
        DELIMITER ;
        """ % (
            operation.target.name, operation.target.sqltext
        )
    )


@Operations.implementation_for(DropSPOp)
def drop_sp(operations, operation):
    operations.execute("DROP PROCEDURE IF EXISTS %s" % operation.target.name)

И вот как я пытаюсь его создать:

my_procedure = ReplaceableObject(
    "my_procedure(IN arg1 TEXT, IN arg2 TEXT, OUT res TEXT)",
    """
    BEGIN
        -- procedure implementation
    END $$   
    """
)

def upgrade():
    op.create_sp(my_procedure)

def downgrade():
    op.drop_sp(my_procedure)

И это ошибка (я использую pymysql в качестве клиента):

sqlalchemy.exc.ProgrammingError: (pymysql.err.ProgrammingError) (1064, "You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'DELIMITER $$\n        CREATE PROCEDURE my_procedure(IN arg1 TEXT, IN arg2 TEXT, OU' at line 1")

person A23149577    schedule 13.09.2019    source источник


Ответы (1)


Вам не нужно DELIMITER, кроме как в среде, которая поддерживает серию операторов, разделенных точкой с запятой — клиент MySQL.

Фактически, команда DELIMITER является встроенным клиентом mysql, и он не распознается сервером MySQL.

Вы не должны использовать DELIMITER при отправке запросов на сервер MySQL через API, даже если это CREATE PROCEDURE.

person Bill Karwin    schedule 13.09.2019