-
Notifications
You must be signed in to change notification settings - Fork 7.9k
Description
Description
A user reported an issue to me where they're trying to insert binary data (a PNG file, bound as a PARAM_LOB
, inserting into a varbinary
column) into an SQL Server database, and unexpectedly get a syntax error back.
It looks like PDO_DBLIB doesn't actually implement binding (no param hook is implemented), so it relies on PDO to do substitution and send the fully substituted query string back to the server (I assume this is a FreeTDS, or TDS protocol limitation. It's been a long, long time since I touched SQL Server...), instead of sending the query and parameters separate. The PDO parsing defers to the driver's quoter to turn the value into a string, escaping things like '
and \
. However, it seems this isn't sufficient with binary data that could have things like nulls in the middle of the string.
The following code:
<?php
try {
$image1 = base64_decode("..."); // a PNG image in this case
$pdo = new PDO("dblib:host=SQLTEST;dbname=test", "sa", "...");
if ($pdo) {
$sql = "insert into dbo.TicketImage (TicketID, Image1, Image2) VALUES (168, ?, NULL)";
$stmt = $pdo->prepare($sql);
$stmt->bindParam(1, $image1, PDO::PARAM_LOB);
$stmt->execute();
}
} catch (PDOException $e) {
var_dump($e);
}
Resulted in this output:
object(PDOException)#3 (8) {
["message":protected]=>
string(165) "SQLSTATE[HY000]: General error: 20018 Incorrect syntax near ''. [20018] (severity 15) [insert into dbo.TicketImage (TicketID, Image1, Image2) VALUES (168, '�PNG
�
]"
["string":"Exception":private]=>
string(0) ""
["code":protected]=>
string(5) "HY000"
["file":protected]=>
string(59) "/home/calvin/src/sqlserver/sql server binary blob issue.php"
["line":protected]=>
int(16)
["trace":"Exception":private]=>
array(1) {
[0]=>
array(5) {
["file"]=>
string(59) "/home/calvin/src/sqlserver/sql server binary blob issue.php"
["line"]=>
int(16)
["function"]=>
string(7) "execute"
["class"]=>
string(12) "PDOStatement"
["type"]=>
string(2) "->"
}
}
["previous":"Exception":private]=>
NULL
["errorInfo"]=>
array(5) {
[0]=>
string(5) "HY000"
[1]=>
int(20018)
[2]=>
string(127) "Incorrect syntax near ''. [20018] (severity 15) [insert into dbo.TicketImage (TicketID, Image1, Image2) VALUES (168, '�PNG
�
]"
[3]=>
int(-1)
[4]=>
int(15)
}
}
But I expected this output instead:
// success...
Other details
SQL Server version:
Microsoft SQL Server 2016 (SP2) (KB4052908) - 13.0.5026.0 (X64)
Mar 18 2018 09:11:49
Copyright (c) Microsoft Corporation
Express Edition (64-bit) on Windows Server 2016 Standard 10.0 <X64> (Build 14393: ) (Hypervisor)
FreeTDS info:
Compile-time settings (established with the "configure" script)
Version: freetds v1.3.3
freetds.conf directory: /etc
MS db-lib source compatibility: yes
Sybase binary compatibility: yes
Thread safety: yes
iconv library: yes
TDS version: auto
iODBC: no
unixodbc: yes
SSPI "trusted" logins: no
Kerberos: yes
OpenSSL: no
GnuTLS: yes
MARS: yes
The provided table:
CREATE TABLE [dbo].[TicketImage](
[TicketID] [int] NOT NULL,
[Image1] [varbinary](max) NULL,
[Image2] [varbinary](max) NULL,
CONSTRAINT [PK_TicketImage] PRIMARY KEY CLUSTERED
(
[TicketID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]
GO
PHP Version
PHP 8.1.13
Operating System
Fedora 37
Activity
NattyNarwhal commentedon Jan 13, 2023
Two ideas:
PARAM_BINARY
type), presume that it's binary data, and instead of escaping it as a string, escape it as a binary literal (0x
and the hex bytes). This works because the parsing in PDO defers quoting to the driver and plops that right in where the string would have gone instead. I've included a diff for a very simple implementation of this approach which seems to resolve the issue for me. If it makes sense, I can create a PR and clean up the code.For both changes, I'm not sure what the impact would be.
NattyNarwhal commentedon Jan 13, 2023
cc @cmb69 since he probably knows the most about SQL Server here
adambaratz commentedon Jan 15, 2023
This has been a long-standing behavior of this extension. It's been a while since I looked at this part of the code, but I think it might be a FreeTDS limitation. I'd suggest trying an ODBC driver and extension if you need this functionality. Microsoft has their own Linux driver which I'd recommend checking out, been available for a number of years now.
cmb69 commentedon Jan 16, 2023
Well, maybe, but I've never used (or had a closer look at) FreeTDS or pdo_dblib.
Right. Maybe we should amend the existing documentation to not only recommend that driver on Windows:
Anyhow, would that be an option for that user?
I think that this makes sense, unless we classify this as WONTFIX (in which case we should improve the documentation).
NattyNarwhal commentedon Jan 16, 2023
The platform the user is running on (IBM i) doesn't have Microsoft's ODBC driver or PHP extension available, so FreeTDS is the only option.
Makes sense. It does change the behaviour, but the previous behaviour seemed pretty broken for the context anyways.