Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 969c47f

Browse files
committedDec 8, 2023
Restore binary offsets of PDOStatement parameters
Entity identifiers that are PHP resource streams need to work for all references, not just the first one. PDOStatement->execute is reading the binary resources but not restoring the original offsets. No other code is restoring these streams to their original position so that they can be reused. Examples of failures include empty collections on read (the first lazy load collection on an entity populates correctly, the second is empty) and foreign-key violations while persisting changes (the first entity join produces the correct SQL, the second has no data to read and the FK is violated with the missing binary data). Making this change as close as possible to the external code that moves the stream pointer eliminates the need to do this in calling code. Resource offsets are retrieved immediately before execute in case they change between the bindValue and execute calls. The request was originally for the PDO driver but IBMDB2, Mysql, and PgSQL drivers are also covered. Other drivers will likely also need work. No attempt has been made to fix the deprecated bindParam code path. I do not believe this is called by the current Doctrine code, and is regardless much harder to patch because the reference variables can be replaced during execute, so the original resources may no longer be available to restore after the call. A functional test was added for bindValue and a resource with a seekable position. #5895
1 parent 7c4aa97 commit 969c47f

File tree

5 files changed

+180
-9
lines changed

5 files changed

+180
-9
lines changed
 

‎src/Driver/IBMDB2/Statement.php

+25-1
Original file line numberDiff line numberDiff line change
@@ -10,12 +10,15 @@
1010
use Doctrine\DBAL\Driver\Statement as StatementInterface;
1111
use Doctrine\DBAL\ParameterType;
1212
use Doctrine\Deprecations\Deprecation;
13+
use Throwable;
1314

1415
use function assert;
1516
use function db2_bind_param;
1617
use function db2_execute;
1718
use function error_get_last;
1819
use function fclose;
20+
use function fseek;
21+
use function ftell;
1922
use function func_num_args;
2023
use function is_int;
2124
use function is_resource;
@@ -28,6 +31,7 @@
2831
use const DB2_LONG;
2932
use const DB2_PARAM_FILE;
3033
use const DB2_PARAM_IN;
34+
use const SEEK_SET;
3135

3236
final class Statement implements StatementInterface
3337
{
@@ -213,8 +217,28 @@ private function createTemporaryFile()
213217
*/
214218
private function copyStreamToStream($source, $target): void
215219
{
220+
$resetTo = false;
221+
if (stream_get_meta_data($source)['seekable']) {
222+
$resetTo = ftell($source);
223+
}
224+
216225
if (@stream_copy_to_stream($source, $target) === false) {
217-
throw CannotCopyStreamToStream::new(error_get_last());
226+
$copyToStreamError = error_get_last();
227+
if ($resetTo !== false) {
228+
try {
229+
fseek($source, $resetTo, SEEK_SET);
230+
} catch (Throwable $e) {
231+
// Swallow, we want the original exception from stream_copy_to_stream
232+
}
233+
}
234+
235+
throw CannotCopyStreamToStream::new($copyToStreamError);
218236
}
237+
238+
if ($resetTo === false) {
239+
return;
240+
}
241+
242+
fseek($source, $resetTo, SEEK_SET);
219243
}
220244
}

‎src/Driver/Mysqli/Statement.php

+23-7
Original file line numberDiff line numberDiff line change
@@ -19,11 +19,16 @@
1919
use function count;
2020
use function feof;
2121
use function fread;
22+
use function fseek;
23+
use function ftell;
2224
use function func_num_args;
2325
use function get_resource_type;
2426
use function is_int;
2527
use function is_resource;
2628
use function str_repeat;
29+
use function stream_get_meta_data;
30+
31+
use const SEEK_SET;
2732

