在PHP中通过ODBC(使用PDO)查询Snowflake返回错误数据
我正在尝试通过PHP(CentOS 6和7)中的ODBC驱动程序连接到Snowflake,并使用PDO进行更轻松的访问。
ODBC驱动程序似乎已安装并且工作正常,因为PHP中的本机ODBC函数工作得很好:
$dsn = "Driver=SnowflakeDSIIDriver;Server=" . SNOWFLAKE_HOST;
$dsn .= ";Account=" . SNOWFLAKE_ACCOUNT;
$dsn .= ";Port=" . SNOWFLAKE_PORT;
$dsn .= ";Schema=" . SNOWFLAKE_SCHEMA;
$dsn .= ";Warehouse=" . SNOWFLAKE_WAREHOUSE;
$dsn .= ";Database=" . SNOWFLAKE_DATABASAE;
$conn_id = odbc_connect($dsn, SNOWFLAKE_USER, SNOWFLAKE_PASSWORD);
odbc_exec($conn_id, "USE WAREHOUSE " . SNOWFLAKE_WAREHOUSE);
$res = odbc_exec($conn_id, 'SHOW TABLES IN SCHEMA ' . SNOWFLAKE_SCHEMA . ';');
if ($res) {
print "Tables in schema
";
while($row = odbc_fetch_array($res)) {
print_r($row);
}
}
$res = odbc_exec($conn_id, 'SELECT * FROM TEST;');
if ($res) {
print "Test table content
";
while($row = odbc_fetch_array($res)) {
print_r($row);
}
}
退货
Tables in schema
Array
(
[created_on] => 2015-09-09 17:34:43.517000
[name] => TEST
[database_name] => TESTSUITE
[schema_name] => TESTSUITE
[kind] => TRANSIENT
[comment] =>
[cluster_by] =>
[rows] => 3
[bytes] => 8192
[owner] => TESTSUITE
[account_name] => ****
[retention_time] => 1
)
Test table content
Array
(
[C1] => c
[C2] =>
)
Array
(
[C1] => a
[C2] =>
)
Array
(
[C1] => a
[C2] =>
)
这正是我直接查询数据库时得到的结果。
但是当我想使用PDO时,结果变得很奇怪。
$dsn = "Driver=SnowflakeDSIIDriver;Server=" . SNOWFLAKE_HOST;
$dsn .= ";Account=" . SNOWFLAKE_ACCOUNT;
$dsn .= ";Port=" . SNOWFLAKE_PORT;
$dsn .= ";Schema=" . SNOWFLAKE_SCHEMA;
$dsn .= ";Database=" . SNOWFLAKE_DATABASE;
$dsn .= ";Warehouse=" . SNOWFLAKE_WAREHOUSE;
$pdo = new PDO("odbc:" . $dsn, SNOWFLAKE_USER, SNOWFLAKE_PASSWORD, array(PDO::ATTR_CURSOR => PDO::CURSOR_SCROLL));
$pdo->exec("USE WAREHOUSE " . SNOWFLAKE_WAREHOUSE);
$query = 'SHOW TABLES IN SCHEMA ' . SNOWFLAKE_SCHEMA . ';';
$statement = $pdo->query($query);
print "Tables in schema
";
print "Rows: " . $statement->rowCount() . "
";
while ($row = $statement->fetch(PDO::FETCH_ASSOC, PDO::FETCH_ORI_NEXT)) {
print_r($row);
}
$query = 'SELECT * FROM TEST;';
$statement = $pdo->prepare($query);
$statement->execute();
print "Test table content
";
print "Rows: " . $statement->rowCount() . "
";
while ($row = $statement->fetch(PDO::FETCH_ASSOC, PDO::FETCH_ORI_NEXT)) {
print_r($row);
}
退货
Tables in schema
Rows: 1
Array
(
[created_on] => 2015-09-09 17:34:43.517000
[name] =>
[database_name] =>
[schema_name] =>
[kind] =>
[comment] =>
[cluster_by] =>
[rows] =>
[bytes] =>
[owner] =>
[account_name] =>
[retention_time] =>
)
Test table content
Rows: 3
注意:Rows: 3
之后没有进一步输出。
因此PDOStatement知道正确的行数,但是SHOW TABLES
的内容不完整,SELECT * FROM TEST
不知何故完全缺失($statement->fetch()
立即返回false
)。
您知道这里是否有任何PDO选项会有帮助?
我还注意到,本例中的原生ODBC函数消耗了大约200MB的内存,这似乎很多。
解决方案
我们遇到了类似的问题,通过使用Snowflake的PDO,CAST无法正常工作。
异常‘PDOException’,消息为‘SQLSTATE[SL009]:<;>:0[unixODBC][驱动程序管理器]之前没有绑定任何列 调用SQLFetch或SQLFetchScroll
长话短说,在与Snowflake Support交换了几封电子邮件后,我们发现Snowflake不支持PDO。
目前,我们不支持由于某些问题而导致的PHP PDO 由于字符串/VARCHAR类型和列大小的绑定而找到 16M字节。这是一个决定,不会在可支持性方面取得进展。 您可以继续将PHP与ODBC一起使用,但是,我们可能无法 在问题/错误出现时解决它们。
所以您最好的选择是使用PHP中的原生odbc_*
系列函数(这就是我们正在做的)。
相关文章