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

After v3.4 all migrations shown as not executed in dev toolbar #584

Open
mmarton opened this issue Jan 16, 2025 · 28 comments · May be fixed by #587
Open

After v3.4 all migrations shown as not executed in dev toolbar #584

mmarton opened this issue Jan 16, 2025 · 28 comments · May be fixed by #587

Comments

@mmarton
Copy link

mmarton commented Jan 16, 2025

Bug Report

Just checked out the new release and all of my past migrations shown as not executed:

Image

Q A
Version 3.4.0
Previous Version if the bug is a regression 3.3.1

Summary

cache cleared, metadatastorage synced, migration table contains all 158 entry. The toolbar still shows everything as new

Current behavior

Expected behavior

How to reproduce

@greg0ire
Copy link
Member

greg0ire commented Jan 16, 2025

Here is the diff: 3.3.1...3.4.0

Maybe the change I did in MigrationsCollector.php was wrong?

Can you please dump $this->data here:

?

Or maybe it's the changes I did in in MigrationsFlattener.php? Can you try reverting them one by one?

@simonsolutions
Copy link

I experience the same problem after upgrade 3.3.1 to 3.4.0.

The array at line 35 of the file is an empty array.

@greg0ire
Copy link
Member

Ok then I guess that change is not the culprit 🤔

@simonsolutions
Copy link

Then dumping in MigrationsFlattener.php:

  • flattenAvailableMigrations -> dumps all available migrations
  • flattenExecutedMigrations -> dumps none of the executed migrations
  • (both dumps before the returns of the static functions)

@greg0ire
Copy link
Member

Can you try following https://dev.to/greg0ire/bisecting-vendors-12kd to pinpoint the first bad commit?

@mmarton
Copy link
Author

mmarton commented Jan 16, 2025

This was mentiont on the original issue, I just found the exact comment. Not sure if it helps, but here it is:
doctrine/migrations#1406 (comment)

@greg0ire
Copy link
Member

Oh right, thanks for digging that up. I'll revert my MR and release 3.4.1

@simonsolutions
Copy link

As far as i have seen now:

$executedMigrations = $metadataStorage->getExecutedMigrations();

the function "getExecutedMigrations" from doctrine migration, the initial "$this->isInitialized" is false so an empty list is returned.

https://github.com/doctrine/migrations/blob/d7d0df5651493e34cd7b6e150381e3102ca0f992/src/Metadata/Storage/TableMetadataStorage.php#L72

@greg0ire
Copy link
Member

greg0ire commented Jan 16, 2025

Or maybe, instead of reverting, the listener could be improved to detect that we are in HTTP context. When in HTTP context, it could disable itself 🤔

Regarding Doctrine commands such as doctrine:migrations:list, maybe the condition here can be made more precise:

