Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

doctrine:schema:validate returns DROP TABLE doctrine_migration_versions; #1406

Closed
ValentinDk opened this issue Feb 22, 2024 · 76 comments
Closed

Comments

@ValentinDk
Copy link

ValentinDk commented Feb 22, 2024

BC Break Report

Q A
BC Break yes
Version 3.0.0

Summary

When I updated doctrine/orm to version 3.0.0, my CI started to fail due to the doctrine:schema check:validate, which returned DROP TABLE doctrine_migration_versions;

I found that adding schema_filter can fix this problem: "^(?!doctrine_migration_versions$)" in the dbal configuration, indeed, then the error no longer appears, but after that my migrations stopped tracking the current state and think that at the moment no migration has been performed.

In version 3.0, the saveMode argument was removed in \Doctrine\ORM\Tools\SchemaTool::getUpdateSchemaSql and because of this, doctrine:schema:validate tracks deleted tables and returns sql for this action.
telegram-cloud-photo-size-2-5384566650814190356-y

Previous behavior

image

Current behavior

image

How to reproduce

  1. composer require doctrine/doctrine-migrations-bundle
  2. composer require doctrine/orm-pack
  3. bin/console doctrine:migration:diff
  4. bin/console doctrine:migration:migrate
  5. bin/console doctrine:schema:validate -v
@dmamchyts
Copy link

+1:

 // 1 schema diff(s) detected:                                                                                          

     DROP TABLE doctrine_migration_versions;

@derrabus derrabus transferred this issue from doctrine/orm Mar 4, 2024
@derrabus derrabus changed the title [3.0.0] doctrine:schema:validate returns DROP TABLE doctrine_migration_versions; doctrine:schema:validate returns DROP TABLE doctrine_migration_versions; Mar 4, 2024
@greg0ire greg0ire pinned this issue Mar 11, 2024
@greg0ire
Copy link
Member

Somebody posted a workaround for Symfony applications relying on the Doctrine bundle:

doctrine:
    dbal:
        schema_filter: ~^(?!doctrine_)~

Can people here please test?

@dmamchyts
Copy link

@greg0ire

[OK] The database schema is in sync with the mapping files.                    

doctrine:migrations:diff also work fine.

image

@greg0ire
Copy link
Member

Possible next steps:

  • determine how this works with DBAL 3
  • determine why it no longer works with DBAL 4

@wmouwen
Copy link

wmouwen commented Mar 12, 2024

The bug exists on ORM 3.1 with DBAL 3.8 as well, it is not per se a DBAL 4.0 issue.

@greg0ire
Copy link
Member

greg0ire commented Mar 12, 2024

OK so possible next steps:

  • determine how this works with ORM 2
  • determine why it no longer works with ORM 3

If somebody could dump their DBAL schema filter at runtime using the DBAL configuration object on ORM 2 then on ORM 3, it might help.

@greg0ire
Copy link
Member

greg0ire commented Mar 13, 2024

Can somebody here please confirm that they get the issue with ORM 2 when using the --complete flag? I just remembered that this is likely the cause for this bug.

@dmamchyts
Copy link

/var/www/html $ composer show | grep doctrine
doctrine/cache                        2.2.0
doctrine/collections                  2.2.1
doctrine/common                       3.4.3
doctrine/data-fixtures                1.7.0
doctrine/dbal                         3.8.3
doctrine/deprecations                 1.1.3
doctrine/doctrine-bundle              2.11.3
doctrine/doctrine-fixtures-bundle     3.5.1
doctrine/doctrine-migrations-bundle   3.3.0
doctrine/event-manager                2.0.0
doctrine/inflector                    2.0.10
doctrine/instantiator                 2.0.0
doctrine/lexer                        3.0.1
doctrine/migrations                   3.7.4
doctrine/orm                          2.19.0
doctrine/persistence                  3.3.2
doctrine/sql-formatter                1.2.0
phpstan/phpstan-doctrine              1.3.62
symfony/doctrine-bridge               v7.0.5