2833
final class Statement implements StatementInterface
2934
{
@@ -213,15 +218,26 @@ private function bindTypedParameters(): void
213218
private function sendLongData(array $streams): void
214219
{
215220
foreach ($streams as $paramNr => $stream) {
216-
while (! feof($stream)) {
217-
$chunk = fread($stream, 8192);
221+
$resetTo = false;
222+
if (stream_get_meta_data($stream)['seekable']) {
223+
$resetTo = ftell($stream);
224+
}
218225

219-
if ($chunk === false) {
220-
throw FailedReadingStreamOffset::new($paramNr);
221-
}
226+
try {
227+
while (! feof($stream)) {
228+
$chunk = fread($stream, 8192);
222229

223-
if (! $this->stmt->send_long_data($paramNr - 1, $chunk)) {
224-
throw StatementError::new($this->stmt);
230+
if ($chunk === false) {
231+
throw FailedReadingStreamOffset::new($paramNr);
232+
}
233+
234+
if (! $this->stmt->send_long_data($paramNr - 1, $chunk)) {
235+
throw StatementError::new($this->stmt);
236+
}
237+
}
238+
} finally {
239+
if ($resetTo !== false) {
240+
fseek($stream, $resetTo, SEEK_SET);
225241
}
226242
}
227243
}

‎src/Driver/PDO/Statement.php

+87
Original file line numberDiff line numberDiff line change
@@ -7,17 +7,27 @@
77
use Doctrine\DBAL\Driver\Statement as StatementInterface;
88
use Doctrine\DBAL\ParameterType;
99
use Doctrine\Deprecations\Deprecation;
10+
use PDO;
1011
use PDOException;
1112
use PDOStatement;
1213

1314
use function array_slice;
15+
use function fseek;
16+
use function ftell;
1417
use function func_get_args;
1518
use function func_num_args;
19+
use function is_resource;
20+
use function stream_get_meta_data;
21+
22+
use const SEEK_SET;
1623

1724
final class Statement implements StatementInterface
1825
{
1926
private PDOStatement $stmt;
2027

28+
/** @var mixed[]|null */
29+
private ?array $paramResources = null;
30+
2131
/** @internal The statement can be only instantiated by its driver connection. */
2232
public function __construct(PDOStatement $stmt)
2333
{
@@ -43,6 +53,9 @@ public function bindValue($param, $value, $type = ParameterType::STRING)
4353
}
4454

4555
$pdoType = ParameterTypeMap::convertParamType($type);
56+
if ($pdoType === PDO::PARAM_LOB) {
57+
$this->trackParamResource($value);
58+
}
4659

4760
try {
4861
return $this->stmt->bindValue($param, $value, $pdoType);
@@ -126,12 +139,86 @@ public function execute($params = null): ResultInterface
126139
);
127140
}
128141

142+
$resourceOffsets = $this->getResourceOffsets();
129143
try {
130144
$this->stmt->execute($params);
131145
} catch (PDOException $exception) {
132146
throw Exception::new($exception);
147+
} finally {
148+
if ($resourceOffsets !== null) {
149+
$this->restoreResourceOffsets($resourceOffsets);
150+
}
133151
}
134152

135153
return new Result($this->stmt);
136154
}
155+
156+
/**
157+
* Track a binary parameter reference at binding time. These
158+
* are cached for later analysis by the getResourceOffsets.
159+
*
160+
* @param mixed $resource
161+
*/
162+
private function trackParamResource($resource): void
163+
{
164+
if (! is_resource($resource)) {
165+
return;
166+
}
167+
168+
$this->paramResources ??= [];
169+
$this->paramResources[] = $resource;
170+
}
171+
172+
/**
173+
* Determine the offset that any resource parameters needs to be
174+
* restored to after the statement is executed. Call immediately
175+
* before execute (not during bindValue) to get the most accurate offset.
176+
*
177+
* @return int[]|null Return offsets to restore if needed. The array may be sparse.
178+
*/
179+
private function getResourceOffsets(): ?array
180+
{
181+
if ($this->paramResources === null) {
182+
return null;
183+
}
184+
185+
$resourceOffsets = null;
186+
foreach ($this->paramResources as $index => $resource) {
187+
$position = false;
188+
if (stream_get_meta_data($resource)['seekable']) {
189+
$position = ftell($resource);
190+
}
191+
192+
if ($position === false) {
193+
continue;
194+
}
195+
196+
$resourceOffsets ??= [];
197+
$resourceOffsets[$index] = $position;
198+
}
199+
200+
if ($resourceOffsets === null) {
201+
$this->paramResources = null;
202+
}
203+
204+
return $resourceOffsets;
205+
}
206+
207+
/**
208+
* Restore resource offsets moved by PDOStatement->execute
209+
*
210+
* @param int[]|null $resourceOffsets The offsets returned by getResourceOffsets.
211+
*/
212+
private function restoreResourceOffsets(?array $resourceOffsets): void
213+
{
214+
if ($resourceOffsets === null || $this->paramResources === null) {
215+
return;
216+
}
217+
218+
foreach ($resourceOffsets as $index => $offset) {
219+
fseek($this->paramResources[$index], $offset, SEEK_SET);
220+
}
221+
222+
$this->paramResources = null;
223+
}
137224
}

‎src/Driver/PgSQL/Statement.php

+20-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010
use TypeError;
1111

1212
use function assert;
13+
use function fseek;
14+
use function ftell;
1315
use function func_num_args;
1416
use function get_class;
1517
use function gettype;
@@ -26,6 +28,9 @@
2628
use function pg_send_execute;
2729
use function sprintf;
2830
use function stream_get_contents;
31+
use function stream_get_meta_data;
32+
33+
use const SEEK_SET;
2934

3035
final class Statement implements StatementInterface
3136
{
@@ -151,10 +156,24 @@ public function execute($params = null): Result
151156
switch ($this->parameterTypes[$parameter]) {
152157
case ParameterType::BINARY:
153158
case ParameterType::LARGE_OBJECT:
159+
$isResource = is_resource($value);
160+
$resource = $value;
161+
$resetTo = false;
162+
if ($isResource) {
163+
if (stream_get_meta_data($resource)['seekable']) {
164+
$resetTo = ftell($resource);
165+
}
166+
}
167+
154168
$escapedParameters[] = $value === null ? null : pg_escape_bytea(
155169
$this->connection,
156-
is_resource($value) ? stream_get_contents($value) : $value,
170+
$isResource ? stream_get_contents($value) : $value,
157171
);
172+
173+
if ($resetTo !== false) {
174+
fseek($resource, $resetTo, SEEK_SET);
175+
}
176+
158177
break;
159178
default:
160179
$escapedParameters[] = $value;

‎tests/Functional/BlobTest.php

+25
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@
1010
use Doctrine\DBAL\Types\Types;
1111

1212
use function fopen;
13+
use function fseek;
14+
use function ftell;
15+
use function fwrite;
1316
use function str_repeat;
1417
use function stream_get_contents;
1518

@@ -198,6 +201,28 @@ public function testBlobBindingDoesNotOverwritePrevious(): void
198201
self::assertEquals(['test1', 'test2'], $actual);
199202
}
200203

204+
public function testBindValueResetsStream(): void
205+
{
206+
if (TestUtil::isDriverOneOf('oci8')) {
207+
self::markTestIncomplete('The oci8 driver does not support stream resources as parameters');
208+
}
209+
210+
$stmt = $this->connection->prepare(
211+
"INSERT INTO blob_table(id, clobcolumn, blobcolumn) VALUES (1, 'ignored', ?)",
212+
);
213+
214+
$stream = fopen('php://temp', 'rb+');
215+
fwrite($stream, 'a test');
216+
fseek($stream, 2);
217+
$stmt->bindValue(1, $stream, ParameterType::LARGE_OBJECT);
218+
219+
$stmt->execute();
220+
221+
self::assertEquals(2, ftell($stream), 'Resource parameter should be reset to position before execute.');
222+
223+
$this->assertBlobContains('test');
224+
}
225+
201226
private function assertBlobContains(string $text): void
202227
{
203228
[, $blobValue] = $this->fetchRow();

0 commit comments

Comments
 (0)
Please sign in to comment.