一次性更改多个MySQL数据库(SAAS数据库更改)

7
我们正在运行一个基于PHP(Zend框架)的应用程序,为每个用户创建一个数据库(出于安全/备份/等原因)。所有这些数据库具有完全相同的结构,并且将始终如此。当我们部署新功能时,我们需要使用新字段/表扩展所有数据库。
我已经了解了使用dbdeploy进行操作的方法,但我不确定他们是否支持一次操作多个数据库(而不是逐个指定名称)。数据库的名称为user1、user2、user3等。
是否有任何好的工具可以使这个过程对我们来说更加容易和少痛苦?我们正在运行phing进行自动化部署,并发现指南http://www.davedevelopment.co.uk/2008/04/14/how-to-simple-database-migrations-with-phing-and-dbdeploy/并不实用,因为它们不支持像我们拥有的多个数据库。
此外,对于我们来说,可以使用可以执行此操作的Windows或Mac MySQL客户端,因此我们开放接受任何建议。

1
这些数据库都在同一台服务器上吗? - Bobby B
是的,所有数据库都在同一台服务器上! - Jorre
1个回答

14
这是一个我为您编写的PHP脚本。它获取所有数据库列表并在数据库名称以user开头时应用更新。
我还让它在应用更改之前备份每个数据库。备份部分目前仅适用于Linux / Unix,但可以进行调整以适用于其他操作系统。
目前它非常冗长,因此您可以根据需要进行更改。您还可以更改行终止符,具体取决于您将从CLI还是浏览器运行它。我建议将其放在您的脚本目录中,并从CLI运行它。
如果您需要其他帮助,或者如果这不适用于您,请告诉我。
<?php
// Configure these as needed
$db_host = 'localhost';
$db_user = 'user';
$db_pass = 'password';

$datetime_pattern       = date('Ymd.His');
$backup_file_path       = "/path/to/db_backups/$datetime_pattern/";
$backup_file_format     = "db_backup.%s.sql";
$backup_syntax_pattern  = "/usr/bin/mysqldump --host=%s --user=%s --password=%s --opt %s > $backup_file_path/db_backup.%s.sql";
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!
// CHANGE THE PERMISSIONS!!!!!!
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!
$backup_file_permission = 0777;

// Choose how to terminate your lines
$line_end = "\n";      // Use for CLI
//$line_end = "<br/>";   // Use for browser

// Match words that begin with 'user', case-insensitive
$pattern = '/^user/i';

// What changes will we be applying?
$db_update_syntax = array("ALTER TABLE foo ADD baz1 VARCHAR(30) AFTER bar1",
                          "ALTER TABLE foo ADD baz2 VARCHAR(30) AFTER bar2",
                          "ALTER TABLE foo ADD baz3 VARCHAR(30) AFTER bar3",
                         );

// END OF CONFIGURATION
/////////////////////////////////////////////////////////////


// Create the database backup directory
if (!mkdir($backup_file_path, $backup_file_permission, true)) {
    die('Failed to create backup directory...');
}

// Connecting to MySQL.
$conn = @mysql_connect($db_host, $db_user, $db_pass)
        or die('Not connected : ' . mysql_errno() . ': ' . mysql_error());

$db_list = mysql_list_dbs($conn);

echo "{$line_end}Starting Database Update.{$line_end}";
while ($row = mysql_fetch_assoc($db_list)) {
    $db_name = $row['Database'];
    if (preg_match($pattern, $db_name)) {
        echo "{$line_end}A match was found: [$db_name]{$line_end}";
        echo "Backing up the database{$line_end}";
        // Backup the database
        $backup_syntax = sprintf($backup_syntax_pattern, $db_host, $db_user, $db_pass, $db_name, $db_name);
        exec($backup_syntax);
        $db_selected = mysql_select_db($db_name, $conn)
                       or die("Can't use [$db_name] : " . mysql_error());

        foreach ($db_update_syntax as $each_update_syntax) {
            echo "Altering using: [$alter_syntax]{$line_end}";
            $update_status = mysql_query($alter_syntax);
            if ($update_status) {
                echo "Success!{$line_end}{$line_end}";
            } else {
                echo "Unable to update [$db_name] : " . mysql_error() . "{$line_end}{$line_end}";
            }
        }
    } else {
        echo "Ignoring: [$db_name]{$line_end}";
    }
}
echo "Finished!{$line_end}";
// Free resources / Close MySQL Connection
mysql_free_result($db_list);
mysql_close($conn);

我正在查看这个,看看它是否可以满足我们的需求。非常感谢您已经发布了这么大的脚本! - Jorre
你能预见在繁忙的数据库中锁定方面会出现什么问题吗? - Jorre
在没有连接的数据库上运行得很好。你在一个有很多用户的实时数据库上的经验如何? - Jorre
我猜问题取决于交易类型和查询代码的编码方式。根据文档[http://dev.mysql.com/doc/refman/5.1/en/alter-table.html]:`在大多数情况下,ALTER TABLE 会创建原始表的临时副本。MySQL 将更改合并到副本中,然后删除原始表并重命名新表。在 ALTER TABLE 执行时,其他会话可以读取原始表。对表格的更新和写入被暂停,直到新表准备好,然后自动重定向到新表,而不会有任何失败的更新。` - pferate
提醒其他使用者注意。是的,它可以工作,但其中有一些错误。例如:"echo "Altering using: [$alter_syntax]{$line_end}"; $alter_syntax从未设置过。应该是$each_update_syntax。还有一些类似的错误。但如果你无法解决它们,绝对不应该使用脚本来更改你的数据库。是的,我可以编辑原始答案,但我认为如果这些错误能够阻止人们在不理解其工作原理的情况下使用它,那么这些错误并不是坏事。 - Steve

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接