Create the database driver for MySQLi for async queries

Created on 21 January 2022, about 3 years ago
Updated 25 January 2023, about 2 years ago

Problem/Motivation

Running database queries as async queries lets us run them in parallel instead of sequentially. For pages with multiple "larger" queries this can speed up rendering such a page significantly. The problem is that the 3 by core supported database driver are all PDO drivers which do not support async queries. The PHP MySQLi extension does support async queries.
The other thing we need is PHP 8.1 with the Fiber functionality. Creating a database driver for MySQLi is the first step we need to make to start using async queries in Drupal.

@catch explained it as following:

Let's take something like https://www.drupal.org/dashboard - there's four or five different views on there in blocks.

If you hit a completely cold cache, you have to execute all the views building those blocks, including both the listing query, then rendering the entities within them.

The idea being explored is that for each lazy builder - let's assume we're able to loop over them, you'd render it up until you have to run a views listing query. When it's time to run that query (or whatever appropriate point to delegate to the Fiber), it's done inside a Fiber with a non-blocking query. You then ::suspend() the Fiber and return to the main process, and start the next lazy builder, if that also has to run a views listing query, you'd also run it non blocking and ::suspend(), then you loop over all the lazy builders again to see if the various queries have come back, then render the results for anything that has, then continue polling until everything is rendered.

The page still can't finish rendering until the slowest query is finished, but if you have five slow queries that take 500ms each, then you might be able to run them all at once in the space of 600ms-1s instead of definitely having to run them sequentially in 2.5s. While you're running those simultaneous queries, you can also get on with PHP/CPU-intensive tasks like rendering - assuming we're able to arrange things so that these can be interleaved.

Fibers only allow us to paralellize i/o, not CPU, because there's not any parallel execution at all, so it has to be something like an async MySQL or SOLR query or http request to make any difference, and there has to be something to be getting on with while it's running - either more i/o or something else.

Proposed resolution

For adding the MySQLi database driver we need the following:

  1. Create the MySQLi database driver which should extend the MySQL database driver.
  2. Add the PHP MySQLi extension as a requirement to the core MySQL and MySQLi database drivers.
  3. Introduce database driver extensions and autoload database drivers' dependencies Fixed which makes extending one database driver on another database driver a lot easier.

Remaining tasks

TBD

User interface changes

TBD

API changes

None

Data model changes

None

Release notes snippet

TBD

📌 Task
Status

Active

Version

10.0

Component
Database 

Last updated about 18 hours ago

  • Maintained by
  • 🇳🇱Netherlands @daffie
Created by

🇳🇱Netherlands daffie

Live updates comments and jobs are added and updated live.
  • Needs release manager review

    It is used to alert the release manager core committer(s) that an issue significantly affects the overall technical debt or release timeline of Drupal, and their signoff is needed. See the governance policy draft for more information.

  • Needs product manager review

    It is used to alert the product manager core committer(s) that an issue represents a significant new feature, UI change, or change to the "user experience" of Drupal, and their signoff is needed. If an issue significantly affects the usability of Drupal, use Needs usability review instead (see the governance policy draft for more information).

Sign in to follow issues

