数据库迁移
迁移是一种以结构化和有序的方式修改数据库的便捷方法。你可以手工编辑 SQL 片段,但这样你就需要告知其他开发者他们需要运行这些片段。你也需要在下次部署到生产环境时跟踪哪些更改需要运行。
数据库表 迁移 用于跟踪已经运行的迁移,因此你只需确保你的迁移文件已经准备好,并运行 spark migrate
命令将数据库更新到最新状态。你还可以使用 spark migrate --all
命令来包括所有命名空间的迁移。
迁移文件命名
迁移文件名由时间戳前缀、下划线(_
)和描述性名称(类名)组成。
2024-09-08-013653_AddBlogTable.php
每个迁移都使用创建迁移时的时间戳(2024-09-08-013653)进行编号,格式为 YYYY-MM-DD-HHIISS。
迁移的描述性名称(AddBlogTable)是 PHP 中的类名,因此你必须为其命名一个有效的类名。
前缀中的年、月、日和时间的分隔符可以是短横线(-
)、下划线(_
)或者不使用任何分隔符。例如:
2012-10-31-100538_AlterBlogTrackViews.php
2012_10_31_100539_AlterBlogAddTranslations.php
20121031100537_AddBlog.php
每个迁移根据所采取的方法按数字顺序向前或向后运行。这有助于在团队协作环境中避免编号冲突。
创建迁移
这将是为一个带博客的新网站创建的第一个迁移。所有迁移都在 app/Database/Migrations/ 目录下,文件名类似 2022-01-31-013057_AddBlog.php。
<?php
namespace App\Database\Migrations;
use CodeIgniter\Database\Migration;
class AddBlog extends Migration
{
public function up()
{
$this->forge->addField([
'blog_id' => [
'type' => 'INT',
'constraint' => 5,
'unsigned' => true,
'auto_increment' => true,
],
'blog_title' => [
'type' => 'VARCHAR',
'constraint' => '100',
],
'blog_description' => [
'type' => 'TEXT',
'null' => true,
],
]);
$this->forge->addKey('blog_id', true);
$this->forge->createTable('blog');
}
public function down()
{
$this->forge->dropTable('blog');
}
}
数据库连接和数据库 Forge 类都可以通过 $this->db
和 $this->forge
获取。
或者,你可以使用命令行调用来生成一个骨架迁移文件。更多详情请参阅 命令行工具 中的 make:migration。
备注
由于迁移类是一个 PHP 类,每个迁移文件中的类名必须是唯一的。
外键
当你的表包含外键时,迁移经常在尝试删除表和列时会遇到问题。要在运行迁移时暂时绕过外键检查,可以在数据库连接上使用 disableForeignKeyChecks()
和 enableForeignKeyChecks()
方法。
<?php
namespace App\Database\Migrations;
use CodeIgniter\Database\Migration;
class AddBlog extends Migration
{
public function up()
{
$this->db->disableForeignKeyChecks();
// Migration rules would go here..
$this->db->enableForeignKeyChecks();
}
}
数据库组
一个迁移只会对单个数据库组执行。如果你在 app/Config/Database.php 中定义了多个组,那么它将按照该配置文件中指定的 $defaultGroup
运行。
有时你可能需要为不同的数据库组使用不同的模式。也许你有一个数据库用于所有常规站点信息,而另一个数据库用于业务关键的数据。
你可以通过在迁移上设置 $DBGroup
属性来确保迁移只针对适当的组运行。此名称必须与数据库组的名称完全匹配:
<?php
namespace App\Database\Migrations;
use CodeIgniter\Database\Migration;
class AddBlog extends Migration
{
protected $DBGroup = 'alternate_db_group';
public function up()
{
// ...
}
public function down()
{
// ...
}
}
备注
跟踪已经运行过的迁移的 migrations 表将始终在默认数据库组中创建。
命名空间
迁移库可以自动扫描你在 app/Config/Autoload.php 中定义的所有命名空间,或者从 Composer 等外部源加载的命名空间,使用 $psr4
属性匹配目录名称。它将包含在 Database/Migrations 中找到的所有迁移。
每个命名空间都有自己的版本序列,这将帮助你升级和降级每个模块(命名空间)而不影响其他命名空间。
例如,假设我们在 Autoload 配置文件中定义了以下命名空间:
$psr4 = [
APP_NAMESPACE => APPPATH,
'MyCompany' => ROOTPATH . 'MyCompany',
];
这将查找 APPPATH/Database/Migrations 和 ROOTPATH/MyCompany/Database/Migrations 中的任何迁移。这使得在你的可重用、模块化代码套件中包含迁移变得很简单。
命令行工具
CodeIgniter 自带了几个 commands,可通过命令行访问,以帮助你使用迁移。这些工具使得使用迁移更加方便。这些工具主要提供了 MigrationRunner 类中可用的相同方法的访问。
migrate
使用所有可用的迁移迁移一个数据库组:
php spark migrate
你可以对 migrate
使用以下选项:
-g
- 用于指定数据库组。如果指定了该选项,只会运行指定数据库组的迁移。如果未指定,则会运行所有迁移。-n
- 用于选择命名空间,否则将使用App
命名空间。--all
- 迁移所有命名空间到最新的迁移。
这个例子将在 test 数据库组上使用任何新的迁移迁移 Acme\Blog
命名空间:
For Unix:
php spark migrate -g test -n Acme\\Blog
For Windows:
php spark migrate -g test -n Acme\Blog
当使用 --all
选项时,它将扫描所有命名空间,尝试找到任何未运行的迁移。这些迁移将一起收集,然后按创建日期排序为一组。这应该有助于最大限度地减少主应用程序和任何模块之间的潜在冲突。
migrate:rollback
回滚所有迁移到空白状态,有效迁移到 0:
php spark migrate:rollback
你可以对 migrate:rollback
使用以下选项:
-b
- 选择批次:自然数指定批次。-f
- 强制绕过确认问题,它仅在生产环境中询问。
migrate:refresh
首先回滚所有迁移,然后迁移所有来刷新数据库状态:
php spark migrate:refresh
你可以对 migrate:refresh
使用以下选项:
-g
- 用于指定数据库组。如果指定了该选项,只会运行指定数据库组的迁移。如果未指定,则会运行所有迁移。-n
- 用于选择命名空间,否则将使用App
命名空间。--all
- 刷新所有命名空间。-f
- 强制绕过确认问题,它仅在生产环境中询问。
migrate:status
显示所有迁移的列表以及它们运行的日期和时间,如果未运行则显示 ‘–‘:
php spark migrate:status
...
+----------------------+-------------------+-----------------------+---------+---------------------+-------+
| Namespace | Version | Filename | Group | Migrated On | Batch |
+----------------------+-------------------+-----------------------+---------+---------------------+-------+
| App | 2022-04-06-234508 | CreateCiSessionsTable | default | 2022-04-06 18:45:14 | 2 |
| CodeIgniter\Settings | 2021-07-04-041948 | CreateSettingsTable | default | 2022-04-06 01:23:08 | 1 |
| CodeIgniter\Settings | 2021-11-14-143905 | AddContextColumn | default | 2022-04-06 01:23:08 | 1 |
+----------------------+-------------------+-----------------------+---------+---------------------+-------+
你可以对 migrate:status
使用以下选项:
-g
- 用于指定数据库组。如果指定了该选项,只会检查指定数据库组的迁移。如果未指定,则会检查所有迁移。
make:migration
在 app/Database/Migrations 中创建一个骨架迁移文件。它会自动在文件名前加上当前时间戳。它创建的类名是文件名的大驼峰版本。
php spark make:migration <class> [options]
你可以对 make:migration
使用以下选项:
--namespace
- 设置根命名空间。默认:APP_NAMESPACE
。--suffix
- 在类名后追加组件标题。
以下选项也可用于为数据库 Sessions 生成迁移文件:
--session
- 为数据库 sessions 生成迁移文件。--table
- 数据库 sessions 使用的表名。默认:ci_sessions
。--dbgroup
- 数据库 sessions 使用的数据库组。默认:default
。
迁移配置
下表列出了所有迁移的配置选项,在 app/Config/Migrations.php 中可用。
首选项 |
默认值 |
可选值 |
描述 |
---|---|---|---|
enabled |
true |
true / false |
启用或禁用迁移。 |
table |
migrations |
None |
用于存储 schema 版本号的表名。该表始终在默认数据库组( |
timestampFormat |
Y-m-d-His_ |
创建迁移时使用的时间戳格式。 |
类参考
- class CodeIgniter\Database\MigrationRunner
- findMigrations()
- 返回:
迁移文件数组
- 返回类型:
array
返回在
path
属性中找到的迁移文件名数组。
- latest($group)
- 参数:
$group (
mixed
) – 数据库组名称,如果为 null 则使用默认数据库组。
- 返回:
成功则为
true
,失败则为false
- 返回类型:
bool
该方法定位命名空间(或所有命名空间)的迁移,确定哪些迁移尚未运行,并按版本顺序运行它们(命名空间交错)。
- regress($targetBatch, $group)
- 参数:
$targetBatch (
int
) – 要迁移到的前一批次; 1+ 指定批次,0 还原全部,负数指相对批次(例如 -3 表示“往前三批”)$group (
?string
) – 数据库组名称,如果为 null 则使用默认数据库组。
- 返回:
成功则为
true
,失败或找不到迁移则为false
- 返回类型:
bool
回滚可用于将更改回滚到以前的状态,逐批进行。
<?php $migration->regress(5); $migration->regress(-1);
- force($path, $namespace, $group)
- 参数:
$path (
mixed
) – 有效迁移文件的路径。$namespace (
mixed
) – 所提供迁移的命名空间。$group (
mixed
) – 数据库组名称,如果为 null 则使用默认数据库组。
- 返回:
成功则为
true
,失败则为false
- 返回类型:
bool
该方法强制单文件迁移,不考虑顺序或批次。基于它是否已经迁移来检测
up()
或down()
方法。备注
该方法仅建议用于测试,可能会导致数据一致性问题。
- setNamespace($namespace)
- 参数:
$namespace (
string|null
) – 应用程序命名空间。null
为所有命名空间。
- 返回:
当前的 MigrationRunner 实例
- 返回类型:
设置库应查找迁移文件的命名空间:
<?php $migration->setNamespace($namespace)->latest();
备注
如果设置为
null
,则它将查找所有命名空间中的迁移文件。
- setGroup($group)
- 参数:
$group (
string
) – 数据库组名称。
- 返回:
当前的 MigrationRunner 实例
- 返回类型:
设置库应查找迁移文件的组:
<?php $migration->setGroup($group)->latest();