/var/www/html $ bin/console doctrine:migration:diff --help

Description:
  Generate a migration by comparing your current database to your mapping information.

Usage:
  doctrine:migrations:diff [options]

Options:
      --configuration=CONFIGURATION                        The path to a migrations configuration file. [default: any of migrations.{php,xml,json,yml,yaml}]
      --em=EM                                              The name of the entity manager to use.
      --conn=CONN                                          The name of the connection to use.
      --namespace=NAMESPACE                                The namespace to use for the migration (must be in the list of configured namespaces)
      --filter-expression=FILTER-EXPRESSION                Tables which are filtered by Regular Expression.
      --formatted                                          Format the generated SQL.
      --line-length=LINE-LENGTH                            Max line length of unformatted lines. [default: "120"]
      --check-database-platform[=CHECK-DATABASE-PLATFORM]  Check Database Platform to the generated code. [default: false]
      --allow-empty-diff                                   Do not throw an exception when no changes are detected.
      --from-empty-schema                                  Generate a full migration as if the current database was empty.
  -h, --help                                               Display help for the given command. When no command is given display help for the list command
  -q, --quiet                                              Do not output any message
  -V, --version                                            Display this application version
      --ansi|--no-ansi                                     Force (or disable --no-ansi) ANSI output
  -n, --no-interaction                                     Do not ask any interactive question
  -e, --env=ENV                                            The Environment name. [default: "dev"]
      --no-debug                                           Switch off debug mode.
      --profile                                            Enables profiling (requires debug).
  -v|vv|vvv, --verbose                                     Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug






/var/www/html $ bin/console doctrine:schema:validate --help

Description:
  Validate the mapping files

Usage:
  doctrine:schema:validate [options]

Options:
      --em=EM                Name of the entity manager to operate on
      --skip-mapping         Skip the mapping validation check
      --skip-sync            Skip checking if the mapping is in sync with the database
      --skip-property-types  Skip checking if property types match the Doctrine types
  -h, --help                 Display help for the given command. When no command is given display help for the list command
  -q, --quiet                Do not output any message
  -V, --version              Display this application version
      --ansi|--no-ansi       Force (or disable --no-ansi) ANSI output
  -n, --no-interaction       Do not ask any interactive question
  -e, --env=ENV              The Environment name. [default: "dev"]
      --no-debug             Switch off debug mode.
      --profile              Enables profiling (requires debug).
  -v|vv|vvv, --verbose       Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug

No --complete flags

@greg0ire
Copy link
Member

--complete is a flag on commands of the doctrine namespace, which are the commands this issue is about, right?

@yapro
Copy link

yapro commented Mar 13, 2024

This is a real problem for our project, especially for production.
And this trick doesn't work:

schema_filter: ~^(?!doctrine_)~

because it causes an even bigger problem:

bin/console doctrine:migrations:migrate --no-interaction

In ExceptionConverter.php line 45:
                                                                                                                                                              
  An exception occurred while executing a query: SQLSTATE[42S01]: Base table or view already exists: 1050 Table 'doctrine_migration_versions' already exists  
                                                                                                                                                              

In Exception.php line 28:
                                                                                                               
  SQLSTATE[42S01]: Base table or view already exists: 1050 Table 'doctrine_migration_versions' already exists  
                                                                                                               

In Connection.php line 33:
                                                                                                               
  SQLSTATE[42S01]: Base table or view already exists: 1050 Table 'doctrine_migration_versions' already exists  
                                                                                                               

doctrine:migrations:migrate [--write-sql [WRITE-SQL]] [--dry-run] [--query-time] [--allow-no-migration] [--all-or-nothing [ALL-OR-NOTHING]] [--configuration CONFIGURATION] [--em EM] [--conn CONN] [--] [<version>]

@wmouwen
Copy link

wmouwen commented Mar 13, 2024

