-
-
Notifications
You must be signed in to change notification settings - Fork 4.6k
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
Implemented PoolOptimizer #9261
Conversation
My spec in #8287 includes details about how conflict rules impact this optimization. But your implementation ignores them entirely, which might lead to breaking the dependency resolution (imagine that the latest version of a package adds a conflict rule forbidding only the latest release of the other one but allowing the previous patch release for instance) |
regarding the time where the pool optimizer is called, I'm not sure it is the best one compared to the |
It would get a different dependency hash and not be removed from the pool. They are actually not ignored at all :) |
ah sorry, I missed that part about the hash. |
@Toflar idea to ensure that the PoolOptimizer has no negative effect on the dependency solving: run the solver functional tests with and without the PoolOptimizer:
What do you think ? |
Should we have some way (CLI flag, env variable, magic incantation, etc...) to disable the PoolOptimizer when running composer, to help identify cases where a dependency solving error is caused by a bug of the optimization (if solving finds a solution without the PoolOptimizer, there is a bug which must then be reported and fixed in Composer) ? |
I just wanted to share results of my tests with Drupal. Before each test, I removed the A not-very-complex Drupal 9.1 dev project (this repo)
I can see the time improvement is not significant but memory improvement is. This is awesome work. Thanks! 👏 |
I also tried this on a project with Drupal 8; this time running
Again, not a significant differnce in time (this could be because it actually times updating packages on disk) but a significant reduction in peak memory. |
A Symfony application using the Symfony Flex plugin (needed because our Symfony applications use
Not much changes, which I think it expected as the Symfony Flex plugin already significantly reduces the versions that are considered. A Neos base distribution - the most complex project on my PC that isn't requiring Symfony Flex:
Similar results as Drupal 8, not a significant step in performance but a significant reduction of (peak) memory usage. Btw, checked the installed versions after each run and all runs install the same versions (between each run the vendor packages + |
Good job man ! I tested this two days ago with almost same runtime. Now this PR is a lot faster. No idea why :)
|
I tested a different Drupal scenario than the two from @hussainweb above, and got less dramatic results, but still a very nice improvement. My interest was for the use-case of doing a patch-level Drupal core update when all other dependencies are already up to date. This is because we're working on an initiative to add an auto-updater to Drupal, where this scenario will be run on production, including on cheap shared hosting plans with low memory limits. Setup: begin with a Drupal installation that's on a recently outdated patch-level version (latest Drupal version is 9.0.6). Also include some non-core dependencies (lightning_media brings in a bunch).
Test: profile an update to the latest Drupal core version. Start with a cold cache, since that's the likely scenario on shared hosting by the time this would be run.
Result:
While not an order of magnitude difference, that's still an excellent reduction in peak memory usage, which can make a big difference on some cheap hosting plans. |
@stof regarding the test suite: We already have loads of integration tests in the
Yes, totally. I've implemented and documented the new To everybody: Thanks a lot for the numbers, they look really promising which makes me very happy 😊 |
The fact that problem reporting is impacted by the pool optimization is precisely the reason why my proposal is that this "no-optimizer" run disables the assertion on the output in case of reported problems (keeping only an assertion that a problem is indeed reported). |
The output can't be the same though. In case of an unresolvable set of dependencies, the list of problems is a lot shorter if not all possible problems are listed but just the ones of the optimized pool (which BTW also improves readability a lot :)). |
I know that. That's why I'm saying that for problems, the non-optimized run must NOT assert the output, but only the fact that the run also triggers some problems (to make sure that the dependency solver is not successful when disabling optimization). |
I'm not sure a no-optimizer flag is the way to go. What about controlling if it is enabled via an environment variable. Composer already does this to control some other stuff. |
In any case, wouldn't this also need to be added to the install command too, since it still defaults to doing an update when there is no lock file? |
Using
For ezsystems/ezplatform-ee:2.5, contains commercial packages so you won't be able to replicate w/o subscription:
|
For the record: I had a look at why Drupal is so much slower compared to all the other projects and the reason is not Composer at all. The SAT solver is super fast for them too but the way Drupal handles package meta information is very, very slow (loads of redirects and 404s on https://www.drupal.org/files/packages and https://packages.drupal.org). It easily accounts for 80-90% of the total time spent during a |
@mvorisek this is not helpful. Work is being tracked and scheduled here, please don't go around nagging people. |
@Seldaek The comment from him might have sounded offensive to you but to be fair people not that closely related to this project have no chance to understand how much communication or agreement between you members and Toflar about a feature like this exists outside of this PR (at least this is what I read in some comments). They see just a feature of a contributor laying around for such a long time and no obvious reason. |
Just in case it may help, from #10200 there is a scenario that benefits greatly from this improvement (from timeout to 2s): {
"require-dev": {
"alcaeus/mongo-php-adapter": "^1.2.1",
"doctrine/data-fixtures": "^1.5.1",
"doctrine/doctrine-bundle": "^2.4.3",
"doctrine/mongodb-odm": "^2.2.2",
"doctrine/mongodb-odm-bundle": "^4.3.0",
"doctrine/orm": "^2.10.1",
"doctrine/phpcr-bundle": "^2.3.0",
"doctrine/phpcr-odm": "^1.5.4",
"jackalope/jackalope-doctrine-dbal": "^1.7.1",
"ocramius/proxy-manager": "^2.13",
"psy/psysh": "^0.10.9",
"symfony/symfony": "^4.4.32 || ^5.3.9",
"theofidry/composer-inheritance-plugin": "^1.2",
"theofidry/psysh-bundle": "^4.4",
"wouterj/eloquent-bundle": "^2.1.1"
}
} (the difference is relevant only if there is no .lock & no vendor). I do not know if this may worsen other cases though. |
Do you want to try https://github.com/spryker-shop/suite ? :)
This is probably the most demanding composer project in the world, and as such most likely will show the most dramatic improvements here. |
696e1db
to
b4eefcd
Compare
f0392aa
to
f8d307b
Compare
tests/Composer/Test/DependencyResolver/Fixtures/pooloptimizer/aliases.test
Show resolved
Hide resolved
ae2f006
to
d0b8072
Compare
@dereuromark here you go 😉
|
d0b8072
to
d0cf7ba
Compare
…ey had not been removed
d0cf7ba
to
4b7a54f
Compare
…nd compare outputs if user opts into optimizer
4b7a54f
to
c06be08
Compare
This looks awesome, about 25% faster and 80% reduction in memory usage for a medium sized Drupal 9 site with 30 modules:
Tested with this:
|
Tested with Open Y ( 229 composer packages )
composer cache warmed up OSX host
Command for testing composer -v -n --profile create-project ymcatwincities/openy-project --no-interaction
Memory improvement - Time is a bit higher +8.8% increase |
@podarok you should test without the network interfering (see #9261 (comment)) :) |
Just wanted to take a minute to thank you for making composer this performant, we really appreciate it! It has been a game changer |
This PR optimizes the
Pool
before the rules are created and passed on to theSolver
. Every package added to the pool results in an exponential increase of the number of rules theSolver
has to analyze so the best we can do is to try to throw away all the packages that aren't of any interest early on.I've implemented a new
PoolOptimizer
and set up a test framework to which we can add more optimizations in the future but the current implementation already provides huge benefits!It works by early selecting the correct package based on the provided policy. It's common to have loads of bugfix releases that all have the same dependency definition. If you imagine e.g.
symfony/routing
in its4.4
LTS branch, all of them hardly ever define newrequire
,replace
,provide
orconflict
definitions. Chances that4.4.0
and4.4.14
are the same, are really high and so we don't need to keep all of the package versions in between, we can drop the irrelevant ones depending on--prefer-stable
and--prefer-lowest
. There's a bit more to it and it's not just that simple but that's the basic idea behind the first optimization in the newPoolOptimizer
.And here are the very promising results:
Contao Managed Edition (https://github.com/contao/managed-edition/blob/master/composer.json):
Magento (https://github.com/magento/magento2/blob/2.4-develop/composer.json):
Laravel 7.x (https://github.com/laravel/laravel/blob/7.x/composer.json):
Packagist (https://github.com/composer/packagist/blob/master/composer.json)
Of course, the more complex the dependencies (or just the more dependencies in general) the higher the benefits of this PR.
I was too lazy to calculate the benefits in % but I guess we can summarize it like so: 🚀
I'm off for a few days now so I might not be replying to questions quickly but would love to see others giving it a try and share some numbers :)
For the lazy folks: Here's a precompiled phar (c996457):
composer.phar.zip