if (! $command instanceof DoctrineCommand) {

@simonsolutions
Copy link

Its not only about the console command, i guess. In every request the migrations are shown as not executed.
On webrequest the $command variable in that function is not dumped and called

@simonsolutions
Copy link

Then i dump this:

$metadataStorage = $this->dependencyFactory->getMetadataStorage();

this is the result:


[Doctrine\Migrations\Metadata\Storage\TableMetadataStorage](phpstorm://open?file=/Users/simonriedmeier/Development/Symfony/runmagix/vendor/doctrine/migrations/src/Metadata/Storage/TableMetadataStorage.php&line=38) {#3255 ▼
  -isInitialized: false
  -schemaUpToDate: false
  -schemaManager: 
Doctrine\DBAL\Schema
\
MySQLSchemaManager {#3249 ▼
    #connection: 
Doctrine\DBAL
\
Connection {[#784 ▶](https://dev-runmagix.wip/_profiler/14ecaf?panel=dump#sf-dump-1291200273-ref2784)}
    #platform: 
Doctrine\DBAL\Platforms
\
MySQL84Platform {[#789 ▶](https://dev-runmagix.wip/_profiler/14ecaf?panel=dump#sf-dump-1291200273-ref2789)}
    -defaultTableOptions: null
  }
  -platform: 
Doctrine\DBAL\Platforms
\
MySQL84Platform {[#789 ▶](https://dev-runmagix.wip/_profiler/14ecaf?panel=dump#sf-dump-1291200273-ref2789)}
  -configuration: 
Doctrine\Migrations\Metadata\Storage
\
TableMetadataStorageConfiguration {#604 ▶}
  -connection: 
Doctrine\DBAL
\
Connection {[#784 ▶](https://dev-runmagix.wip/_profiler/14ecaf?panel=dump#sf-dump-1291200273-ref2784)}
  -comparator: 
Doctrine\Migrations\Version
\
AlphabeticalComparator {#3312}
  -migrationRepository: 
Doctrine\Migrations
\
FilesystemMigrationsRepository {#3321 ▼
    -migrationsLoaded: false
    -migrations: []
    -migrationDirectories: array:1 [▶]
    -migrationFinder: 
Doctrine\Migrations\Finder
\
GlobFinder {#3317}
    -versionFactory: 
Doctrine\Migrations\Version
\
DbalMigrationFactory {#3315 ▶}
  }
}

@greg0ire
Copy link
Member

Its not only about the console command, i guess. In every request the migrations are shown as not executed.

Yes, and again, the listener should not just listen to console events, but also to http events, and disable itself.

@simonsolutions
Copy link

Okay the __invoke function of the EventListener is called with enabled is set to true.
But onConsoleCommand is never called on a web request.

@greg0ire
Copy link
Member

greg0ire commented Jan 17, 2025

Yes. So we have to add another function (or just make disable public?), similar to onConsoleCommand, but for the web, that will fix one issue out of the two issues (the other issue I think there might be is with doctrine:migrations:list, but nobody confirmed it yet).

Related docs: https://symfony.com/doc/current/event_dispatcher.html

@eltharin
Copy link

Hi,
Same problem here, on profiler, migrations are "not executed",
A doctrine:migrations:list show good migrated date
but doctrine:make:migration show "[WARNING] You have X available migrations to execute."
if click yes, no new migration if created (if not neaded)

@greg0ire
Copy link
Member

@eltharin can't look into it until next week, but I think the fixes should be easy, in case you want to try

@eltharin
Copy link

Ok no problem,
It's not blocking.

@eltharin
Copy link

OK, I found the problem,
In migrations, in TableMetaDataStorage, it use tableExist to see if doctrine migration table exists.
but this function in DBAL use listTableNames in AbstractSchemaManager whitch filter assert

here are my proposes:
doctrine/dbal#6729
doctrine/migrations#1484

@greg0ire
Copy link
Member

There are already 3 bugs with the same cause here, I think it maybe a better way to do this might be to have the filter disabled by default instead of enabled by default, and to have a list of situations where it should be enabled.

@eltharin
Copy link

eltharin commented Jan 18, 2025

I don't understand, you want I update my Pr's with the new argument set to false by default or I make a new PR here for disable your SchemaFilterListener?

@greg0ire
Copy link
Member

I was only thinking out loud, I have yet to understand what your PRs do exactly (I'm away for the weekend so not really in a position to properly evaluate them).

My thinking was that there seems to be many situations were the filter should be disabled, and I'm not sure in how many situations it should be enabled. Before working on any piece of code, it might be great to list these situations.

@greg0ire greg0ire pinned this issue Jan 19, 2025
@greg0ire
Copy link
Member

Ok, I've read and understood your PRs, and I don't think they're going to be accepted (the DBAL one is breaking change for classes extending AbstractSchemaManager).

Now let's say we try to list situations where the filter I wrote is needed:

  • when running doctrine:schema:update
  • when running doctrine:schema:validate

Anything else?

@eltharin
Copy link

I can't anwser you for where other^^.
For a solution maybe in DBAL we can add a tableExistWithoutAsset
and remove $this->filterAssetNames from listTableNames and doListTableNames ?

@greg0ire
Copy link
Member

greg0ire commented Jan 19, 2025

I don't know, maybe this is easier: #587

Can everybody here test it and report back?

@eltharin
Copy link

with mariadb it's ok, migrations are shown as executed on profiler and no errors was shown in make:migration command for me! Nice!

@greg0ire greg0ire linked a pull request Jan 19, 2025 that will close this issue
@nicklog
Copy link

nicklog commented Jan 20, 2025

We use TableMetadataStorage in our own custom command and unfortunately this would not solve the problem. An interface or another check would be good. In any case, you should be able to deactivate the filter somehow for your own commands.

We currently use this in our own command.

    /** @return array<AvailableMigration> */
    public function getNewMigrations(): array
    {
        return $this->dependencyFactory->getMigrationStatusCalculator()->getNewMigrations()->getItems();
    }

@greg0ire
Copy link
Member

@nicklog with my changes, the filter is disabled by default, so, it should be for your custom commands.

@sgehrig
Copy link

sgehrig commented Jan 21, 2025

While debugging why our /health endpoint consistently returns an error indicating that not all migrations have been executed, I stumbled upon this issue.

In our case, the root cause appears to be the same as previously described: the schema filter automatically applied by the bundle. This filter is only disabled in the \Doctrine\Migrations\Tools\Console\Command\DoctrineCommand CLI commands. As a result, when attempting to determine the migration status within a controller, the list of executed migrations is always empty, leading to inaccurate results.

To address this temporarily, I implemented the following solution (summarized for brevity):

$config                     = $connection->getConfiguration();
$trueSchemaAssetsFilter     = static fn() => true;
$previousSchemaAssetsFilter = $config->getSchemaAssetsFilter() ?? $trueSchemaAssetsFilter;
$config->setSchemaAssetsFilter($trueSchemaAssetsFilter);

try {
    $statusCalculator = $migrationDependencyFactory->getMigrationStatusCalculator();
    $newMigrations    = $statusCalculator->getNewMigrations();
    // ...
}  finally {
    $config->setSchemaAssetsFilter($previousSchemaAssetsFilter);
} 

As far as I can tell, PR #587 will fix this issue however. So thanks a lot for your great work!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

6 participants