Merge Requests

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • 🇮🇹Italy mondrake 🇮🇹

    Nice writeup re differences between PDOMYSql and mysqli, https://websitebeaver.com/php-pdo-vs-mysqli

  • 🇬🇧United Kingdom catch

    From the github repo, it looks like any core inclusion is currently blocked on #3110546: Allow contributed modules (mostly database drivers) to override tests in core and Introduce database driver extensions and autoload database drivers' dependencies Fixed , so I'm going to mark this PP-2.

    None of this would stop us trying out some proof of concept stuff with PHP Fibers though.

  • 🇮🇹Italy mondrake 🇮🇹

    Actually, #3110546: Allow contributed modules (mostly database drivers) to override tests in core is only necessary on GitHub to skip running the ApcuBackendTest on GHA - I couldn't find a away to enable APCu properly there, so I am still skipping it.

    However, I would say that to be on a preferrable scenario, we would need two more issues in first:

    📌 Convert select query extenders to backend-overrideable services RTBC - this left harbour one year ago, then sank couple of weeks later, with only limited SAR activities afterwards. It's tagged for framework manager review.

    📌 Deprecate usage of Connection::getDriverClass for some classes, and use standard autoloading instead Fixed - to reduce boilerplate empty classes, which is blocked on the above

  • 🇬🇧United Kingdom catch

    Last blocker is in.

  • 🇬🇧United Kingdom catch

    From the issue summary:

    Create the MySQLi database driver which should extend the MySQL database driver.
    Add the PHP MySQLi extension as a requirement to the core MySQL and MySQLi database drivers.

    Why would we need the MySQLi extension for the current PDO-based MySQL driver?

  • 🇫🇷France andypost

    Very probably it means mysqlnd extension which is primary driver for Mysql and it's reused by PDO_mysql at least

  • 🇮🇹Italy mondrake 🇮🇹

    I'm still doing some changes on the mysqli module on GitHub - I'd like to have it pass tests there before moving the code here. Also, I think we should try to refactor there the parser code that was taken from doctrine/dbal to manage converting SQL statements from having named placeholders (':placeholder') to have positional placeholders ('?') - it's overkill for the specific purpose here.

    Question: do we want to introduce a new 'mysqli' module, or shall we try to have a single 'mysql' module with two drivers, 'mysql' and 'mysqli'?

  • 🇬🇧United Kingdom catch

    @lauriii posted some question in slack, added them, and the answers to the issue summary.

  • 🇬🇧United Kingdom catch

    Question: do we want to introduce a new 'mysqli' module, or shall we try to have a single 'mysql' module with two drivers, 'mysql' and 'mysqli'?

    Two drivers in module might be simpler, but:

    1. Can we make it work where only one option is available due to installed extensions? And also not show drivers that won't work to people?
    2. Would there be any need for the new driver to be experimental?

    btw depending on the answers to this, there might be issue summary updates needed now.

  • 🇬🇧United Kingdom catch
  • 🇬🇧United Kingdom catch
  • 🇬🇧United Kingdom catch
  • 🇬🇧United Kingdom catch
  • 🇭🇺Hungary Gábor Hojtsy Hungary

    Lauri pinged me to provide feedback as a product manager. These two statements are contradictory and are critical to asses the impact of the proposal I think :)

    We need to continue to support PDO since not all hosting has php_mysqli enabled, so it has to be a new driver. However it might be fine to have both drivers in the same module.

    vs

    Add the PHP MySQLi extension as a requirement to the core MySQL and MySQLi database drivers.

  • 🇮🇹Italy mondrake 🇮🇹

    IMHO there's no need to add the PHP MySQLi extension as a requirement to the core (PDO, existing) MySQL database driver.

    That will obviously have to be a requirement for the NEW MySQLi driver, though.

    The installability of a driver is driven by the installable() method in the Install/Tasks class of each driver.

    For the existing PDO/MySQL driver it is inheriting from the base class

      public function installable() {
        return $this->hasPdoDriver() && empty($this->error);
      }
    

    For the new driver it will be something like

      public function installable() {
        return extension_loaded('mysqli');
      }
    
  • 🇫🇷France andypost

    I think in case if both drivers are available some description should explain benefits using mysqli
    So on the most of hostings user could contact support to ask more performant driver

    If benefits is obvious then we can add some logic to set "preferred driver" at install screen

  • Assigned to mondrake
  • 🇮🇹Italy mondrake 🇮🇹

    I'll start adding the code from GitHub as a separate module, since I understand the driver may start as experimental, and I do not think we can have a module with a released driver and an experimental one in the same base.

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,183 pass
  • 🇮🇹Italy mondrake 🇮🇹

    OK, now code checks pass. I suppose the next step is to get tests run using the new driver, but I have no clue how this can happen. I assume infra need to do infra things?

  • 🇫🇷France andypost

    I filed #3384661: Upgrade PHP 8.2 image to 8.2.10 and as the new image will pass tests we can and mysqli extension

  • Status changed to Needs review over 1 year ago
  • 🇫🇷France andypost

    The extension already in the CI image

    $ docker run --rm -it drupalci/php-8.2-apache:production php --ri mysqli
    
    mysqli
    
    MysqlI Support => enabled
    Client API library version => mysqlnd 8.2.0
    Active Persistent Links => 0
    Inactive Persistent Links => 0
    Active Links => 0
    
    Directive => Local Value => Master Value
    mysqli.max_links => Unlimited => Unlimited
    mysqli.max_persistent => Unlimited => Unlimited
    mysqli.allow_persistent => On => On
    mysqli.rollback_on_cached_plink => Off => Off
    mysqli.default_host => no value => no value
    mysqli.default_user => no value => no value
    mysqli.default_pw => no value => no value
    mysqli.default_port => 3306 => 3306
    mysqli.default_socket => no value => no value
    mysqli.allow_local_infile => Off => Off
    mysqli.local_infile_directory => no value => no value
    
  • 🇫🇷France andypost

    @mondrake I think you can edit drupalci.yml to run only new module's testgroup

  • 🇮🇹Italy mondrake 🇮🇹

    #50 no I think we need first to be able to install using the new driver. I.e. a new test environment option like Test PHP 8.1 & MySQL 5.7 via MySQLi. Editing drupalci.yml we can only get the new module test to run with the other PDO drivers under the hood.

  • 🇮🇹Italy mondrake 🇮🇹

    Attention - the driver requires minimum PHP 8.1.2 where this was fixed.

  • 🇫🇷France andypost

    @mondrake as I see last CI run https://dispatcher.drupalci.org/job/drupal_patches/201116/testReport/Dat...
    there's mysqli tests running fine)

  • 🇷🇺Russia Chi

    Posgres also supports async queries
    https://www.php.net/manual/en/function.pg-send-query.php

    Is there a chance that Drupal some day would have a driver for it?

  • 🇬🇧United Kingdom catch

    I've updated the issue summary to remove the bit about the mysqli requirement on the PDO driver and to say we'll start with an experimental module but eventually merge the stable driver into the main mysql module.

    @Chi a pgsql driver should be able to build on the work done here, so that seems possible if someone's motivated to work on it.

  • 🇷🇺Russia Chi

    I think mysqli module should implement hook_requirements() and check if PHP MySQLI extension is available.

  • Issue was unassigned.
  • 🇮🇹Italy mondrake 🇮🇹

    #56 maybe, but I do not think that's relevant. During early installation there's no module concept yet, and installability of a driver is checked as described in #43.

  • 🇬🇧United Kingdom catch

    We already recommend PHP 8.1.6 or above. Easy to add a hard requirement on PHP 8.1.6 to the experimental module, but we could probably only add a hook_requirements() warning checking the database settings + PHP version once it's incorporated into the main mysql module.

    I wonder if we should try to do the mysqli testing directly on gitlab CI rather than getting a new environment on DrupalCI since it's getting a lot closer to the point where core can switch over entirely - not sure exactly how close though and wouldn't want to artificially hold this up on that either.

  • 🇬🇧United Kingdom catch

    I did some googling about PDO support for async and found the PDO brainstorming page: https://wiki.php.net/internals/pdo/brainstorming

    If that's up-to-date, then it's been discussed, but nothing has happened for years. It's possibly that PHP Fibers support will increase demand and might spur some activity, but I think we're looking at more years before it would be supported, if ever.

    Should we open a follow-up on how to handle the actual async support? For the BigPipe/Renderer implementations of fibers the main use case is to fire off a single async query, suspend the fiber so the calling code can do something else, then when we're resumed see if it came back. It looks like this might require multiple database connections (since 'something else' could also be executing database queries) so could be a bit tricky.

    I can't see us needing or wanting to execute multiple queries async in core although support for that would be good to add at some point, maybe its own issue.

  • 🇮🇹Italy mondrake 🇮🇹

    ##6 Yeah I think better have that as a follow-up, the issue here I think it's to get a mysqli driver that is like-for-like the pdo_mysql one.

  • 🇳🇱Netherlands daffie

    @mondrake: It looks good. Are you going to be the official maintainer of this new database ddriver?

  • 🇪🇸Spain fjgarlin

    Replied via slack, copying here too:

    This is the way we are defining the database and variables for GitlabCI for core: https://git.drupalcode.org/project/gitlab_ci_testbed_for_drupal_core/-/b...

    Note that this uses the images generated from https://git.drupalcode.org/project/drupalci_environments but does not use https://git.drupalcode.org/project/drupalci_testbot to run the tests.

    We connect the images via the .gitlab-ci.yml file and just call the core/scripts/run-tests.sh file to run the tests, so everything you need should be able to be set here: https://git.drupalcode.org/project/gitlab_ci_testbed_for_drupal_core/-/b...

  • last update over 1 year ago
    30,222 pass
  • 🇮🇹Italy mondrake 🇮🇹

    I *think* now on GitlabCI we can run tests on the new driver by only changing the database URL. Just for the sake of getting it run I changed in the MR from the PDO MySQL driver to mysqli, obviously the change would have to be structured moving forward.

  • 🇮🇹Italy mondrake 🇮🇹

    Yes, a failure like

        PHPUnit Test failed to complete; Error: PHPUnit 9.6.8 by Sebastian Bergmann
        and contributors.
        
        Testing Drupal\Tests\migrate_drupal\Kernel\MigrateMissingDatabaseTest
        E                                                                   1 / 1
        (100%)R
        
        Time: 00:01.211, Memory: 4.00 MB
        
        There was 1 error:
        
        1)
        Drupal\Tests\migrate_drupal\Kernel\MigrateMissingDatabaseTest::testMissingDatabase
        mysqli_sql_exception: php_network_getaddresses: getaddrinfo for
        does_not_exist failed: Name or service not known
        

    tells us that the pipeline used the new driver for testing.

  • last update over 1 year ago
    30,222 pass
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,222 pass
  • Status changed to Needs work over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    There are quite a few things to fix looking at the pipeline failures.

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,222 pass
  • last update over 1 year ago
    30,222 pass
  • last update over 1 year ago
    30,210 pass, 2 fail
  • Status changed to Postponed over 1 year ago
  • last update over 1 year ago
    30,210 pass, 2 fail
  • last update over 1 year ago
    30,259 pass
  • last update over 1 year ago
    30,258 pass, 2 fail
  • last update over 1 year ago
    30,259 pass
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,258 pass, 1 fail
  • last update over 1 year ago
    30,353 pass, 1 fail
  • last update over 1 year ago
    30,433 pass
  • last update over 1 year ago
    30,432 pass, 1 fail
  • 🇮🇹Italy mondrake 🇮🇹

    There are update tests failing because the database dumps contain the 'mysql' module installed, but since the test are run with 'mysqli', when updating we get a requirement error 'install the mysqli module that contains the database driver'.

    Not sure how to tackle that.

  • 🇫🇷France andypost

    it could use update hook to pass upgrade tests, or it will need common fixture to apply to all this tests

  • 🇬🇧United Kingdom catch

    There are update tests failing because the database dumps contain the 'mysql' module installed, but since the test are run with 'mysqli', when updating we get a requirement error 'install the mysqli module that contains the database driver'.

    Not sure how to tackle that.

    I think we've had the same problem when the database driver modules were introduced in the first place, the way to fix it is to re-import the database dump into a real site, install the mysqli module, then re-export the database dump. Then it gets picked up when you try to update from the dump and everything works (or hopefully). You have to rewrite history to make it seem as if the mysqli module existed at the time of the dump - so it might mean manually copying the module folder back to the old commit hash that you import the database dump to, or potentially manually editing the uncompressed dump to add the correct lines etc.

  • last update over 1 year ago
    30,437 pass, 1 fail
  • last update over 1 year ago
    30,437 pass, 1 fail
  • last update over 1 year ago
    30,383 pass, 64 fail
  • 🇮🇹Italy mondrake 🇮🇹

    Thanks for #71 (that's what I was afraid of :))

    Tried manually updating the dump files, just to add 'mysqli' to the system.schema key_value collection (should suffice? module has no config), then compressing them with
    $ tar -czf drupal-9.4.0.bare.standard.php.gz drupal-9.4.0.bare.standard.php

    But now the import fails during the update test with
    ParseError: syntax error, unexpected character 0x00, expecting end of file

    It could be some wrong way of calling tar?

    Just throwing here if anyone has faced this earlier and has a ready-made answer

  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • 🇮🇹Italy mondrake 🇮🇹

    FTR, the changes to the database dumps:

    1. gunzip core/modules/system/tests/fixtures/update/drupal-9.4.0.[type].standard.php.gz ([type] three of them, bare, filled and phpass)
    
    2. Edited and manually added the module: add 'mysqli' to the system.schema key_value collection and to the extension.core collection
    
    3. gzip core/modules/system/tests/fixtures/update/drupal-9.4.0.[type].standard.php
    

    Note - do not use tar for the purpose as it fills files with null characters at the end

  • 🇮🇹Italy mondrake 🇮🇹

    We are almost there. There are still a handful of installer test failures, again because an attempted uninstall of the 'mysql' module fails since it is a dependency of the new module, and 2 kernel tests that fail on MySql 8 but not on MySql 5.7.

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,438 pass
  • last update over 1 year ago
    30,442 pass
  • Status changed to Needs review over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    So, this is passing on MySQL 5.7 now, but not on MySQL 8 because of this test failing:

    PHPUnit 9.6.8 by Sebastian Bergmann
        and contributors.
        
        Testing Drupal\KernelTests\Core\Database\BasicSyntaxTest
        ...E...                                                             7 / 7
        (100%)
        
        Time: 00:05.293, Memory: 4.00 MB
        
        There was 1 error:
        
        1) Drupal\KernelTests\Core\Database\BasicSyntaxTest::testConcatWsFields
        Drupal\Core\Database\DatabaseExceptionWrapper: Illegal mix of collations
        for operation 'concat_ws': SELECT CONCAT_WS('-', ?, "name", ?, "age") FROM
        "test48178452test" WHERE "age" = ?; Array
        (
            [:a1] => name
            [:a2] => age
            [:age] => 25
        )
        
        
        /builds/issue/drupal-3259709/core/modules/mysqli/src/Driver/Database/mysqli/ExceptionHandler.php:56
        /builds/issue/drupal-3259709/core/lib/Drupal/Core/Database/Connection.php:858
        /builds/issue/drupal-3259709/core/tests/Drupal/KernelTests/Core/Database/BasicSyntaxTest.php:67
        /builds/issue/drupal-3259709/vendor/phpunit/phpunit/src/Framework/TestResult.php:728
        
        Caused by
        mysqli_sql_exception: Illegal mix of collations for operation 'concat_ws'
        
        /builds/issue/drupal-3259709/core/modules/mysqli/src/Driver/Database/mysqli/Statement.php:84
        /builds/issue/drupal-3259709/core/lib/Drupal/Core/Database/Connection.php:826
        /builds/issue/drupal-3259709/core/tests/Drupal/KernelTests/Core/Database/BasicSyntaxTest.php:67
        /builds/issue/drupal-3259709/vendor/phpunit/phpunit/src/Framework/TestResult.php:728
        
        ERRORS!
        Tests: 7, Assertions: 8, Errors: 1.

    I cannot reproduce this locally; can it be it's some setting of MySQL8 on the runner?

    Also, Drupal\Tests\mysqli\Kernel\mysqli\LargeQueryTest behaves erratically, if I remove the dump() in it then the statement errors under destruction (?).

    Apart from these two, I think this can go through more heavy-lifting reviewing.

    Some help would be appreciated on the GitlabCI set-up to make this run independently from the PDO-Mysql tests; the current changes are only a shortcut to make test run with mysqli instead of pdo_mysql.

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,442 pass
  • last update over 1 year ago
    30,442 pass
  • last update over 1 year ago
    30,442 pass
  • 🇪🇸Spain fjgarlin

    I added the async job as a separate job and left the others as they were. Note that this issue contains 🐛 _TARGET_DB_TYPE does not exist Active that will be committed shortly, so we might get a conflict on the rebase. I can take care of it.

  • 🇬🇧United Kingdom catch

    We might want to call it 'MYSQLI' instead of 'ASYNC' but otherwise that looks great.

  • last update over 1 year ago
    30,442 pass
  • 🇮🇹Italy mondrake 🇮🇹

    Now I understand the logic, thanks a lot @fjgarlin.

    How about generalizing more, and instead of DB_ASYNC, add two more variables

    _TARGET_DB_DRIVER
    _TARGET_DB_DRIVER_MODULE

    and use them too in building the database URL?

  • last update over 1 year ago
    30,442 pass
  • 🇪🇸Spain fjgarlin

    @mondrake - we initially had some more variables around the database type and version, but wanted to simplify the job definition as much as possible, so it's easy to understand and to create new jobs. Having said that, it's an option.

    Right now, we only added an optional variable, which I think keeps the simplicity of new jobs. I renamed it as suggested in #78 but feel free to rename it again if there is a better name.

  • 🇨🇦Canada Charlie ChX Negyesi 🍁Canada

    I had some misgivings on user space parsing of SQL but there's nothing better we can do: I hoped we could reuse the parser in PDO. But as far as I can tell, this is only called on execute and even then only when PDO::ATTR_EMULATE_PREPARES is set to true and then the placeholders are replaced by the arguments not ?. There's no doubt PDO creates the ? placeholder'd query because MySQL API only supports that but if it is exposed anywhere, I can't figure out where, PDOStatement::debugDumpParams does not display it. (I ran all four combinations of MYSQL_ATTR_DIRECT_QUERY / ATTR_EMULATE_PREPARES )

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,452 pass
  • Assigned to mondrake
  • Status changed to Needs work over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    working on a unit test for NamedPlaceholderConverter

  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,490 pass
  • last update over 1 year ago
    30,490 pass
  • Issue was unassigned.
  • Status changed to Needs review over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    Added tests for NamedPlaceholderConverter.

  • 🇮🇹Italy mondrake 🇮🇹

    @chx thanks for #81.

    Clearly, NamedPlaceholderConverter is not optimal: it destroys and rebuilds a SQL statement that was built already.

    A solution to try and retrieve from pdo_mysql something to be used by mysqli is for me ... weird. It'd keep PDO as a required PHP extension, and would just move the parsing from our class to PHP internals. If (if, I dunno) that would also require establishing a client connection it would be even more weird.

    On the other side, IIUC mysqli is meant to be a very thin client. Expecting it to implement positional placeholders would be against its concept.

    Personally, I think we should aim at trying to embed alternative placeholder generation (positional/named) into dynamic queries instead - and avoid reparsing post-factum. It would definitely increase performance. Then, we should be able to still do the placeholder conversion for static queries - allowing an option to be passed to ::query(). But that would be D11 material IMHO.

  • last update over 1 year ago
    30,509 pass
  • Status changed to Needs work over 1 year ago
  • 🇳🇱Netherlands daffie

    I have talked to @longwave, @alexpott and @catch about the new MySQLi database driver. The plan that we came up with is the following:

    1. The MySQLi module goes into core as experimental. The is important because we need the freedom to change anything in this module that we would like to change without triggering a BC break. Also when users install a new Drupal site they need to see this module as experimental instead of it being a regular module.
    2. The next phase for the module is to add all the parallel query functionality to make the PHP Fibers part work for the database queries. Maybe add other improvements? After this is done we are going to ask expert Drupal users to test out the database driver.
    3. When a number of expert Drupal users have been using this database driver without any problems and all other bugs have been fixed that are the result of using the MySQLi database driver and the PHP mysqli extension, the module can be marked as stable.
    4. When the module is marked as stable, we can start informing site owners that we have a new database driver and what is improved compared to the existing MySQL driver. The only thing that existing site owners need to do start using the MySQLi driver is to update their database driver settings in settings.php. Change 'driver' => 'mysql to 'driver' => 'mysqli'. Just adding the i to mysql. For new site installs the MySQLi database driver becomes the default option.
    5. When the MySQLi database driver is ready for becoming the default driver for MySQL, the are going to remove all code from the MySQL module that is overridden in the MySQLi module and all code from the MySQLi module is copied to the MySQL module. All override code in MySQLi module will be removed. The MySQL and the MySQLi module are now the same. The MySQL module will stay the default module for a MySQL database. The MySQLi module will now be marked as deprecated. Now all site owners who are using a MySQL or equivalent are now using the code from the original MySQLi module. Existing site owners who did not change to the MySQLi driver do not have to do anything and they are now using the code from the original MySQLi driver. Site owners who changed their site to switch to the MySQLi module shall need to change back to the default MySQL driver. We are now finished!

    To do in this patch:
    - Make it is experimental. Add testing for this.
    - Add in the autoloading of database driver module dependencies an override to when the driver is "mysqli" to also autoload the "mysql" module. This has the advantage that existing site owners can switch to the MySQLi driver without adding the autoload dependency to the database driver settings to the settings.php file. Do the same for the "autoload" and "namespace" settings. Lets make it easy for site owners to try out the MySQLi driver and revert back to MySQL driver. Add testing for this.

  • 🇬🇧United Kingdom longwave UK

    From a release management point of view the plan in #85 looks good to me. I suggest also opening a plan issue for the mysqli driver with the steps from #85 so we have them documented as what is left after this issue is committed.

  • 🇳🇱Netherlands daffie

    I have talked to @laurii in person and as long as the module is hidden for new site install as long as the module is experimental, am I allowed to remove the "Needs product manager review". Updated comment #85.

  • 🇮🇹Italy mondrake 🇮🇹

    Thanks for #85! Looks clear to me. Re. #85.2, I suggest to add #84

    embed alternative placeholder generation (positional/named) into dynamic queries

    to the meta plan suggested in #86.

  • last update over 1 year ago
    30,522 pass
  • 🇮🇹Italy mondrake 🇮🇹

    Now, we have a failure in the recently fixed SqlContentEntityStorageRevisionDataCleanupTest.

    This is due to this code

        $result = $connection->query('SELECT nid, vid, langcode FROM {node_field_revision} WHERE nid = :nid', [
          'nid' => 8,
        ])->fetchAll();
    

    The arguments in the queries of this test DO NOT HAVE the colon in front, i.e. it should be ':nid' => 8.

    Apparently the PDO-based drivers do not care, even if https://www.drupal.org/docs/drupal-apis/database-api/static-queries#plac... examples clearly indicate placeholders should have the colon.

    I'm not sure whether we should fix the test, or relax NamedPlaceholderConverter to also allow placeholder's array keys not to have a colon.

  • last update over 1 year ago
    30,522 pass
  • 🇮🇹Italy mondrake 🇮🇹

    Done the two points from #85 that are for this issue,

    • Make it is experimental.
    • Add in the autoloading of database driver module dependencies an override to when the driver is "mysqli" to also autoload the "mysql" module.

    but tests are missing. Any idea/reference on how to do those?

  • 🇬🇧United Kingdom catch

    Just a note I have [PP-1] Async database query + fiber support Active open for the next step (or next two steps, depending on how much async support we add in one issue vs. two).

  • last update over 1 year ago
    30,523 pass
  • 🇮🇹Italy mondrake 🇮🇹

    Updated IS with a list of todos remaining here; feel free to add what is necessary.

    Also help on the points would be appreciated.

  • last update over 1 year ago
    Custom Commands Failed
  • 🇮🇹Italy mondrake 🇮🇹

    Re #89 I eventually thought that even if SqlContentEntityStorageRevisionDataCleanupTest should be probably fixed, nevertheless it's better allow parameter arrays with keys missing the initial colon, given that PDO seems to be supporting both. So we can support contrib code that might have missed this detail.

  • last update over 1 year ago
    30,524 pass
  • last update over 1 year ago
    30,527 pass
  • last update over 1 year ago
    Custom Commands Failed
  • last update over 1 year ago
    30,526 pass, 1 fail
  • last update over 1 year ago
    30,527 pass
  • last update over 1 year ago
    30,527 pass
  • Pipeline finished with Success
    over 1 year ago
    #35941
  • Pipeline finished with Success
    over 1 year ago
    Total: 709s
    #35945
  • last update over 1 year ago
    30,527 pass
  • Pipeline finished with Success
    over 1 year ago
    Total: 14865s
    #35953
  • Status changed to Needs review over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    Fixed the collation bug affecting BasicSyntaxTest in MySQL 8. I had to move the test to driver-specific, and for mysqli force conversion of the parameters' values to utf8mb4. Not sure it's the right thing to do but could not find other options. It's an edge case anyway, and could not replicate the issue locally either.

  • last update over 1 year ago
    30,535 pass
  • Pipeline finished with Success
    over 1 year ago
    #37687
  • last update over 1 year ago
    30,536 pass
  • Pipeline finished with Success
    over 1 year ago
    Total: 3349s
    #37700
  • last update over 1 year ago
    30,537 pass
  • 🇮🇹Italy mondrake 🇮🇹

    Added test to ensure module is experimental and hidden.

  • Pipeline finished with Success
    over 1 year ago
    #37745
  • last update over 1 year ago
    30,533 pass, 2 fail
  • Pipeline finished with Success
    over 1 year ago
    #37788
  • Status changed to Needs work over 1 year ago
  • Status changed to Postponed over 1 year ago
  • 🇮🇹Italy mondrake 🇮🇹

    Postponing on the referenced issues, that will make the management of the lack of transactional DDL support consistent across the two drivers.

  • 🇳🇱Netherlands daffie

    Maybe the module should not be hidden and removed from the database selection during the install process. We want to allow site owners to experiment with the mysqli database driver and therefor the module should be visible on the module list.

  • 🇮🇹Italy mondrake 🇮🇹

    Uhm... I don't remember how it works. Definitely, a 'module' extension is different from a 'database_driver' extension. The list of database to be presented for installation is independent from the modules, simply because there's is even no notion of the 'module' at that stage of the installation process. So, it could be the database driver 'mysqli' is presented for (a grassroot) installation if the PHP mysqli extension is available, even if the module itself is hidden. Then, during the (grassroot) installation, the mysqli module is installed automatically (even if hidden). But since it's hidden, there will be no way to uninstall/install it from the UI (that, when it's accessed, presumes all the installation, config and bootstrap to be complete).

    Worth giving a manual test to all this process, I guess.

    Then, if the expectation is to just allow changing 'driver' => 'mysql', to 'driver' => 'mysqli', in settings.php to play around, then we have more to do - i.e. disallow reading the namespace from settings.php in this case (because that will be still the mysql one, and an existing 'namespace' key in the settings takes priority on the 'driver' key), and allow manually installing the mysqli module (so removing the hidden: true from the module yaml).

  • Pipeline finished with Success
    over 1 year ago
    Total: 2308s
    #41659
  • Pipeline finished with Success
    over 1 year ago
    Total: 872s
    #42696
  • Pipeline finished with Failed
    over 1 year ago
    Total: 197s
    #42706
  • Pipeline finished with Failed
    over 1 year ago
    Total: 160s
    #42712
  • 🇮🇹Italy mondrake 🇮🇹

    Thanks for your reviews and input, @daffie. Would you mind checking back the inline comments and replies and not which ones we can mark as solved from your POV, so we can just keep the open ones to work on?

  • Pipeline finished with Success
    over 1 year ago
    #42716
  • Pipeline finished with Failed
    over 1 year ago
    Total: 825s
    #42814
  • Pipeline finished with Success
    over 1 year ago
    Total: 760s
    #42939
  • Pipeline finished with Success
    over 1 year ago
    Total: 745s
    #42961
  • Pipeline finished with Failed
    about 1 year ago
    #72208
  • 🇮🇹Italy mondrake 🇮🇹

    Rebased, with the exception of database dumps which is a pain in the neck to do; that update IMHO should be done just before commit.

  • Pipeline finished with Failed
    about 1 year ago
    Total: 749s
    #72218
  • Pipeline finished with Failed
    about 1 year ago
    Total: 803s
    #72230
  • Pipeline finished with Failed
    about 1 year ago
    #72241
  • Pipeline finished with Failed
    about 1 year ago
    #104420
  • 🇺🇸United States bstan

    The mysqli construction approach has limited ability to define an SSL connection, but leveraging the procedural approach allows you to establish options prior to making a connection.

    For example:

          $mysqli = \mysqli_init();
          // Reusing PDO keys for simplicity and commonality with the `mysql` driver
          $mysqli->ssl_set(
            $connection_options['pdo'][\PDO::MYSQL_ATTR_SSL_KEY],
            $connection_options['pdo'][\PDO::MYSQL_ATTR_SSL_CERT],
            $connection_options['pdo'][\PDO::MYSQL_ATTR_SSL_CA],
            $connection_options['pdo'][\PDO::MYSQL_ATTR_SSL_CAPATH],
            $connection_options['pdo'][\PDO::MYSQL_ATTR_SSL_CIPHER]
          );
    
          $mysqli->real_connect(
            $connection_options['host'],
            $connection_options['username'],
            $connection_options['password'],
            $connection_options['database'] ?? '',
            !empty($connection_options['port']) ? (int) $connection_options['port'] : 3306,
            $connection_options['unix_socket'] ?? '',
            MYSQLI_CLIENT_SSL // This can likely be made through a connection option
          );
    
  • 🇮🇹Italy mondrake 🇮🇹
  • 🇬🇧United Kingdom catch

    Which issue is this postponed on? I can't see anything listed in the issue summary.

  • 🇬🇧United Kingdom catch

    Thanks added a note to the issue summary.

  • Pipeline finished with Failed
    5 months ago
    Total: 150s
    #342644
  • Pipeline finished with Failed
    5 months ago
    Total: 132s
    #343479
  • Pipeline finished with Failed
    5 months ago
    Total: 122s
    #343524
  • Pipeline finished with Success
    5 months ago
    Total: 798s
    #343557
  • Pipeline finished with Success
    4 months ago
    Total: 9485s
    #371556
  • Pipeline finished with Failed
    3 months ago
    Total: 680s
    #382120
  • Pipeline finished with Failed
    3 months ago
    Total: 629s
    #382172
  • Pipeline finished with Failed
    3 months ago
    #382192
  • Pipeline finished with Failed
    about 1 month ago
    Total: 134s
    #435609
  • Pipeline finished with Failed
    about 1 month ago
    Total: 816s
    #435645
  • Pipeline finished with Failed
    about 1 month ago
    Total: 115s
    #435661
  • Pipeline finished with Failed
    about 1 month ago
    Total: 622s
    #435668
  • Pipeline finished with Failed
    about 1 month ago
    Total: 790s
    #435675
  • Status changed to Needs work about 1 month ago
  • 🇮🇹Italy mondrake 🇮🇹

    I am working on 📌 [PP-1] Introduce a StatementBase abstract class Postponed which I believe would be simplifying the Statement class here.

  • Merge request !11355Including StatementBase → (Open) created by mondrake
  • Pipeline finished with Failed
    about 1 month ago
    Total: 115s
    #438716
  • 🇮🇹Italy mondrake 🇮🇹

    MR!1135 incorporates the changes of 📌 [PP-1] Introduce a StatementBase abstract class Postponed and makes evident the simplification possible on the Statement class.

  • Pipeline finished with Success
    about 1 month ago
    Total: 457s
    #438769
  • Pipeline finished with Failed
    about 1 month ago
    Total: 119s
    #439853
  • Pipeline finished with Success
    about 1 month ago
    Total: 493s
    #440028
  • 🇳🇱Netherlands daffie

    We need the following code change to fix the failing Drupal\Tests\help\Functional\HelpTest.

    diff --git a/core/modules/mysqli/src/Hook/MysqliHooks.php b/core/modules/mysqli/src/Hook/MysqliHooks.php
    index 3f7e6683704..5fae187d16c 100644
    --- a/core/modules/mysqli/src/Hook/MysqliHooks.php
    +++ b/core/modules/mysqli/src/Hook/MysqliHooks.php
    @@ -17,7 +17,7 @@ class MysqliHooks {
        * Implements hook_help().
        */
       #[Hook('help')]
    -  public function help($route_name, RouteMatchInterface $route_match): array {
    +  public function help($route_name, RouteMatchInterface $route_match): ?string {
         switch ($route_name) {
           case 'help.page.mysqli':
             $output = '';
    @@ -26,7 +26,7 @@ public function help($route_name, RouteMatchInterface $route_match): array {
             return $output;
     
         }
    -    return [];
    +    return NULL;
       }
     
     }
    
  • 🇳🇱Netherlands daffie

    The NamedPlaceholderConverterTest has been added. Therefore removing the tag "Needs tests".

  • 🇳🇱Netherlands daffie

    A lot of update tests are failing because they use core/modules/system/tests/fixtures/update/drupal-10.3.0.bare.standard.php.gz or core/modules/system/tests/fixtures/update/drupal-10.3.0.filled.standard.php.gz and the module mysqli has not been included in those fixtures.

  • 🇳🇱Netherlands daffie

    The Drupal\Core\EventSubscriber\ConfigImportSubscriber can be fixed with the following code change:

    diff --git a/core/lib/Drupal/Core/EventSubscriber/ConfigImportSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ConfigImportSubscriber.php
    index bc85f19ecaf..5b34360f9da 100644
    --- a/core/lib/Drupal/Core/EventSubscriber/ConfigImportSubscriber.php
    +++ b/core/lib/Drupal/Core/EventSubscriber/ConfigImportSubscriber.php
    @@ -102,6 +103,7 @@ protected function validateModules(ConfigImporter $config_importer) {
         // Get the install profile from the site's configuration.
         $current_core_extension = $config_importer->getStorageComparer()->getTargetStorage()->read('core.extension');
         $install_profile = $current_core_extension['profile'] ?? NULL;
    +    $database_driver_module = \Drupal::database()->getProvider();
         $new_install_profile = $core_extension['profile'] ?? NULL;
     
         // Ensure the profile is not changing.
    @@ -159,7 +161,7 @@ protected function validateModules(ConfigImporter $config_importer) {
         $uninstalls = $config_importer->getExtensionChangelist('module', 'uninstall');
         foreach ($uninstalls as $module) {
           foreach (array_keys($module_data[$module]->required_by) as $dependent_module) {
    -        if ($module_data[$dependent_module]->status && !in_array($dependent_module, $uninstalls, TRUE) && $dependent_module !== $install_profile) {
    +        if ($module_data[$dependent_module]->status && !in_array($dependent_module, $uninstalls, TRUE) && !in_array($dependent_module, [$install_profile, $database_driver_module], TRUE)) {
               $module_name = $module_data[$module]->info['name'];
               $dependent_module_name = $module_data[$dependent_module]->info['name'];
               $config_importer->logError($this->t('Unable to uninstall the %module module since the %dependent_module module is installed.', ['%module' => $module_name, '%dependent_module' => $dependent_module_name]));
    
  • 🇮🇹Italy mondrake 🇮🇹
  • Pipeline finished with Failed
    13 days ago
    Total: 221s
    #455423
  • 🇮🇹Italy mondrake 🇮🇹

    mondrake changed the visibility of the branch 3259709-create-the-database to hidden.

  • 🇮🇹Italy mondrake 🇮🇹

    Made some progress. This is now postponed on 📌 [PP-1] Introduce a StatementBase abstract class Postponed .

  • Pipeline finished with Failed
    13 days ago
    Total: 723s
    #455424
  • Pipeline finished with Success
    12 days ago
    Total: 2363s
    #455867
  • Pipeline finished with Failed
    5 days ago
    Total: 166s
    #461646
  • 🇮🇹Italy mondrake 🇮🇹
  • Pipeline finished with Failed
    5 days ago
    Total: 133s
    #461668
  • Pipeline finished with Failed
    5 days ago
    Total: 185s
    #461678
  • Pipeline finished with Success
    5 days ago
    Total: 796s
    #461691
  • Pipeline finished with Canceled
    5 days ago
    Total: 71s
    #461710
  • Pipeline finished with Failed
    5 days ago
    Total: 151s
    #461712
  • Pipeline finished with Failed
    5 days ago
    Total: 183s
    #461718
  • Pipeline finished with Failed
    5 days ago
    Total: 150s
    #461721
  • Pipeline finished with Failed
    5 days ago
    Total: 124s
    #461722
  • Pipeline finished with Failed
    5 days ago
    Total: 141s
    #461727
  • 🇮🇹Italy mondrake 🇮🇹

    For review. Fixing fixtures for update tests is a real PITA, so please let's focus on other items for now and let's address them when we are all comfortable with all the rest. Thanks.

  • Pipeline finished with Failed
    5 days ago
    Total: 120s
    #461731
  • Pipeline finished with Success
    5 days ago
    Total: 2347s
    #461735
  • 🇮🇹Italy mondrake 🇮🇹

    Filed 🐛 @phpstan-ignore-next-line annotation collides with PHPCS Active in the Coder issue queue.

  • 🇮🇹Italy mondrake 🇮🇹

    There's a smarter way than adding a Schema class override. Working on it.

  • Pipeline finished with Failed
    4 days ago
    Total: 190s
    #461896
  • 🇮🇹Italy mondrake 🇮🇹

    Introduced a SchemaPrimaryKeyMustBeDroppedException to abstract the error and drop the Schema class.

  • Pipeline finished with Success
    4 days ago
    Total: 642s
    #461903
  • Pipeline finished with Failed
    4 days ago
    Total: 183s
    #462254
  • 🇮🇹Italy mondrake 🇮🇹

    Pre-8.0 MySql ANSI modes were removed in 📌 Update INSTALL.txt and friends with Drupal 11 platform requirements Postponed , adjusted Connection::__construct() accordingly.

  • Pipeline finished with Success
    4 days ago
    Total: 869s
    #462259
  • 🇮🇹Italy mondrake 🇮🇹

    Charset toggling was removed in 📌 [policy] Decide what to do with MYSQLND_MINIMUM_VERSION and LIBMYSQLCLIENT_MINIMUM_VERSION Active , adjusted Connection::open() accordingly.

  • Pipeline finished with Success
    4 days ago
    Total: 1270s
    #462278
  • 🇮🇹Italy mondrake 🇮🇹

    Filed https://github.com/drush-ops/drush/issues/6266 in drush issue queue; drush will need some adjustment to make it work with this module when it will be committed to core.

  • 🇳🇱Netherlands daffie
  • Pipeline finished with Failed
    3 days ago
    #463533
  • 🇮🇹Italy mondrake 🇮🇹

    @daffie

    A number of tests only are run for certain database drivers. When such a test is run for the "mysql" database driver, the "mysqli" database driver needs to be added.

    Are you talking about the kernel driver-specific tests? If so, they are run for mysqli based on inheritance from the mysql ones, we need not do anything. If something else, can you elaborate?

  • 🇮🇹Italy mondrake 🇮🇹

    Added a mock CR... please help filling it in :)

  • 🇮🇹Italy mondrake 🇮🇹

    I know we have made this module "hidden", but should we have installertests?

    I am not familiar with such tests; can anybody point me to an example to work from, in case?

  • Pipeline finished with Success
    3 days ago
    #463546
  • 🇳🇱Netherlands daffie

    I am not familiar with such tests; can anybody point me to an example to work from, in case?

    The test in the namespace Drupal\FunctionalTests\Installer

    Added a mock CR... please help filling it in :)

    I will do that.

    For #131: For example core/modules/system/tests/src/Functional/System/DatabaseDriverProvidedByModuleTest.php

  • Pipeline finished with Running
    2 days ago
    #463628
  • Pipeline finished with Success
    2 days ago
    Total: 923s
    #463629
  • Pipeline finished with Failed
    2 days ago
    Total: 373s
    #463682
  • Pipeline finished with Failed
    2 days ago
    Total: 231s
    #463685
  • Pipeline finished with Success
    2 days ago
    Total: 1094s
    #463692
  • 🇮🇹Italy mondrake 🇮🇹

    #134 I looked at tests in that namespace, but still I miss what such a test should be written for, here.

  • 🇳🇱Netherlands daffie

    #134 I looked at tests in that namespace, but still I miss what such a test should be written for, here.

    Yes, you are right. The mysqli driver is hidden and therefore does not show up in the install process. My bad.

    I have updated the CR.

    Now we just need to update the fixtures and the CI pipeline to be green, then it is RTBC for me.

  • 🇮🇹Italy mondrake 🇮🇹

    Thanks @daffie

    Sanity check on #136, before I get hands dirty with db fixtures: apart from Drupal\Tests\mysql\Functional\RequirementsTest, all the other failing tests are update tests that rely on the 10.3 db dump fixtures.

    Are we really sure we want to rewrite history and pretend a mysqli module was available in 10.3? In real life no one will ever update from a 10.3 install that already has mysqli installed. So is it right to rewrite history to make the tests happy, or should we just skip somehow the tests?

    The situation in #71 was different - there we had the 'modulification' of db drivers that were already present beforehand, here we are introducing a totally new module.

    I'll ping @catch for an opinion here too

  • 🇳🇱Netherlands daffie

    I am fine with skipping Drupal\Tests\mysql\Functional\RequirementsTest.

  • 🇮🇹Italy mondrake 🇮🇹

    #138 well we are missing a c/p of mysql.install in this module so that the requirements can be rendered... that file was added to core 2 years ago in #2733675: Warning when mysql is not set to READ-COMMITTED but did not get its way into the mysqli module. So the test failure is legitimate I think and we should fix it.

  • Pipeline finished with Running
    2 days ago
    #464338
  • 🇮🇹Italy mondrake 🇮🇹

    In https://git.drupalcode.org/project/drupal/-/merge_requests/11355/diffs?c..., we are skipping the update tests if the driver is mysqli and the database dump is from drupal-10.3.0.

    I think this is more realistic and avoids tampering with database dump fixtures.

  • Pipeline finished with Success
    1 day ago
    Total: 1818s
    #464503
  • 🇳🇱Netherlands daffie

    @mondrake: Skipping the exiting update tests is not a problem. Only we need those fixtures for all the future update tests. Do you want me to do it?

  • 🇮🇹Italy mondrake 🇮🇹

    The future update tests, relevant for mysqli, will be those introduced to update from 11.2 to above, right? Because those introduced before 11.2 are not relevant since the module is not released yet.

    Then the task will be to add a new database dump for 11.2.0, at that moment including the install of myqli.

    But that's a separate issue from here if we agree to proceed as per #140. A new dump will get its own 11.2.0 version name and this check

    
          // Determine the version of the database dump if specified.
          $matches = [];
          $dumpVersion = preg_match('/drupal-(\d+.\d+.\d+)/', $file, $matches) === 1 ? $matches[1] : NULL;
    
          // If the db driver is mysqli, we do not need to run the update tests for
          // db dumps prior to 11.2 when the module was introduced.
          if (Database::getConnection()->getProvider === 'mysqli' && $dumpVersion && version_compare($dumpVersion, '11.2.0', '<')) {
            $this->markTestSkipped("The mysqli driver was introduced in Drupal 11.2, skip update tests from database at version {$dumpVersion}");
          }
    
    

    will no longer skip the test, if the test requires that dump. But it will keep skipping tests using the 10.3.0 dump.

    In other words: if we do this, we do not need to change the fixtures now. We can wait for 11.2 to be released.

  • Pipeline finished with Success
    1 day ago
    Total: 688s
    #464561
  • Pipeline finished with Canceled
    1 day ago
    Total: 493s
    #464570
  • Pipeline finished with Success
    1 day ago
    Total: 673s
    #464595
  • 🇮🇹Italy mondrake 🇮🇹

    Pipelines are green, yay.

  • 🇳🇱Netherlands daffie

    All code changes look good to me.
    The new database driver is marked as "hidden" and therefore does not show up on in the install process.
    The new database driver is marked as "experimental" and therefore not yet fully supported.
    The CI pipeline is green for the mysql and the new mysqli drivers.
    The IS and the CR are in order.
    For me it is RTBC.

    For the committer: As the database driver will be added to Drupal 11.2, all update test are skipped. There are no Drupal sites <11.2 with the new MySQLi database driver. The problem is that the first person who want to add an update test to Drupal 11.3 will have to create new fixtures which need to include the mysqli module.

  • 🇫🇷France andypost

    Rtbc++ for db dumps there's follow-up 📌 Create standard steps for creating database dumps Active

Production build 0.71.5 2024