Here is a shell script to trigger both a succeeding workflow with ORM 2.19 and a failing one with ORM 3.0.

Script
#!/usr/bin/env sh

# Set up new Symfony 7 project
composer create-project symfony/skeleton:"7.0.*" my_test_project
cd my_test_project

# Install Doctrine bundles
composer require --no-interaction "doctrine/doctrine-bundle:~2.11" "doctrine/doctrine-migrations-bundle:~3.3" "doctrine/orm:~2.19" "doctrine/dbal:~3.8"

# Use sqlite for local testing
echo "DATABASE_URL=\"sqlite:///%kernel.project_dir%/var/data.db\"" > .env.local

# Install empty migration. This creates the sqlite database with the doctrine_migration_versions table
bin/console doctrine:migrations:generate
bin/console doctrine:migrations:migrate

# Validate schema: succeeds
bin/console doctrine:schema:validate -v

# Upgrade to ORM 3.*
composer require "doctrine/orm:~3.0"

# Validate schema: fails
bin/console doctrine:schema:validate -v

The difference causing the change is that within the callstack of the command, the variable $saveMode in \Doctrine\DBAL\Schema\SchemaDiff::_toSql (link) is true when using ORM 2.19 and false when using ORM 3.0, causing dropped tables to be included in the changeset.

Backtrace ORM 2.19
[generate trace] at vendor/doctrine/dbal/src/Schema/SchemaDiff.php:252
 Doctrine\DBAL\Schema\SchemaDiff->_toSql() at vendor/doctrine/dbal/src/Schema/SchemaDiff.php:225
 Doctrine\DBAL\Schema\SchemaDiff->toSaveSql() at vendor/doctrine/orm/src/Tools/SchemaTool.php:992
 Doctrine\ORM\Tools\SchemaTool->getUpdateSchemaSql() at vendor/doctrine/orm/src/Tools/SchemaValidator.php:358
 Doctrine\ORM\Tools\SchemaValidator->getUpdateSchemaList() at vendor/doctrine/orm/src/Tools/SchemaValidator.php:344
 Doctrine\ORM\Tools\SchemaValidator->schemaInSyncWithMetadata() at vendor/doctrine/orm/src/Tools/Console/Command/ValidateSchemaCommand.php:75
 Doctrine\ORM\Tools\Console\Command\ValidateSchemaCommand->doExecute() at vendor/doctrine/orm/src/Tools/Console/CommandCompatibility.php:18
 Doctrine\ORM\Tools\Console\Command\ValidateSchemaCommand->execute() at vendor/symfony/console/Command/Command.php:279
 Symfony\Component\Console\Command\Command->run() at vendor/symfony/console/Application.php:1049
 Symfony\Component\Console\Application->doRunCommand() at vendor/symfony/framework-bundle/Console/Application.php:125
 Symfony\Bundle\FrameworkBundle\Console\Application->doRunCommand() at vendor/symfony/console/Application.php:318
 Symfony\Component\Console\Application->doRun() at vendor/symfony/framework-bundle/Console/Application.php:79
 Symfony\Bundle\FrameworkBundle\Console\Application->doRun() at vendor/symfony/console/Application.php:169
 Symfony\Component\Console\Application->run() at vendor/symfony/runtime/Runner/Symfony/ConsoleApplicationRunner.php:49
 Symfony\Component\Runtime\Runner\Symfony\ConsoleApplicationRunner->run() at vendor/autoload_runtime.php:29
 require_once() at bin/console:11
