Skip to content

Commit

Permalink
bugfix: deallocate mysqli prepared statement
Browse files Browse the repository at this point in the history
Long running processes might hit the `max_prepared_stmt_count` due not
deallocating the statement correctly.
  • Loading branch information
petermein committed Jan 6, 2025
1 parent 0ac3d64 commit d41af03
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 4 deletions.
17 changes: 14 additions & 3 deletions src/Driver/Mysqli/Result.php
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,11 @@ final class Result implements ResultInterface
{
private mysqli_stmt $statement;

/**
* Holds a reference to the Statement that created the result
*/
private ?Statement $statementReference = null; // @phpstan-ignore property.onlyWritten

/**
* Whether the statement result has columns. The property should be used only after the result metadata
* has been fetched ({@see $metadataFetched}). Otherwise, the property value is undetermined.
Expand All @@ -40,11 +45,17 @@ final class Result implements ResultInterface
/**
* @internal The result can be only instantiated by its driver connection or statement.
*
* @param Statement|null $statementReference This reference is to not trigger destruction
* of the Statement object too early.
*
* @throws Exception
*/
public function __construct(mysqli_stmt $statement)
{
$this->statement = $statement;
public function __construct(

Check warning on line 53 in src/Driver/Mysqli/Result.php

View check run for this annotation

Codecov / codecov/patch

src/Driver/Mysqli/Result.php#L53

Added line #L53 was not covered by tests
mysqli_stmt $statement,
?Statement $statementReference = null,
) {
$this->statement = $statement;
$this->statementReference = $statementReference;

Check warning on line 58 in src/Driver/Mysqli/Result.php

View check run for this annotation

Codecov / codecov/patch

src/Driver/Mysqli/Result.php#L57-L58

Added lines #L57 - L58 were not covered by tests

$meta = $statement->result_metadata();

Expand Down
7 changes: 6 additions & 1 deletion src/Driver/Mysqli/Statement.php
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,11 @@ public function __construct(mysqli_stmt $stmt)
$this->boundValues = array_fill(1, $paramCount, null);
}

public function __destruct()

Check warning on line 64 in src/Driver/Mysqli/Statement.php

View check run for this annotation

Codecov / codecov/patch

src/Driver/Mysqli/Statement.php#L64

Added line #L64 was not covered by tests
{
@$this->stmt->close();

Check warning on line 66 in src/Driver/Mysqli/Statement.php

View check run for this annotation

Codecov / codecov/patch

src/Driver/Mysqli/Statement.php#L66

Added line #L66 was not covered by tests
}

/**
* @deprecated Use {@see bindValue()} instead.
*
Expand Down Expand Up @@ -159,7 +164,7 @@ public function execute($params = null): ResultInterface
throw StatementError::new($this->stmt);
}

return new Result($this->stmt);
return new Result($this->stmt, $this);

Check warning on line 167 in src/Driver/Mysqli/Statement.php

View check run for this annotation

Codecov / codecov/patch

src/Driver/Mysqli/Statement.php#L167

Added line #L167 was not covered by tests
}

/**
Expand Down
50 changes: 50 additions & 0 deletions tests/Functional/Driver/Mysqli/StatementTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
<?php

declare(strict_types=1);

namespace Doctrine\DBAL\Tests\Functional\Driver\Mysqli;

use Doctrine\DBAL\Driver\Mysqli\Statement;
use Doctrine\DBAL\Statement as WrapperStatement;
use Doctrine\DBAL\Tests\FunctionalTestCase;
use Doctrine\DBAL\Tests\TestUtil;
use Error;
use PHPUnit\Framework\Attributes\RequiresPhpExtension;

Check failure on line 12 in tests/Functional/Driver/Mysqli/StatementTest.php

View workflow job for this annotation

GitHub Actions / Coding Standards / Coding Standards (PHP: 8.4)

Type PHPUnit\Framework\Attributes\RequiresPhpExtension is not used in this file.
use ReflectionProperty;

/** @requires extension mysqli */
class StatementTest extends FunctionalTestCase
{
protected function setUp(): void
{
parent::setUp();

if (TestUtil::isDriverOneOf('mysqli')) {
return;
}

self::markTestSkipped('This test requires the mysqli driver.');
}

public function testStatementsAreDeallocatedProperly(): void
{
$statement = $this->connection->prepare('SELECT 1');

$property = new ReflectionProperty(WrapperStatement::class, 'stmt');
$property->setAccessible(true);

$driverStatement = $property->getValue($statement);

$mysqliProperty = new ReflectionProperty(Statement::class, 'stmt');
$mysqliProperty->setAccessible(true);

$mysqliStatement = $mysqliProperty->getValue($driverStatement);

unset($statement, $driverStatement);

$this->expectException(Error::class);
$this->expectExceptionMessage('mysqli_stmt object is already closed');

$mysqliStatement->execute();
}
}

0 comments on commit d41af03

Please sign in to comment.