4

我们从 1 个月开始使用 Flyway,没有遇到任何问题。

但是,今天我尝试添加一个非常长(超过 1500 行)的新迁移脚本,并遇到了一个奇怪的 MySQL 语法错误。

我在 MySQL Workbench 中打开了这个脚本,没有报告语法错误,脚本执行没有错误。

这个名为“V10012__insert-acceptance-testing-event-moment-passed.sql”的脚本包含以下说明。

  • 插入语句 1
  • 插入语句 2
  • ...
  • INSERT 语句 LAST - 1
  • INSERT 语句最后

MySQL报告的错误如下:

[ERROR] 由 com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException 引起:您的 SQL 语法有错误;检查与您的 MySQL 服务器版本相对应的手册,以在第 232 行 [ERROR] com.googlecode.flyway.core. migration.MigrationException: 迁移到版本 10012 失败!请恢复备份并回滚数据库和代码!

该错误在“INSERT statement LAST”语句中报告。

但是,如果我在脚本中反转“INSERT 语句 LAST”和“INSERT 语句 LAST - 1”,则会在“INSERT 语句 LAST - 1”(现在位于文件末尾)上报告错误。所以我在“INSERT statement LAST”中没有错误,因为 Flyway 成功执行了它。

此外,如果现在我从脚本“V10012_insert-acceptance-testing-event-moment-passed.sql”中完全删除“INSERT statement LAST”语句,并将该语句放入一个名为“V10013 _test.sql”的新文件中,flyway 执行我所有的迁移脚本都成功了!

那么,我最初的“V10012__insert-acceptance-testing-event-moment-passed.sql”脚本中可能存在什么问题?

是否有可能的脚本大小限制?

以下是有关我的环境的有用信息:

  • 我的脚本使用了很多 /* */ 注释
  • Flyway Maven 插件 1.7
  • Maven 2.0.3
  • mysql:mysql-connector-java:5.1.21
  • MySQL 5.5.X
  • Java JDK 1.7.0_09-b05
  • Windows 7的
4

2 回答 2

3

我终于找到了问题的原因。

在我的脚本中,我有:

... 
...

/* INSERT statement LAST - 1 */ 
INSERT INTO `table_1` (`id`, `string_1`, `string_2`)
VALUES (
    1, 
    'aaa',   /* COMMENT 1*/
    'bbb'    /* COMMENT 2*/
);

/* INSERT statement LAST */
INSERT INTO `table_2` (`id`, `string_3`)
VALUES (
    1,       /* COMMENT 3   */
    'cccc'   /* COMMENT 4    */
);

问题似乎出在 COMMENT 2 和 COMMENT 4 上。如果我删除主题 flyway 成功执行我的所有迁移脚本。

例如,此脚本将起作用:

... 
...

/* INSERT statement LAST - 1 */ 
INSERT INTO `table_1` (`id`, `string_1`, `string_2`)
VALUES (
    1, 
    'aaa',   /* COMMENT 1*/
    'bbb'    
);

/* INSERT statement LAST */
INSERT INTO `table_2` (`id`, `string_3`)
VALUES (
    1,   /* COMMENT 3   */
    'cccc'   
);

那么,也许这是 Flyway 解析器中的一个错误?

我今天没有时间对其进行测试,但似乎仅在以下情况下才会出现此错误:

  • 我们在 VARCHAR (或我想的其他字符串列类型)之后有评论
  • 此注释位于 ')' 字符之前

这个问题似乎只有在一个 SQL 脚本中存在多个插入语句时才能重现。

此外,我认为没有放在')'之前的评论是正确的,比如:

/* INSERT statement LAST */
INSERT INTO `table_2` (`id`, `string_3`)
VALUES (
    1,   /* COMMENT 3   */
    'cccc',  /* COMMENT 4   */
    'dddd'
);

这里 COMMENT 3 和 COMMENT 4 在 Flyway 中通过,因为它们位于 ',' 之后和字符串值之前,但它们没有放在 ')' 字符之前。

希望这可以帮助;-)

于 2012-10-24T15:17:58.590 回答
1

由于飞行路线,这不太可能。这几乎肯定与 flyway 使用的 MySQL 驱动程序有关。使用 MySql 本机客户端,您自然会认为它可以处理任何合法语法。即使是“可选”的评论块

我知道 MySql 的 jdbc 驱动程序中SURE的一个问题是它无法处理动态分隔符交换。因此,触发器和存储过程可能会成为一个问题,特别是如果它们是使用生成的,mysqldump但随后您使用 Java 应用程序重新摄取。

如果您查看此处并转到spring.datasource.separator您会看到这是在系统范围内(基本上)设置为;.

现在想象一下当涉及到脚本中的这一行时它将如何表现:

DELIMITER ;;
CREATE FUNCTION `int2vancode`(id BIGINT(20)) RETURNS varchar(255)
BEGIN
    DECLARE num VARCHAR(10);
    DECLARE length INT;
    SET num = conv(id, 10, 36);
    SET length = char_length(num);

    return CONCAT(conv(length -1 , 10, 36), num);
END ;;
DELIMITER ;

Java 驱动程序是愚蠢的,并且不会解释任何这些语句,即使该DELIMITER语句显然是针对客户端的

我现在遇到一个问题,即本机客户端执行每个脚本都没有问题,但 flyway 说我有一个语法错误,即使 intellij 也看不到。

** 更新**

我解决了我的问题,它或多或少证实了我看到的问题:我有一个创建表:

CREATE TABLE `Position` (
  PositionID INT NOT NULL AUTO_INCREMENT,
  `Name` varchar(200) COLLATE utf8_unicode_ci NOT NULL
);

即使我的 bash 脚本迭代执行这些脚本没有问题,我也不得不逃脱Name并让 flyway 接受它。Position

也许我可以告诉 Java 驱动程序来帮助它?

于 2017-10-03T21:10:56.843 回答