Backtrace ORM 3.0
[generate trace] at vendor/doctrine/dbal/src/Schema/SchemaDiff.php:252
 Doctrine\DBAL\Schema\SchemaDiff->_toSql() at vendor/doctrine/dbal/src/Schema/SchemaDiff.php:242
 Doctrine\DBAL\Schema\SchemaDiff->toSql() at vendor/doctrine/dbal/src/Platforms/AbstractPlatform.php:2391
 Doctrine\DBAL\Platforms\AbstractPlatform->getAlterSchemaSQL() at vendor/doctrine/orm/src/Tools/SchemaTool.php:900
 Doctrine\ORM\Tools\SchemaTool->getUpdateSchemaSql() at vendor/doctrine/orm/src/Tools/SchemaValidator.php:322
 Doctrine\ORM\Tools\SchemaValidator->getUpdateSchemaList() at vendor/doctrine/orm/src/Tools/SchemaValidator.php:308
 Doctrine\ORM\Tools\SchemaValidator->schemaInSyncWithMetadata() at vendor/doctrine/orm/src/Tools/Console/Command/ValidateSchemaCommand.php:71
 Doctrine\ORM\Tools\Console\Command\ValidateSchemaCommand->execute() at vendor/symfony/console/Command/Command.php:279
 Symfony\Component\Console\Command\Command->run() at vendor/symfony/console/Application.php:1049
 Symfony\Component\Console\Application->doRunCommand() at vendor/symfony/framework-bundle/Console/Application.php:125
 Symfony\Bundle\FrameworkBundle\Console\Application->doRunCommand() at vendor/symfony/console/Application.php:318
 Symfony\Component\Console\Application->doRun() at vendor/symfony/framework-bundle/Console/Application.php:79
 Symfony\Bundle\FrameworkBundle\Console\Application->doRun() at vendor/symfony/console/Application.php:169
 Symfony\Component\Console\Application->run() at vendor/symfony/runtime/Runner/Symfony/ConsoleApplicationRunner.php:49
 Symfony\Component\Runtime\Runner\Symfony\ConsoleApplicationRunner->run() at vendor/autoload_runtime.php:29
 require_once() at bin/console:11

@greg0ire
Copy link
Member

Related PRs:

Instead of using SchemaDiff::toSaveSql() for discarding schema changes in 3rd-party database objects (if that's the case), it is recommended to use asset filtering.

I think we should leverage asset filtering to do this. If it causes issues with command from doctrine/migrations, then that needs to be investigated (maybe the filter needs to be dynamic).

@greg0ire
Copy link
Member

greg0ire commented Mar 13, 2024

Just remembered this solution that does exactly that (using a dynamic filter).

Something better that might benefit not just Symfony users might be to do it directly from code inside doctrine/migrations, at runtime.

@gjuric
Copy link

gjuric commented Mar 14, 2024

Proposed schema_filter solution does not work because it hides the doctrine_migrations_versions table from the rest of the code that needs to see it.

If you run for example doctrine:migrations:list you can see that Doctrine thinks that none of the migrations have been executed because it believes doctrine_migrations_versions table does not exist.

@greg0ire
Copy link
Member

@gjuric yes, @yapro already established that in #1406 (comment)

Can you try the solution in my last command? I'm currently working on integrated the solution from my previous comment in the migrations bundle.

@greg0ire
Copy link
Member

Here is my attempt: doctrine/DoctrineMigrationsBundle#526

@Brewal
Copy link

Brewal commented Mar 15, 2024

For now, I manually add the migration table to the schema with a doctrine event listener on postGenerateSchema event and don't use schema_filter.

Here's a gist if anyone interested : https://gist.github.com/Brewal/d0fe0792a69e7e5fdf3fb06898b20d35

@stof
Copy link
Member

stof commented Mar 15, 2024

Adding a listener for postGenerateSchema to register the migration table in the schema is an interesting approach. Maybe doctrine/migrations should ship with such a listener.

@greg0ire
Copy link
Member

If it relies solely on the ORM's event system and not on Symfony or the DoctrineBundle, I can see how it would be superior to the asset filtering solution, in that it could benefit to other frameworks such as Laminas or Laravel as well.

@Brewal @stof are you willing to open a pull request? Otherwise I might work on it soon.

If I understand properly, the ORM command works by diffing 2 schemas.
@FlyingDR's solution works by making sure the migrations table appears in neither schemas while @Brewal's solution works by making sure it appears in both schemas.

@FlyingDR's relies on asset filtering at the DBAL level, while Symfony's event system, which means it works only for Symfony (I think? Can orm:schema-tool:update be used in other frameworks, and if yes, are Symfony events also fired?). It means it also has to be disabled for migrations commands which need to know about the table.

@Brewal's solution relies on the ORM event system, which will kick in only for the ORM. I hope we are not forgetting about other cases relying on schema diffing and not relying on the ORM somehow.

Are there other pros or cons I'm forgetting ?

@wmouwen
Copy link

wmouwen commented Mar 15, 2024

by making sure the migrations table appears in neither schemas

Removing the table from the schemas won't detect changes to version_column_name or the other column-related settings. Having the table appear in both schemas does cover this.

@greg0ire
Copy link
Member

Gave it a try, and I'm starting to see some cons:

  1. Is it really orm:schema-tool:update's responsibility to create that migrations table? If you are using orm:schema-tool:update, then you are bypassing migrations, and do not need that table so… there is no point in detecting changes in it, is there? Likewise, the role of doctrine:schema:validate is, according to the error message in @ValentinDk 's screenshot above, to validate that the schema is in sync with the mapping files. The migrations table is not represented by mapping files, and is not part of the ORM schema.
  2. It leads to a duplication between the listener code and this method: https://github.com/doctrine/migrations/blob/3.7.x/lib/Doctrine/Migrations/Metadata/Storage/TableMetadataStorage.php#L231-L246

I think I'd rather we go with @FlyingDR's solution, where only doctrine/migrations knows about the metadata storage table.

@wmouwen
Copy link

wmouwen commented Mar 17, 2024

I gave it a try as well and ended up opening three PRs where each PR requires the previous one.

  1. Create postGenerateComparisonSchema event orm#11374
  2. Remove migration table from generated comparison schema #1418
  3. Register event listener to remove migration table from comparison schema DoctrineMigrationsBundle#529

Is it really orm:schema-tool:update's responsibility to create that migrations table? If you are using orm:schema-tool:update, then you are bypassing migrations, and do not need that table so… there is no point in detecting changes in it, is there?

I don't believe so as doctrine/migrations is using doctrine/orm, however doctrine/orm can open up its core through events to allow doctrine/migrations to alter its behavior. It has allowed it to do so with the postGenerateSchema event, so adding postGenerateComparisonSchema felt logical. The SchemaAssetFilter serves the same purpose but with a different, less open approach.

The migrations table is not represented by mapping files, and is not part of the ORM schema.

Hence the choice to remove the table from the comparison schema instead of adding it to the generated (metadata) schema. This doesn't catch changes to version_column_name as I mentioned in a previous comment, but there's no proper way to put that change in a migration anyway.

One thing that is missing is documentation on how to use #1418 without DoctrineMigrationsBundle. I feel like we can and should add that to the docs of this repository.

@stof
Copy link
Member

stof commented Oct 10, 2024

As long as the issue of generating a DROP SCHEMA public in the down method is not solved, this is not even a workaround at all, as the diff will never be empty due to this statement.

Also, checking diff for failure is not reliable. If a PHP exception happens, you will also get a failure exit code, but this shouldn't indicate a success of the comparison. We really need a checking command.

So there must be a crucial difference between how both places implement this feature

The reason is that the DiffGenerator temporarily changes the configuration of DBAL to ignore that table from the introspection when generating the diff (so that it does not think it exists):

$this->dbalConfiguration->setSchemaAssetsFilter(
(to be exact, it permanently changes the config until the end of the PHP process, making this class unsuitable for usages outside the diff command that exits the process once generated)

@greg0ire
Copy link
Member

Oh right, this is not a workaround for Postgres users indeed. We'll have to give this more thought.

@ghost
Copy link

ghost commented Oct 10, 2024

Also: running bin/console doctrine:migrations:diff in a pipeline might be problematic, since it tries to actually create a migration class / file. There doesn't appear to be a --dry-run flag on that command? A dry run might make it easier to replace bin/console doctrine:schema:validate -v.

@greg0ire
Copy link
Member

Well… the CREATE SCHEMA public in the down method issue just got fixed: #1463 🙂

might be problematic, since it tries to actually create a migration class / file

Do you mean because your CI filesystem is readonly or what? Please elaborate.

A dry-run flag might make sense indeed, it could be useful for other purposes than checking that the diff is empty, but I think a standalone command that would exit with 0 in case of an empty diff would be better, because you would not have to negate it like I did.

@cosminsandu
Copy link

cosminsandu commented Oct 11, 2024

Just Upgrading doctrine/migrations (3.8.1 => 3.8.2) https://github.com/doctrine/migrations/releases/tag/3.8.2
and the php bin/console doctrine:schema:validate -v returns

// 1 schema diff(s) detected:

     DROP TABLE doctrine_migration_versions;

But the php bin/console make:migration correctly returns [WARNING] No database changes were detected.

@ghost
Copy link

ghost commented Oct 11, 2024

might be problematic, since it tries to actually create a migration class / file

Do you mean because your CI filesystem is readonly or what? Please elaborate.

A dry-run flag might make sense indeed, it could be useful for other purposes than checking that the diff is empty, but I think a standalone command that would exit with 0 in case of an empty diff would be better, because you would not have to negate it like I did.

I seem to have misunderstood this part:

We'd be moving the sync test functionality to a place where it would belong more (the migrations package)
The part that we need in the pipeline, is the mapping validation, which I suppose is not planned for any changes?

We have a couple of use cases for schema:validate - one is to check mappings in the pipeline step, but that one runs with --skip-sync, and check the exit status. We probably could live with a unnecessary Migration being created here, since the pipeline step would run in a container anyway, which at the moment is able to write to the folder.

The other use case could be compared to schema:update, where we use it when a developer have switched branches a couple of times, and gotten their database out of sync.
Since we have over 200 tables, some of them with gigabytes of data, it's usually not feasible to restore the database to an earlier point, when all we need to 'fix' is a couple of columns or indexes.

I usually use schema:validate -v to print out which columns are in error, and either use schema:update to fix it, or run the queries in a console.

If schema:update was replaced with migration:diff I would have to create a unwanted migration, and either run it, and wipe it from the migration metadata table, or copy the queries out of it, and run it manually.
That seems like an degradation in workflow experience, over having the schema:update command at hand, as it is now.

Also: a couple of projects are not using the migration system for security reasons. If the schema:validate sync part / schema:update part got moved to migrations, would there be no way to check if those are in sync with the development environment?

@greg0ire
Copy link
Member

The part that we need in the pipeline, is the mapping validation, which I suppose is not planned for any changes?

The mapping validation is not planned for any changes, no.

We probably could live with a unnecessary Migration being created here, since the pipeline step would run in a container anyway, which at the moment is able to write to the folder.

Yes… and more importantly, the exit code of the command being negated in my proposal would make your deployment fail right after the file is created.

If schema:update was replaced with migration:diff I would have to create a unwanted migration, and either run it, and wipe it from the migration metadata table, or copy the queries out of it, and run it manually.

Our plan would be to have a mode for the migrations package. When in changelog mode, everything would work as it works now. When in "no-changelog" mode, things would work "in memory": no migration file, and the metadata table would be ignored. So basically it would work like schema:update, and the end result would be that we would have this functionality reside in a single package instead of 2, making maintenance easier.

Also: a couple of projects are not using the migration system for security reasons. If the schema:validate sync part / schema:update part got moved to migrations, would there be no way to check if those are in sync with the development environment?

Can you elaborate on the "security reasons"?

@ghost
Copy link

ghost commented Oct 11, 2024

Also: a couple of projects are not using the migration system for security reasons. If the schema:validate sync part / schema:update part got moved to migrations, would there be no way to check if those are in sync with the development environment?

Can you elaborate on the "security reasons"?

In some environments, database schema changes need to be controlled more tightly, using a two-man rule, and all schema changes are deployed by hand. The database user which Doctrine uses, have only a limited set of grants, and do not allow for schema changes. Those projects do not have the Doctrine Migrations Bundle installed at all.

@arderyp
Copy link

arderyp commented Oct 11, 2024

Also: a couple of projects are not using the migration system for security reasons. If the schema:validate sync part / schema:update part got moved to migrations, would there be no way to check if those are in sync with the development environment?

I don't fall into this camp, but this is a good question

@greg0ire
Copy link
Member

greg0ire commented Oct 11, 2024

I see. With the new design you would still have to install doctrine/migrations bundle on those projects. Not for migrating anything, but just to run that check command I guess, in non-changelog mode. A read-only user would be sufficient for that purpose. Feels a bit overkill, so I'll make sure to mention your point if we have internal discussions about this. Note that if we decide against moving the schema:validate sync part, you would still need the bundle in dev to do the schema:update part.

@VincentLanglet
Copy link
Contributor

Coming from doctrine/orm#11750

The fact that doctrine:schema:validate does multiple things and is used for migrations might be the initial issue.

I thought about the @greg0ire workaround

bin/console doctrine:schema:validate  --skip-sync && bin/console doctrine:migrations:up-to-date && ! bin/console doctrine:migrations:diff

And maybe to avoid any error bin/console doctrine:migrations:diff an improvement could

bin/console doctrine:migrations:diff --allow-empty-diff

and then check that no file was generated.

But Imho the right way could be to

  • Split doctrine:schema:validate in two (one command for mapping, one command for sync)
  • Provide a doctrine:migrations:valide commande to ensure there is no diff

This way, schema users could run doctrine:mapping:validate and doctrine:schema:validate.
And migrations users could run doctrine:mapping:validate and doctrine:migrations:validate.

@greg0ire
Copy link
Member

greg0ire commented Dec 5, 2024

Sounds fine, but bear in mind that in the end, we would like to remove the SchemaTool from the ORM altogether, so every user would end up being a migrations user, it's just that there would be a changelog mode you could turn off. So there would be no doctrine:schema:validate. The initial idea comes from @beberlei , I'll try to get him to comment on this, but in any case, I think building doctrine:migrations:validate is the best starting point for this.

@beberlei
Copy link
Member

beberlei commented Dec 5, 2024

@greg0ire we did check removing SchemaTool until we found out that Migrations relies on it completly for generating the diffs 😆 One option we could consider is removing the doctrine:schema:* cmmands from the ORM and only making SchemaTool available programatically.

@VincentLanglet I think your thoughts on separating the commands make total sense, just wondering how we can do this in a way where people upgrading sillently loose their mapping validation. schema:validate might need to be split into two new commands, leaving the original as-is and then remove it after deprecation.

@greg0ire
Copy link
Member

greg0ire commented Dec 5, 2024

we did check removing SchemaTool until we found out that Migrations relies on it completly for generating the diffs 😆 One option we could consider is removing the doctrine:schema:* cmmands from the ORM and only making SchemaTool available programatically

Oh right… if we go with that option, should we move SchemaTool to the migrations package? Or would there be something preventing the move?

@Crovitche-1623
Copy link

For me, the previous answers did not work.

Here is my workaround using MySQL until another solution is found :

  1. Create a custom SchemaManager to prevent the doctrine migration table to be created :
<?php

declare(strict_types=1);

namespace App\Doctrine\DBAL\Schema;

use Doctrine\DBAL\Schema\{MySQLSchemaManager, Table};

class CustomMySqlSchemaManager extends MySQLSchemaManager
{
    #[\Override]
    public function createTable(Table $table): void
    {
        if ($table->getName() === 't___doctrine_migration_versions') {
            return;
        }

        parent::createTable($table);
    }
}
  1. Create a dedicated migration to create it :
<?php

declare(strict_types=1);

namespace DoctrineMigrations;

use Doctrine\DBAL\Schema\Schema;
use Doctrine\Migrations\AbstractMigration;

final class Version20221104163712 extends AbstractMigration
{
    /**
     * @return  non-empty-string
     */
    #[\Override]
    public function getDescription(): string
    {
        return 'Create the `t___doctrine_migration_versions` table ';
    }

    #[\Override]
    public function up(Schema $schema): void
    {
        // language=MySQL
        $sql = '
            CREATE TABLE t___doctrine_migration_versions(
                version VARCHAR(191) NOT NULL,
                executed_at DATETIME NOT NULL,
                execution_time INT UNSIGNED NOT NULL,
                PRIMARY KEY (version)
            ) DEFAULT CHARACTER SET utf8mb4 COLLATE `utf8mb4_unicode_ci` ENGINE = InnoDB;
        ';

        $this->addSql($sql);
    }

    #[\Override]
    public function down(Schema $schema): void
    {
        // language=MySQL
        $sql = '
            DROP TABLE t___doctrine_migration_versions;
        ';

        $this->addSql($sql);
    }

    #[\Override]
    public function isTransactional(): bool
    {
        return false;
    }
}
  1. Configure the schema_filter to exclude the table. In my case, I decided to exclude all table that start with t___.
doctrine:
    dbal:
        connections:
            default:
                schema_filter: /^(?!t___)/
  1. Configure the table_storage
# config/packages/doctrine_migrations.yaml

doctrine_migrations:
    storage:
        table_storage:
            table_name: 't___doctrine_migration_versions'

@greg0ire
Copy link
Member

greg0ire commented Jan 16, 2025

I just released doctrine/doctrine-migrations-bundle 3.4.0, it should fix this issue, please confirm.

@xabbuh
Copy link
Member

xabbuh commented Jan 16, 2025

@greg0ire looks good on an application I just checked where this was an issue before

@VincentLanglet
Copy link
Contributor

I just released doctrine/doctrine-migrations-bundle 3.4.0, it should fix this issue, please confirm.

I can confirm I don't have the issue in 3.4.0

@greg0ire
Copy link
Member

Thank you both :)

@devnix
Copy link

devnix commented Jan 17, 2025

Not sure if related, but after upgrading to 3.4.0 from 3.3.1, migrations have stopped working.

Image

A status shows all the migrations as not executed, but the table (as stated on the screenshot) exists and is populated.

Image

@greg0ire
Copy link
Member

Yes, I think my code needs a few slight adjustments I've described in doctrine/DoctrineMigrationsBundle#584

@cosminsandu
Copy link

It seems it does not work with multiple entity manager connections

@tobias-93
Copy link

tobias-93 commented Jan 21, 2025

I have the same issue as @cosminsandu. I solved it by adding an extra tag in a compile pass for the extra entity manager which also uses migrations:

<?php

namespace App\DependencyInjection\Compiler;

use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;

class DoctrineMigrationsFilterCompilePass implements CompilerPassInterface
{
  public function process(ContainerBuilder $container): void
  {
    $schemaFilter = $container->findDefinition('doctrine_migrations.schema_filter_listener');
    $schemaFilter->addTag('doctrine.dbal.schema_filter', ['connection' => 'your_custom_connection']);
  }
}

Then add it to the Kernel with a higher priority than the DbalSchemaFilterPass (which has priority 0):

// src/Kernel.php
namespace App;

use App\DependencyInjection\Compiler\DoctrineMigrationsFilterCompilePass;
use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\HttpKernel\Kernel as BaseKernel;

class Kernel extends BaseKernel
{
    use MicroKernelTrait;

    // ...

    protected function build(ContainerBuilder $container): void
    {
        $container->addCompilerPass(new DoctrineMigrationsFilterCompilePass (), priority: 1);
    }
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment