Important
You are browsing upcoming documentation for version 6.1 of OroCommerce, scheduled for release in 2025. Read the documentation for version 6.0 (the latest LTS version) to get up-to-date information.
See our Release Process documentation for more information on the currently supported and upcoming releases.
Database Structure Migrations
Each bundle can have migration files that enable you to update the database schema.
To create a schema (database structure) migration, follow the steps below.
Create Schema Migration
Create Database Dump
It is required to create a database dump before any database changes.
Synchronize Code with the Database
After you have modeled your entities, you need to update the database schema. To update the schema, use the doctrine:schema:update command
. Use the --dump-sql
option first to make sure that Doctrine makes the expected changes:
Double-check the configured mapping information and rerun the command if the command displays unexpected information.
When everything is displayed as expected, update the database schema by passing the --force
option:
php bin/console doctrine:schema:update --force
Tip
Doctrine caches mapping metadata. If the doctrine:schema:update
command does not recognize your changes to the entity mapping, clear the metadata cache manually and update the schema again:
# clear the metadata cache
php bin/console doctrine:cache:clear-metadata
# check the schema change queries to be executed
php bin/console doctrine:schema:update --dump-sql
# apply the schema changes to the database
php bin/console doctrine:schema:update --force
Caution
Do not use the doctrine:schema:update
command with your production database. Instead,
create migrations to update the schema of your database. You can read more about using
migrations in the Update Database Schema section. To run migrations
and emulate the complete migration process, use the oro:platform:update
command.
Generate an Installer
Generate an Installer for a Bundle
When you have implemented new entities, ensure that the entities are added to the database on installing the application. For this, you need to create an installer migration. You can do it manually, however, it is more convenient to use a database dump as a template.
To create an installer for AcmeDemoBundle:
Clear the application cache:
php bin/console cache:clear
Apply the changes that you defined in your code to the database:
php bin/console doctrine:schema:update
Generate an installer and save it to the AcmeDemoBundleInstaller.php:
php bin/console oro:migration:dump --bundle=AcmeDemoBundle
The generated AcmeDemoBundleInstaller.php will be placed into the AcmeDemoBundle/Migrations/Schema directory.
Reinstall your application instance.
Check that the database is synced with your code:
php bin/console doctrine:schema:update --dump-sql
If the database is successfully synchronized, you will see the following message:
Nothing to update - your database is already in sync with the current entity metadata.
Migrations Dump Command
Use the oro:migration:dump command to help create installation files. This command outputs the current database structure as plain SQL or as Doctrine\DBAL\Schema\Schema
queries.
This command supports the following additional options:
plain-sql - Outputs schema as plain SQL queries
bundle - The bundle name for which the migration is generated
migration-version - Migration version number. This option sets the value returned by the getMigrationVersion method of the generated installation file.
Each bundle can have an installation file. This migration file replaces running multiple migration files. Install migration class must implement the Installation interface and the up and getMigrationVersion methods. The getMigrationVersion method must return the max migration version number that this installation file replaces.
When an install migration file is found during the install process (when you install the system from scratch), it is loaded first, followed by the migration files with versions greater than the version returned by the getMigrationVersion method.
For example, let’s assume we have migrations v1_0, v1_1, v1_2, v1_3 and installed the migration class. This class returns v1_2 as the migration version. That is why, during the installation process, the install migration file is loaded first, followed only by the migration file v1_3. In this case, migrations from v1_0 to v1_2 are not loaded.
Below is an example of an install migration file:
namespace Acme\Bundle\DemoBundle\Migrations\Schema;
use Doctrine\DBAL\Schema\Schema;
use Oro\Bundle\MigrationBundle\Migration\Installation;
use Oro\Bundle\MigrationBundle\Migration\QueryBag;
/**
* Creates all tables required for the bundle.
*/
class AcmeDemoBundleInstaller implements
Installation
{
public function getMigrationVersion()
{
return 'v1_0';
}
public function up(Schema $schema, QueryBag $queries)
{
/** Tables generation **/
$this->createAcmeDemoPriorityTable($schema);
$this->createAcmeDemoDocumentTable($schema);
/** Foreign keys generation **/
$this->addAcmeDemoPriorityForeignKeys($schema);
$this->addAcmeDemoDocumentForeignKeys($schema);
$this->addAcmeDemoNotManageableEntity($schema);
}
private function createAcmeDemoPriorityTable(Schema $schema): void
{
$table = $schema->createTable('acme_demo_priority');
$table->addColumn('id', 'integer', ['autoincrement' => true]);
$table->addColumn('organization_id', 'integer', ['notnull' => false]);
$table->setPrimaryKey(['id']);
$table->addUniqueIndex(['label'], 'uidx_label_doc');
}
private function createAcmeDemoDocumentTable(Schema $schema): void
{
$table = $schema->createTable('acme_demo_document');
$table->addColumn('id', 'integer', ['autoincrement' => true]);
$table->addColumn('subject', 'string', ['length' => 255]);
$table->addColumn('description', 'string', ['length' => 255]);
$table->addColumn('organization_id', 'integer', ['notnull' => false]);
$table->addColumn('priority_id', 'integer', ['notnull' => false]);
$table->setPrimaryKey(['id']);
}
private function addAcmeDemoDocumentForeignKeys(Schema $schema): void
{
$table = $schema->getTable('acme_demo_document');
$table->addForeignKeyConstraint(
$schema->getTable('acme_demo_priority'),
['priority_id'],
['id'],
['onUpdate' => null, 'onDelete' => 'SET NULL']
);
}
Create Versioned Schema Migrations
A good practice is for a bundle to have the installation file for the current version and migration files for migrating from the previous to the current version.
Migration files should be located in the Migrations\Schema\version_number
folder. A version number must be a PHP-standardized version number string but with some limitations. This string must not contain “.” and “+” characters as a version parts separator. You can find more information about PHP-standardized version number string in the PHP manual.
Each migration class must implement the Migration interface and the up method. This method receives a current database structure in the schema and queries parameters, adding additional queries.
With the schema parameter, you can create or update the database structure without fear of compatibility between database engines. You can use the’ queries’ parameter if you want to execute additional SQL queries before or after applying a schema modification. This parameter represents a query bag and allows adding additional queries, which will be executed before (addPreQuery method) or after (addQuery or addPostQuery method). A query can be a string or an instance of a class that implements MigrationQuery interface. There are several ready-to-use implementations of this interface:
SqlMigrationQuery - represents one or more SQL queries
ParametrizedSqlMigrationQuery - similar to the previous class, but each query can have its own parameters.
If you need to create your own implementation of the MigrationQuery, implement ConnectionAwareInterface in your migration query class if you need a database connection. You can also use the ParametrizedMigrationQuery class as the base class for your migration query.
If you have several migration classes within the same version and you need to make sure that they are executed in a specific order, use OrderedMigrationInterface.
Below is an example of a migration file:
namespace Acme\Bundle\DemoBundle\Migrations\Schema\v1_1;
use Doctrine\DBAL\Schema\Schema;
use Oro\Bundle\MigrationBundle\Migration\Migration;
use Oro\Bundle\MigrationBundle\Migration\QueryBag;
class AddTmpTestTable implements Migration
{
/**
* @inheritDoc
*/
public function up(Schema $schema, QueryBag $queries)
{
$table = $schema->createTable('tmp_test_table');
$table->addColumn('id', 'integer', ['autoincrement' => true]);
$table->addColumn('created', 'datetime', []);
$table->addColumn('field', 'string', ['length' => 500]);
$table->addColumn('another_field', 'string', ['length' => 255]);
$table->addColumn('test_column', 'json', []);
$table->setPrimaryKey(['id']);
}
}
Restore the Database Dump
In case of problems or a database crash, you can restore the database dump.
Load Schema Migrations
To run migrations, use the oro:migration:load command. This command collects migration files from bundles, sorts them by their version number, and applies changes.
This command supports the following additional options:
force - Causes the generated by migrations SQL statements to be physically executed against your database;
dry-run - Outputs list of migrations without applying them;
show-queries - Outputs list of database queries for each migration file;
bundles - A list of bundles from which to load data. If the option is not set, migrations are taken from all bundles;
exclude - A list of bundle names where migrations should be skipped.
Examples of Database Structure Migrations
Extensions for Database Structure Migrations
You cannot always use standard Doctrine methods to modify the database structure. For example, Schema::renameTable
does not work because it drops an existing table and then creates a new one. To help you manage such a case and enable you to add additional functionality to any migration, use the extensions mechanism. The following example illustrates how RenameExtension can be used:
namespace Acme\Bundle\DemoBundle\Migrations\Schema\v1_2;
use Doctrine\DBAL\Schema\Schema;
use Oro\Bundle\MigrationBundle\Migration\Migration;
use Oro\Bundle\MigrationBundle\Migration\QueryBag;
use Oro\Bundle\MigrationBundle\Migration\Extension\RenameExtensionAwareInterface;
use Oro\Bundle\MigrationBundle\Migration\Extension\RenameExtensionAwareTrait;
class TestRenameTable implements Migration, RenameExtensionAwareInterface
{
use RenameExtensionAwareTrait;
public function up(Schema $schema, QueryBag $queries)
{
$this->renameExtension->renameTable(
$schema,
$queries,
'tmp_test_table',
'new_test_table'
);
}
}
As you can see from the example above, your migration class should implement RenameExtensionAwareInterface and setRenameExtension method in order to use the RenameExtension.
Another example below illustrates how to use database-specific features in migration:
namespace Acme\Bundle\DemoBundle\Migrations\Schema\v1_3;
use Doctrine\DBAL\Platforms\PostgreSQL94Platform;
use Doctrine\DBAL\Schema\Schema;
use Oro\Bundle\MigrationBundle\Migration\Extension\DatabasePlatformAwareInterface;
use Oro\Bundle\MigrationBundle\Migration\Extension\DatabasePlatformAwareTrait;
use Oro\Bundle\MigrationBundle\Migration\Migration;
use Oro\Bundle\MigrationBundle\Migration\QueryBag;
use Oro\Bundle\MigrationBundle\Migration\SqlMigrationQuery;
class CreateFunctionalIndex implements Migration, DatabasePlatformAwareInterface
{
use DatabasePlatformAwareTrait;
public function up(Schema $schema, QueryBag $queries)
{
if ($this->platform instanceof PostgreSQL94Platform) {
$query = new SqlMigrationQuery(
"CREATE INDEX test_idx1 ON new_test_table (LOWER(test_column->>'test_key'))"
);
} else {
$query = new SqlMigrationQuery(
"CREATE INDEX test_idx1 ON new_test_table ((LOWER(JSON_VALUE(test_column, '\$.test_key'))))"
);
}
$queries->addPostQuery($query);
}
}
You can also use the following additional interfaces in your migration class:
ContainerAwareInterface - provides access to Symfony dependency container.
DatabasePlatformAwareInterface - allows to write database type independent migrations.
ConnectionAwareInterface - provides access to the database connection.
NameGeneratorAwareInterface - provides access to the DbIdentifierNameGenerator class used to generate names of indices, foreign key constraints, etc.
Here is a list of available extensions:
Commerce
PaymentTermExtension - Adds payment term association to the entity.
SlugExtension - Adds slugs to the entity. More information is available in the RedirectBundle documentation.
CustomerExtension - Adds association between the target customer table and the customer table. More information is available in the Migration Extension documentation.
Platform
ActivityExtension - Adds association between the given table and the table that contains activity records.
ActivityListExtension - Adds association between the given table and the activity list table. See an example of usage in the Activity List Inheritance Targets documentation.
AttachmentExtension - Provides an ability to create file and attachment fields and attachment associations. More information is available in the Use Migration Extension Example in AttachmentBundle.
CommentExtension - Adds comments association to the entity. More information is available in Enable Comment Association with New Activity Entity.
AuditFieldExtension - Add a possibility for developers to extend data types for DataAudit. More information is available in the Add New Auditable Types topic.
ChangeTypeExtension - Allows to change the type of entity primary column type.
ExtendExtension - Provides the ability to create extended enum tables and fields and add relations between tables. More information is available in the Create Custom Entities topic.
ConvertToExtendExtension - Allows to convert existing entity field to extended.
RenameExtension - Allows to rename an extended table or an extended column without losing data.
DataStorageExtension- Used ito exchange data between different migrations.
ScopeExtension - Adds association between the target table and the scope table.
SerializedFieldsExtension - The migration extension that helps manage serialized fields of extended entities. More information is available in the Serialized Fields topic.
Create Extensions for Database Structure Migrations
To create your own extension:
Create an extension class in the
YourBundle/Migration/Extension
directory. Using theYourBundle/Migration/Extension
directory is not mandatory but highly recommended. For example:namespace Acme\Bundle\DemoBundle\Migration\Extension; use Doctrine\DBAL\Schema\Schema; use Oro\Bundle\MigrationBundle\Migration\QueryBag; class MyExtension { public function doSomething(Schema $schema, QueryBag $queries, /* other parameters, for example */ $tableName) { $table = $schema->getTable($tableName); // highly recommended to make sure that a table exists $query = 'SOME SQL'; /* or $query = new SqlMigrationQuery('SOME SQL'); */ $queries->addQuery($query); } }
Create *AwareInterface in the same namespace. It is important that the interface name is
{ExtensionClass}AwareInterface
and the set method isset{ExtensionClass}({ExtensionClass} ${extensionName})
. For example:namespace Acme\Bundle\DemoBundle\Migration\Extension; /** * This interface should be implemented by migrations that depend on {@see MyExtension}. */ interface MyExtensionAwareInterface { public function setMyExtension(MyExtension $myExtension): void; }
Register an extension in the dependency container. For example:
services: Acme\Bundle\DemoBundle\Migration\Extension\MyExtension: tags: - { name: oro_migration.extension, extension_name: test /*, priority: -10 - priority attribute is optional and can be helpful if you need to override existing extension */ }
To access the database platform or the name generator, your extension class should implement DatabasePlatformAwareInterface or NameGeneratorAwareInterface appropriately.
To use another extension in your extension, the extension class should implement *AwareInterface
of the extension you need.
Events During Migration
The Oro\Bundle\MigrationBundle\Migration\Loader\MigrationsLoader
dispatches two events when migrations are being executed, oro_migration.pre_up and oro_migration.post_up. You can listen to either event and register your own migrations in your event listener. Use the Oro\Bundle\MigrationBundle\Event\MigrationEvent::addMigration
method of the passed event instance to register your custom migrations:
namespace Acme\Bundle\DemoBundle\EventListener;
use Acme\Bundle\DemoBundle\Migration\CustomMigration;
use Oro\Bundle\MigrationBundle\Event\PostMigrationEvent;
use Oro\Bundle\MigrationBundle\Event\PreMigrationEvent;
class RegisterCustomMigrationListener
{
/**
* Listening to the oro_migration.pre_up event
*
* @param PreMigrationEvent $event
* @return void
*/
public function onPreUp(PreMigrationEvent $event): void
{
$event->addMigration(new CustomMigration());
}
/**
* Listening to the oro_migration.post_up event
*
* @param PostMigrationEvent $event
* @return void
*/
public function onPostUp(PostMigrationEvent $event): void
{
$event->addMigration(new CustomMigration());
}
}
Tip
You can learn more about custom event listeners in the Symfony documentation.
Migrations registered in the oro_migration.pre_up event are executed before the main migrations, while migrations registered in the oro_migration.post_up event are executed after the main migrations have been processed.
Business Tip
Digital transformation has been fueled by technologies in core industries such as manufacturing. Read more about the role of eCommerce in the technological age of manufacturing.