Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.0.x will end 8 May 2023 (12 months).
  • Bug fixes for security issues in 4.0.x will end 13 November 2023 (18 months).
  • PHP version: minimum PHP 7.3.0 Note: the minimum PHP version has increased since Moodle 3.10. PHP 7.4.x is also supported.
<?php
/**
< @version v5.20.16 12-Jan-2020 < @copyright (c) 2000-2013 John Lim (jlim#natsoft.com). All rights reserved. < @copyright (c) 2014 Damien Regad, Mark Newnham and the ADOdb community < < This is a version of the ADODB driver for DB2. It uses the 'ibm_db2' PECL extension < for PHP (http://pecl.php.net/package/ibm_db2), which in turn requires DB2 V8.2.2 or < higher. < < Originally tested with PHP 5.1.1 and Apache 2.0.55 on Windows XP SP2. < More recently tested with PHP 5.1.2 and Apache 2.0.55 on Windows XP SP2. < < This file was ported from "adodb-odbc.inc.php" by Larry Menard, "larry.menard#rogers.com". < I ripped out what I believed to be a lot of redundant or obsolete code, but there are < probably still some remnants of the ODBC support in this file; I'm relying on reviewers < of this code to point out any other things that can be removed.
> * IBM DB2 Native Client driver. > * > * Originally DB2 drivers were dependent on an ODBC driver, and some installations > * may still use that. To use an ODBC driver connection, use the odbc_db2 > * ADOdb driver. For Linux, you need the 'ibm_db2' PECL extension for PHP, > * For Windows, you need to locate an appropriate version of the php_ibm_db2.dll, > * as well as the IBM data server client software. > * This is basically a full rewrite of the original driver, for information > * about all the changes, see the update information on the ADOdb website > * for version 5.21.0. > * > * @link http://pecl.php.net/package/ibm_db2 PECL Extension For DB2 > * > * This file is part of ADOdb, a Database Abstraction Layer library for PHP. > * > * @package ADOdb > * @link https://adodb.org Project's web site and documentation > * @link https://github.com/ADOdb/ADOdb Source code and issue tracker > * > * The ADOdb Library is dual-licensed, released under both the BSD 3-Clause > * and the GNU Lesser General Public Licence (LGPL) v2.1 or, at your option, > * any later version. This means you can use it in proprietary products. > * See the LICENSE.md file distributed with this source code for details. > * @license BSD-3-Clause > * @license LGPL-2.1-or-later > * > * @copyright 2000-2013 John Lim > * @copyright 2014 Damien Regad, Mark Newnham and the ADOdb community > * @author Mark Newnham
*/ // security - hide paths if (!defined('ADODB_DIR')) die(); define("_ADODB_DB2_LAYER", 2 );
< /*-------------------------------------------------------------------------------------- < --------------------------------------------------------------------------------------*/ < < < <
class ADODB_db2 extends ADOConnection { var $databaseType = "db2"; var $fmtDate = "'Y-m-d'"; var $concat_operator = '||'; var $sysTime = 'CURRENT TIME'; var $sysDate = 'CURRENT DATE'; var $sysTimeStamp = 'CURRENT TIMESTAMP'; var $fmtTimeStamp = "'Y-m-d H:i:s'"; var $replaceQuote = "''"; // string to use to replace quotes var $dataProvider = "db2"; var $hasAffectedRows = true; var $binmode = DB2_BINARY;
< var $useFetchArray = false; // setting this to true will make array elements in FETCH_ASSOC mode case-sensitive < // breaking backward-compat < var $_bindInputArray = false;
> /* > * setting this to true will make array elements in FETCH_ASSOC > * mode case-sensitive breaking backward-compat > */ > var $useFetchArray = false; > var $_bindInputArray = true;
var $_genIDSQL = "VALUES NEXTVAL FOR %s";
< var $_genSeqSQL = "CREATE SEQUENCE %s START WITH %s NO MAXVALUE NO CYCLE";
> var $_genSeqSQL = " > CREATE SEQUENCE %s START WITH %s > NO MAXVALUE NO CYCLE INCREMENT BY 1 NO CACHE > ";
var $_dropSeqSQL = "DROP SEQUENCE %s"; var $_autocommit = true;
< var $_haserrorfunctions = true;
var $_lastAffectedRows = 0;
< var $uCaseTables = true; // for meta* functions, uppercase table names
var $hasInsertID = true;
> var $hasGenID = true; > > /* function _insertid() > * Character used to wrap column and table names for escaping special { > * characters in column and table names as well as forcing upper and return ADOConnection::GetOne('VALUES IDENTITY_VAL_LOCAL()'); > * lower case } > */ > public $nameQuote = '"'; function __construct() > { > /* $this->_haserrorfunctions = ADODB_PHPVER >= 0x4050; > * Executed after successful connection } > */ > public $connectStmt = ''; // returns true or false > function _connect($argDSN, $argUsername, $argPassword, $argDatabasename) > /* { > * Holds the current database name if (!function_exists('db2_connect')) { > */ ADOConnection::outp("Warning: The old ODBC based DB2 driver has been renamed 'odbc_db2'. This ADOdb driver calls PHP's native db2 extension which is not installed."); > private $databaseName = ''; return null; > } > /* // This needs to be set before the connect(). > * Holds information about the stored procedure request // Replaces the odbc_binmode() call that was in Execute() > * currently being built ini_set('ibm_db2.binmode', $this->binmode); > */ > private $storedProcedureParameters = false;
< function _insertid()
> function __construct() {} > > protected function _insertID($table = '', $column = '')
< function __construct()
> public function _connect($argDSN, $argUsername, $argPassword, $argDatabasename)
< $this->_haserrorfunctions = ADODB_PHPVER >= 0x4050;
> return $this->doDB2Connect($argDSN, $argUsername, $argPassword, $argDatabasename);
< // returns true or false < function _connect($argDSN, $argUsername, $argPassword, $argDatabasename)
> public function _pconnect($argDSN, $argUsername, $argPassword, $argDatabasename) > { > return $this->doDB2Connect($argDSN, $argUsername, $argPassword, $argDatabasename,true); > } > > private function doDB2Connect($argDSN, $argUsername, $argPassword, $argDatabasename, $persistent=false)
if ($argDatabasename) $schema = $argDatabasename;
>
< ADOConnection::outp("Warning: The old ODBC based DB2 driver has been renamed 'odbc_db2'. This ADOdb driver calls PHP's native db2 extension which is not installed.");
> ADOConnection::outp("DB2 extension not installed."); > return null; > } > > $connectionParameters = $this->unpackParameters($argDSN, > $argUsername, > $argPassword, > $argDatabasename); > > if ($connectionParameters == null) > { > /* > * Error thrown > */
< // This needs to be set before the connect(). < // Replaces the odbc_binmode() call that was in Execute()
> > $argDSN = $connectionParameters['dsn']; > $argUsername = $connectionParameters['uid']; > $argPassword = $connectionParameters['pwd']; > $argDatabasename = $connectionParameters['database']; > $useCataloguedConnection = $connectionParameters['catalogue']; > > if ($this->debug){ > if ($useCataloguedConnection){ > $connectMessage = "Catalogued connection using parameters: "; > $connectMessage .= "DB=$argDatabasename / "; > $connectMessage .= "UID=$argUsername / "; > $connectMessage .= "PWD=$argPassword"; > } > else > { > $connectMessage = "Uncatalogued connection using DSN: $argDSN"; > } > ADOConnection::outp($connectMessage); > } > /* > * This needs to be set before the connect(). > */
< if ($argDatabasename && empty($argDSN)) {
> if ($persistent) > $db2Function = 'db2_pconnect'; > else > $db2Function = 'db2_connect';
< if (stripos($argDatabasename,'UID=') && stripos($argDatabasename,'PWD=')) $this->_connectionID = db2_connect($argDatabasename,null,null); < else $this->_connectionID = db2_connect($argDatabasename,$argUsername,$argPassword); < } else { < if ($argDatabasename) $schema = $argDatabasename; < if (stripos($argDSN,'UID=') && stripos($argDSN,'PWD=')) $this->_connectionID = db2_connect($argDSN,null,null); < else $this->_connectionID = db2_connect($argDSN,$argUsername,$argPassword);
> /* > * We need to flatten out the connectionParameters > */ > > $db2Options = array(); > if ($this->connectionParameters) > { > foreach($this->connectionParameters as $p) > foreach($p as $k=>$v) > $db2Options[$k] = $v;
< // For db2_connect(), there is an optional 4th arg. If present, it must be < // an array of valid options. So far, we don't use them.
> if ($useCataloguedConnection) > $this->_connectionID = $db2Function($argDatabasename, > $argUsername, > $argPassword, > $db2Options); > else > $this->_connectionID = $db2Function($argDSN, > null, > null, > $db2Options);
< if (isset($this->connectStmt)) $this->Execute($this->connectStmt);
< if ($this->_connectionID && isset($schema)) $this->Execute("SET SCHEMA=$schema");
> if ($this->_connectionID && $this->connectStmt) > $this->execute($this->connectStmt); >
// returns true or false
>
< // returns true or false < function _pconnect($argDSN, $argUsername, $argPassword, $argDatabasename)
> /** > * Validates and preprocesses the passed parameters for consistency > * > * @param string $argDSN Either DSN or database > * @param string $argUsername User name or null > * @param string $argPassword Password or null > * @param string $argDatabasename Either DSN or database > * > * @return mixed array if correct, null if not > */ > private function unpackParameters($argDSN, $argUsername, $argPassword, $argDatabasename)
< if (!function_exists('db2_connect')) return null;
< // This needs to be set before the connect(). < // Replaces the odbc_binmode() call that was in Execute() < ini_set('ibm_db2.binmode', $this->binmode);
> $connectionParameters = array('dsn'=>'', > 'uid'=>'', > 'pwd'=>'', > 'database'=>'', > 'catalogue'=>true > );
< $this->_errorMsg = '';
> /* > * Uou can either connect to a catalogued connection > * with a database name e.g. 'SAMPLE' > * or an uncatalogued connection with a DSN like connection > * DATABASE=database;HOSTNAME=hostname;PORT=port;PROTOCOL=TCPIP;UID=username;PWD=password; > */
< if ($argDatabasename && empty($argDSN)) {
> if (!$argDSN && !$argDatabasename) > { > $errorMessage = 'Supply either catalogued or uncatalogued connection parameters'; > $this->_errorMsg = $errorMessage; > if ($this->debug) > ADOConnection::outp($errorMessage); > return null; > }
< if (stripos($argDatabasename,'UID=') && stripos($argDatabasename,'PWD=')) $this->_connectionID = db2_pconnect($argDatabasename,null,null); < else $this->_connectionID = db2_pconnect($argDatabasename,$argUsername,$argPassword); < } else { < if ($argDatabasename) $schema = $argDatabasename; < if (stripos($argDSN,'UID=') && stripos($argDSN,'PWD=')) $this->_connectionID = db2_pconnect($argDSN,null,null); < else $this->_connectionID = db2_pconnect($argDSN,$argUsername,$argPassword);
> $useCataloguedConnection = true; > $schemaName = ''; > > if ($argDSN && $argDatabasename) > { > /* > * If a catalogued connection if provided, > * as well as user and password > * that will take priority > */ > if ($argUsername && $argPassword && !$this->isDsn($argDatabasename)) > { > if ($this->debug){ > $errorMessage = 'Warning: Because you provided user,'; > $errorMessage.= 'password and database, DSN connection '; > $errorMessage.= 'parameters were discarded'; > ADOConnection::outp($errorMessage); > > } > $argDSN = ''; > } > else if ($this->isDsn($argDSN) && $this->isDsn($argDatabasename)) > { > $errorMessage = 'Supply uncatalogued connection parameters '; > $errorMessage.= 'in either the database or DSN arguments, '; > $errorMessage.= 'but not both'; > if ($this->debug) > ADOConnection::outp($errorMessage); > return null; > }
}
< $this->_errorMsg = @db2_conn_errormsg(); < if ($this->_connectionID && $this->autoRollback) @db2_rollback($this->_connectionID); < if (isset($this->connectStmt)) $this->Execute($this->connectStmt);
> if (!$this->isDsn($argDSN) && $this->isDsn($argDatabasename)) > { > /* > * Switch them around for next test > */ > $temp = $argDSN; > $argDsn = $argDatabasename; > $argDatabasenME = $temp; > }
< if ($this->_connectionID && isset($schema)) $this->Execute("SET SCHEMA=$schema"); < return $this->_connectionID != false;
> if ($this->isDsn($argDSN)) > { > > if (!preg_match('/uid=/i',$argDSN) > || !preg_match('/pwd=/i',$argDSN)) > { > $errorMessage = 'For uncatalogued connections, provide '; > $errorMessage.= 'both UID and PWD in the connection string'; > if ($this->debug) > ADOConnection::outp($errorMessage); > return null; > } > > if (preg_match('/database=/i',$argDSN)) > { > if ($argDatabasename) > { > $argDatabasename = ''; > if ($this->debug) > { > $errorMessage = 'Warning: Because you provided '; > $errorMessage.= 'database information in the DSN '; > $errorMessage.= 'parameters, the supplied database '; > $errorMessage.= 'name was discarded'; > ADOConnection::outp($errorMessage); > } > } > $useCataloguedConnection = false; > > } > elseif ($argDatabasename) > { > $this->databaseName = $argDatabasename; > $argDSN .= ';database=' . $argDatabasename; > $argDatabasename = ''; > $useCataloguedConnection = false; > > } > else > { > $errorMessage = 'Uncatalogued connection parameters '; > $errorMessage.= 'must contain a database= argument'; > if ($this->debug) > ADOConnection::outp($errorMessage); > return null; > } > } > > if ($argDSN && !$argDatabasename && $useCataloguedConnection) > { > $argDatabasename = $argDSN; > $argDSN = ''; > } > > > if ($useCataloguedConnection > && (!$argDatabasename > || !$argUsername > || !$argPassword)) > { > > $errorMessage = 'For catalogued connections, provide '; > $errorMessage.= 'database, username and password'; > $this->_errorMsg = $errorMessage; > if ($this->debug) > ADOConnection::outp($errorMessage); > return null; > > } > > if ($argDatabasename) > $this->databaseName = $argDatabasename; > elseif (!$this->databaseName) > $this->databaseName = $this->getDatabasenameFromDsn($argDSN); > > > $connectionParameters = array('dsn'=>$argDSN, > 'uid'=>$argUsername, > 'pwd'=>$argPassword, > 'database'=>$argDatabasename, > 'catalogue'=>$useCataloguedConnection > ); > > return $connectionParameters; > > } > > /** > * Does the provided string look like a DSN > * > * @param string $dsnString > * > * @return bool > */ > private function isDsn($dsnString){ > $dsnArray = preg_split('/[;=]+/',$dsnString); > if (count($dsnArray) > 2) > return true; > return false;
}
< // format and return date string in database timestamp format < function DBTimeStamp($ts, $isfld = false)
> > /** > * Gets the database name from the DSN > * > * @param string $dsnString > * > * @return string > */ > private function getDatabasenameFromDsn($dsnString){ > > $dsnArray = preg_split('/[;=]+/',$dsnString); > $dbIndex = array_search('database',$dsnArray); > > return $dsnArray[$dbIndex + 1]; > } > > > /** > * format and return date string in database timestamp format > * > * @param mixed $ts either a string or a unixtime > * @param bool $isField discarded > * > * @return string > */ > function dbTimeStamp($ts,$isField=false)
{ if (empty($ts) && $ts !== 0) return 'null';
< if (is_string($ts)) $ts = ADORecordSet::UnixTimeStamp($ts);
> if (is_string($ts)) $ts = ADORecordSet::unixTimeStamp($ts);
return 'TO_DATE('.adodb_date($this->fmtTimeStamp,$ts).",'YYYY-MM-DD HH24:MI:SS')"; }
< // Format date column in sql string given an input format that understands Y M D < function SQLDate($fmt, $col=false)
> /** > * Format date column in sql string given an input format that understands Y M D > * > * @param string $fmt > * @param bool $col > * > * @return string > */ > function sqlDate($fmt, $col=false)
{
< // use right() and replace() ?
if (!$col) $col = $this->sysDate; /* use TO_CHAR() if $fmt is TO_CHAR() allowed fmt */ if ($fmt== 'Y-m-d H:i:s') return 'TO_CHAR('.$col.", 'YYYY-MM-DD HH24:MI:SS')"; $s = ''; $len = strlen($fmt); for ($i=0; $i < $len; $i++) { if ($s) $s .= $this->concat_operator; $ch = $fmt[$i]; switch($ch) { case 'Y': case 'y': if ($len==1) return "year($col)"; $s .= "char(year($col))"; break; case 'M': if ($len==1) return "monthname($col)"; $s .= "substr(monthname($col),1,3)"; break; case 'm': if ($len==1) return "month($col)"; $s .= "right(digits(month($col)),2)"; break; case 'D': case 'd': if ($len==1) return "day($col)"; $s .= "right(digits(day($col)),2)"; break; case 'H': case 'h': if ($len==1) return "hour($col)"; if ($col != $this->sysDate) $s .= "right(digits(hour($col)),2)"; else $s .= "''"; break; case 'i': case 'I': if ($len==1) return "minute($col)"; if ($col != $this->sysDate) $s .= "right(digits(minute($col)),2)"; else $s .= "''"; break; case 'S': case 's': if ($len==1) return "second($col)"; if ($col != $this->sysDate) $s .= "right(digits(second($col)),2)"; else $s .= "''"; break; default: if ($ch == '\\') { $i++; $ch = substr($fmt,$i,1); } $s .= $this->qstr($ch); } } return $s; }
< function ServerInfo()
> function serverInfo()
{
< $row = $this->GetRow("SELECT service_level, fixpack_num FROM TABLE(sysproc.env_get_inst_info()) < as INSTANCEINFO");
> $sql = "SELECT service_level, fixpack_num > FROM TABLE(sysproc.env_get_inst_info()) > AS INSTANCEINFO"; > $row = $this->GetRow($sql);
if ($row) { $info['version'] = $row[0].':'.$row[1]; $info['fixpack'] = $row[1]; $info['description'] = ''; } else {
< return ADOConnection::ServerInfo();
> return ADOConnection::serverInfo();
} return $info; }
< function CreateSequence($seqname='adodbseq',$start=1)
> function createSequence($seqname='adodbseq',$start=1)
{
< if (empty($this->_genSeqSQL)) return false; < $ok = $this->Execute(sprintf($this->_genSeqSQL,$seqname,$start)); < if (!$ok) return false;
> if (empty($this->_genSeqSQL)) > return false; > > $ok = $this->execute(sprintf($this->_genSeqSQL,$seqname,$start)); > if (!$ok) > return false;
return true; }
< function DropSequence($seqname = 'adodbseq')
> function dropSequence($seqname='adodbseq')
{ if (empty($this->_dropSeqSQL)) return false;
< return $this->Execute(sprintf($this->_dropSeqSQL,$seqname));
> return $this->execute(sprintf($this->_dropSeqSQL,$seqname));
}
< function SelectLimit($sql, $nrows = -1, $offset = -1, $inputArr = false, $secs2cache = 0)
> function selectLimit($sql,$nrows=-1,$offset=-1,$inputArr=false,$secs2cache=0) > { > $nrows = (integer) $nrows; > > if ($offset <= 0) > { > if ($nrows >= 0) > $sql .= " FETCH FIRST $nrows ROWS ONLY "; > > $rs = $this->execute($sql,$inputArr); > > } > else
{
< $nrows = (int) $nrows; < $offset = (int) $offset; < if ($offset <= 0) { < // could also use " OPTIMIZE FOR $nrows ROWS " < if ($nrows >= 0) $sql .= " FETCH FIRST $nrows ROWS ONLY "; < $rs = $this->Execute($sql,$inputArr); < } else {
if ($offset > 0 && $nrows < 0);
< else {
> > else > {
$nrows += $offset; $sql .= " FETCH FIRST $nrows ROWS ONLY "; }
< $rs = ADOConnection::SelectLimit($sql,-1,$offset,$inputArr);
> > /* > * DB2 has no native support for mid table offset > */ > $rs = ADOConnection::selectLimit($sql,$nrows,$offset,$inputArr); >
} return $rs; }
< /* < This algorithm is not very efficient, but works even if table locking < is not available.
< Will return false if unable to generate an ID after $MAXLOOPS attempts. < */ < function GenID($seq='adodbseq',$start=1)
> function errorMsg()
{
< // if you have to modify the parameter below, your database is overloaded, < // or you need to implement generation of id's yourself! < $num = $this->GetOne("VALUES NEXTVAL FOR $seq"); < return $num; < }
> if ($this->_errorMsg !== false) > return $this->_errorMsg;
> if (empty($this->_connectionID)) > return @db2_conn_errormsg();
< function ErrorMsg() < { < if ($this->_haserrorfunctions) { < if ($this->_errorMsg !== false) return $this->_errorMsg; < if (empty($this->_connectionID)) return @db2_conn_errormsg();
return @db2_conn_errormsg($this->_connectionID);
< } else return ADOConnection::ErrorMsg();
}
< function ErrorNo()
> function errorNo()
{
< if ($this->_haserrorfunctions) { < if ($this->_errorCode !== false) { < // bug in 4.0.6, error number can be corrupted string (should be 6 digits) < return (strlen($this->_errorCode)<=2) ? 0 : $this->_errorCode; < }
> if ($this->_errorCode !== false) > return $this->_errorCode; >
< if (empty($this->_connectionID)) $e = @db2_conn_error(); < else $e = @db2_conn_error($this->_connectionID);
> if (empty($this->_connectionID)) > $e = @db2_conn_error(); > > else > $e = @db2_conn_error($this->_connectionID);
< // bug in 4.0.6, error number can be corrupted string (should be 6 digits) < // so we check and patch < if (strlen($e)<=2) return 0;
return $e;
< } else return ADOConnection::ErrorNo();
}
< function BeginTrans()
> function beginTrans()
{
< if (!$this->hasTransactions) return false; < if ($this->transOff) return true;
> if (!$this->hasTransactions) > return false; > if ($this->transOff) > return true; >
$this->transCnt += 1;
>
$this->_autocommit = false;
>
return db2_autocommit($this->_connectionID,false); } function CommitTrans($ok=true) {
< if ($this->transOff) return true; < if (!$ok) return $this->RollbackTrans(); < if ($this->transCnt) $this->transCnt -= 1;
> if ($this->transOff) > return true; > > if (!$ok) > return $this->RollbackTrans(); > > if ($this->transCnt) > $this->transCnt -= 1; >
$this->_autocommit = true;
< $ret = db2_commit($this->_connectionID); < db2_autocommit($this->_connectionID,true);
> $ret = @db2_commit($this->_connectionID); > @db2_autocommit($this->_connectionID,true);
return $ret; } function RollbackTrans() { if ($this->transOff) return true; if ($this->transCnt) $this->transCnt -= 1; $this->_autocommit = true;
< $ret = db2_rollback($this->_connectionID); < db2_autocommit($this->_connectionID,true);
> $ret = @db2_rollback($this->_connectionID); > @db2_autocommit($this->_connectionID,true);
return $ret; }
< function MetaPrimaryKeys($table, $owner = false)
> /** > * Return a list of Primary Keys for a specified table > * > * We don't use db2_statistics as the function does not seem to play > * well with mixed case table names > * > * @param string $table > * @param bool $primary (optional) only return primary keys > * @param bool $owner (optional) not used in this driver > * > * @return string[] Array of indexes > */ > public function metaPrimaryKeys($table,$owner=false)
{
> global $ADODB_FETCH_MODE; > $primaryKeys = array(); >
< if ($this->uCaseTables) $table = strtoupper($table);
$schema = ''; $this->_findschema($table,$schema);
> $table = $this->getTableCasedValue($table); $savem = $ADODB_FETCH_MODE; >
$ADODB_FETCH_MODE = ADODB_FETCH_NUM;
< $qid = @db2_primarykeys($this->_connectionID,'',$schema,$table);
> $this->setFetchMode(ADODB_FETCH_NUM); >
< if (!$qid) {
> $sql = "SELECT * > FROM syscat.indexes > WHERE tabname='$table'"; > > $rows = $this->getAll($sql); > > $this->setFetchMode($savem);
$ADODB_FETCH_MODE = $savem;
> return false; > if (empty($rows))
< } < $rs = new ADORecordSet_db2($qid); < $ADODB_FETCH_MODE = $savem;
< if (!$rs) return false;
> foreach ($rows as $r) > { > if ($r[7] != 'P') > continue;
< $arr = $rs->GetArray(); < $rs->Close(); < $arr2 = array(); < for ($i=0; $i < sizeof($arr); $i++) { < if ($arr[$i][3]) $arr2[] = $arr[$i][3];
> $cols = explode('+',$r[6]); > foreach ($cols as $colIndex=>$col) > { > if ($colIndex == 0) > continue; > $columnName = $this->getMetaCasedValue($col); > $primaryKeys[] = $columnName; > } > break;
}
< return $arr2;
> return $primaryKeys;
}
< function MetaForeignKeys($table, $owner = FALSE, $upper = FALSE, $asociative = FALSE )
> /** > * returns assoc array where keys are tables, and values are foreign keys > * > * @param string $table > * @param string $owner [optional][discarded] > * @param bool $upper [optional][discarded] > * @param bool $associative[optional][discarded] > * > * @return mixed[] Array of foreign key information > */ > public function metaForeignKeys($table, $owner = FALSE, $upper = FALSE, $asociative = FALSE )
{
>
global $ADODB_FETCH_MODE;
< if ($this->uCaseTables) $table = strtoupper($table);
$schema = ''; $this->_findschema($table,$schema); $savem = $ADODB_FETCH_MODE; $ADODB_FETCH_MODE = ADODB_FETCH_NUM;
< $qid = @db2_foreign_keys($this->_connectionID,'',$schema,$table); < if (!$qid) {
> > $this->setFetchMode(ADODB_FETCH_NUM); > > $sql = "SELECT SUBSTR(tabname,1,20) table_name, > SUBSTR(constname,1,20) fk_name, > SUBSTR(REFTABNAME,1,12) parent_table, > SUBSTR(refkeyname,1,20) pk_orig_table, > fk_colnames > FROM syscat.references > WHERE tabname = '$table'"; > > $results = $this->getAll($sql); >
$ADODB_FETCH_MODE = $savem;
> $this->setFetchMode($savem); return false; > } > if (empty($results))
$rs = new ADORecordSet_db2($qid);
> > $foreignKeys = array(); $ADODB_FETCH_MODE = $savem; > /* > foreach ($results as $r) $rs->fields indices > { 0 PKTABLE_CAT > $parentTable = trim($this->getMetaCasedValue($r[2])); 1 PKTABLE_SCHEM > $keyName = trim($this->getMetaCasedValue($r[1])); 2 PKTABLE_NAME > $foreignKeys[$parentTable] = $keyName; 3 PKCOLUMN_NAME > } 4 FKTABLE_CAT > 5 FKTABLE_SCHEM > return $foreignKeys;
6 FKTABLE_NAME
> 7 FKCOLUMN_NAME > /** */ > * Returns a list of tables if (!$rs) return false; > * > * @param string $ttype (optional) $foreign_keys = array(); > * @param string $schema (optional) while (!$rs->EOF) { > * @param string $mask (optional) if (strtoupper(trim($rs->fields[2])) == $table && (!$schema || strtoupper($rs->fields[1]) == $schema)) { > * if (!is_array($foreign_keys[$rs->fields[5].'.'.$rs->fields[6]])) > * @return array $foreign_keys[$rs->fields[5].'.'.$rs->fields[6]] = array(); > */ $foreign_keys[$rs->fields[5].'.'.$rs->fields[6]][$rs->fields[7]] = $rs->fields[3]; > public function metaTables($ttype=false,$schema=false,$mask=false) } > { $rs->MoveNext(); > } > global $ADODB_FETCH_MODE; > $rs->Close(); > $savem = $ADODB_FETCH_MODE; return $foreign_key; > $ADODB_FETCH_MODE = ADODB_FETCH_NUM; } > > /* > * Values for TABLE_TYPE function MetaTables($ttype = false, $schema = false, $mask = false) > * --------------------------- { > * ALIAS, HIERARCHY TABLE, INOPERATIVE VIEW, NICKNAME, global $ADODB_FETCH_MODE; > * MATERIALIZED QUERY TABLE, SYSTEM TABLE, TABLE, > * TYPED TABLE, TYPED VIEW, and VIEW $savem = $ADODB_FETCH_MODE; > * $ADODB_FETCH_MODE = ADODB_FETCH_NUM; > * If $ttype passed as '', match 'TABLE' and 'VIEW' $qid = db2_tables($this->_connectionID); > * If $ttype passed as 'T' it is assumed to be 'TABLE' > * if $ttype passed as 'V' it is assumed to be 'VIEW' $rs = new ADORecordSet_db2($qid); > */ > $ttype = strtoupper($ttype); $ADODB_FETCH_MODE = $savem; > if ($ttype) { if (!$rs) { > /* $false = false; > * @todo We could do valid type checking or array type return $false; > */ } > if ($ttype == 'V') > $ttype = 'VIEW'; $arr = $rs->GetArray(); > if ($ttype == 'T') $rs->Close(); > $ttype = 'TABLE'; $arr2 = array(); > } > if ($ttype) { > if (!$schema) $isview = strncmp($ttype,'V',1) === 0; > $schema = '%'; } > for ($i=0; $i < sizeof($arr); $i++) { > if (!$mask) if (!$arr[$i][2]) continue; > $mask = '%'; $type = $arr[$i][3]; > $owner = $arr[$i][1]; > $qid = @db2_tables($this->_connectionID,NULL,$schema,$mask,$ttype); $schemaval = ($schema) ? $arr[$i][1].'.' : ''; >
if ($ttype) {
> if ($isview) { > if (!$rs) if (strncmp($type,'V',1) === 0) $arr2[] = $schemaval.$arr[$i][2]; > return false; } else if (strncmp($owner,'SYS',3) !== 0) $arr2[] = $schemaval.$arr[$i][2]; > } else if (strncmp($owner,'SYS',3) !== 0) $arr2[] = $schemaval.$arr[$i][2]; > $arr = $rs->getArray(); } > return $arr2; > $rs->Close(); } > > $tableList = array(); /* >
< $rs->fields indices < 0 PKTABLE_CAT < 1 PKTABLE_SCHEM < 2 PKTABLE_NAME < 3 PKCOLUMN_NAME < 4 FKTABLE_CAT < 5 FKTABLE_SCHEM < 6 FKTABLE_NAME < 7 FKCOLUMN_NAME
> * Array items > * --------------------------------- > * 0 TABLE_CAT The catalog that contains the table. > * The value is NULL if this table does not have catalogs. > * 1 TABLE_SCHEM Name of the schema that contains the table. > * 2 TABLE_NAME Name of the table. > * 3 TABLE_TYPE Table type identifier for the table. > * 4 REMARKS Description of the table.
< if (!$rs) return false;
< $foreign_keys = array(); < while (!$rs->EOF) { < if (strtoupper(trim($rs->fields[2])) == $table && (!$schema || strtoupper($rs->fields[1]) == $schema)) { < if (!is_array($foreign_keys[$rs->fields[5].'.'.$rs->fields[6]])) < $foreign_keys[$rs->fields[5].'.'.$rs->fields[6]] = array(); < $foreign_keys[$rs->fields[5].'.'.$rs->fields[6]][$rs->fields[7]] = $rs->fields[3];
> for ($i=0; $i < sizeof($arr); $i++) > { > > $tableRow = $arr[$i]; > $tableName = $tableRow[2]; > $tableType = $tableRow[3]; > > if (!$tableName) > continue; > > if ($ttype == '' && (strcmp($tableType,'TABLE') <> 0 && strcmp($tableType,'VIEW') <> 0)) > continue; > > /* > * Set metacasing if required > */ > $tableName = $this->getMetaCasedValue($tableName); > > /* > * If we requested a schema, we prepend the schema > name to the table name > */ > if (strcmp($schema,'%') <> 0) > $tableName = $schema . '.' . $tableName; > > $tableList[] = $tableName; >
< $rs->MoveNext();
> return $tableList;
< $rs->Close(); < return $foreign_key; < }
> /** > * Return a list of indexes for a specified table > * > * We don't use db2_statistics as the function does not seem to play > * well with mixed case table names > * > * @param string $table > * @param bool $primary (optional) only return primary keys > * @param bool $owner (optional) not used in this driver > * > * @return string[] Array of indexes > */ > public function metaIndexes($table, $primary = false, $owner = false) { > > global $ADODB_FETCH_MODE; > > /* Array( > * [name_of_index] => Array( > * [unique] => true or false > * [columns] => Array( > * [0] => firstcol > * [1] => nextcol > * [2] => etc........ > * ) > * ) > * ) > */ > $indices = array(); > $primaryKeyName = ''; > > $table = $this->getTableCasedValue($table); > > > $savem = $ADODB_FETCH_MODE; > $ADODB_FETCH_MODE = ADODB_FETCH_NUM; > $this->setFetchMode(ADODB_FETCH_NUM); > > $sql = "SELECT * > FROM syscat.indexes > WHERE tabname='$table'"; > > $rows = $this->getAll($sql); > > $this->setFetchMode($savem); > $ADODB_FETCH_MODE = $savem;
#define SQL_TYPE_TIME 92
> if (empty($rows)) #define SQL_TYPE_TIMESTAMP 93 > return false;
< function MetaTables($ttype = false, $schema = false, $mask = false)
> foreach ($rows as $r)
#define SQL_UNICODE (-95)
> #define SQL_UNICODE_VARCHAR (-96) > $primaryIndex = $r[7] == 'P'?1:0; #define SQL_UNICODE_LONGVARCHAR (-97) > if (!$primary) */ > /* function DB2Types($t) > * Primary key not requested, ignore that one { > */ switch ((integer)$t) { > if ($r[7] == 'P') case 1: > continue; case 12: > case 0: > $indexName = $this->getMetaCasedValue($r[1]); case -95: > if (!isset($indices[$indexName])) case -96: > { return 'C'; > $unique = ($r[7] == 'U')?1:0; case -97: > $indices[$indexName] = array('unique'=>$unique, case -1: //text > 'primary'=>$primaryIndex, return 'X'; > 'columns'=>array() case -4: //image > ); return 'B'; > } > $cols = explode('+',$r[6]); case 9: > foreach ($cols as $colIndex=>$col) case 91: > { return 'D'; > if ($colIndex == 0) > continue; case 10: > $columnName = $this->getMetaCasedValue($col); case 11: > $indices[$indexName]['columns'][] = $columnName; case 92: > } case 93: > return 'T'; > } > case 4: > return $indices; case 5: > case -6: > } return 'I'; > > /** case -11: // uniqidentifier > * List procedures or functions in an array. return 'R'; > * case -7: //bit > * We interrogate syscat.routines instead of calling the PHP return 'L'; > * function procedures because ADOdb requires the type of procedure > * this is not available in the php function default: > * return 'N'; > * @param string $procedureNamePattern (optional) } > * @param string $catalog (optional) } > * @param string $schemaPattern (optional) > function MetaColumns($table, $normalize=true) > * @return array of procedures on current database. { > * global $ADODB_FETCH_MODE; > */ > public function metaProcedures($procedureNamePattern = null, $catalog = null, $schemaPattern = null) { $false = false; > if ($this->uCaseTables) $table = strtoupper($table); >
$schema = '';
> $metaProcedures = array(); $this->_findschema($table,$schema); > $procedureSQL = ''; > $catalogSQL = ''; $savem = $ADODB_FETCH_MODE; > $schemaSQL = ''; $ADODB_FETCH_MODE = ADODB_FETCH_NUM; >
< $qid = db2_tables($this->_connectionID);
< $rs = new ADORecordSet_db2($qid);
> if ($procedureNamePattern) > $procedureSQL = "AND ROUTINENAME LIKE " . strtoupper($this->qstr($procedureNamePattern)); > > if ($catalog) > $catalogSQL = "AND OWNER=" . strtoupper($this->qstr($catalog)); > > if ($schemaPattern) > $schemaSQL = "AND ROUTINESCHEMA LIKE {$this->qstr($schemaPattern)}"; > > > $fields = " > ROUTINENAME, > CASE ROUTINETYPE > WHEN 'P' THEN 'PROCEDURE' > WHEN 'F' THEN 'FUNCTION' > ELSE 'METHOD' > END AS ROUTINETYPE_NAME, > ROUTINESCHEMA, > REMARKS"; > > $SQL = "SELECT $fields > FROM syscat.routines > WHERE OWNER IS NOT NULL > $procedureSQL > $catalogSQL > $schemaSQL > ORDER BY ROUTINENAME > "; > > $result = $this->execute($SQL);
< if (!$rs) { < $false = false; < return $false; < }
< $arr = $rs->GetArray(); < $rs->Close(); < $arr2 = array();
> if (!$result) > return false;
< if ($ttype) { < $isview = strncmp($ttype,'V',1) === 0;
> while ($r = $result->fetchRow()){ > $procedureName = $this->getMetaCasedValue($r[0]); > $schemaName = $this->getMetaCasedValue($r[2]); > $metaProcedures[$procedureName] = array('type'=> $r[1], > 'catalog' => '', > 'schema' => $schemaName, > 'remarks' => $r[3] > );
< for ($i=0; $i < sizeof($arr); $i++) { < if (!$arr[$i][2]) continue; < $type = $arr[$i][3]; < $owner = $arr[$i][1]; < $schemaval = ($schema) ? $arr[$i][1].'.' : ''; < if ($ttype) { < if ($isview) { < if (strncmp($type,'V',1) === 0) $arr2[] = $schemaval.$arr[$i][2]; < } else if (strncmp($owner,'SYS',3) !== 0) $arr2[] = $schemaval.$arr[$i][2]; < } else if (strncmp($owner,'SYS',3) !== 0) $arr2[] = $schemaval.$arr[$i][2];
> > return $metaProcedures; >
< return $arr2;
> > /** > * Lists databases. Because instances are independent, we only know about > * the current database name > * > * @return string[] > */ > public function metaDatabases(){ > > $dbName = $this->getMetaCasedValue($this->databaseName); > > return (array)$dbName; >
7 LENGTH
> 8 SCALE > 9 RADIX >
< function MetaColumns($table, $normalize=true)
> public function metaColumns($table, $normalize=true)
< $false = false; < if ($this->uCaseTables) $table = strtoupper($table); < $schema = ''; < $this->_findschema($table,$schema); <
< $ADODB_FETCH_MODE = ADODB_FETCH_NUM;
$fld->type = $this->DB2Types($rs->fields[4]);
> $schema = '%'; > $this->_findschema($table,$schema); // ref: http://msdn.microsoft.com/library/default.asp?url=/archive/en-us/dnaraccgen/html/msdn_odk.asp > $table = $this->getTableCasedValue($table);
< $qid = db2_columns($this->_connectionID, "", $schema, $table, $colname); < if (empty($qid)) return $false;
> $qid = db2_columns($this->_connectionID, null, $schema, $table, $colname); > if (empty($qid)) > { > if ($this->debug) > { > $errorMessage = @db2_conn_errormsg($this->_connectionID); > ADOConnection::outp($errorMessage); > } > return false; > }
< $ADODB_FETCH_MODE = $savem;
< if (!$rs) return $false;
> if (!$rs) > return false; >
else
> 12 Column Default $fld->max_length = $rs->fields[7]; > 13 SQL Data Type } else > 14 SQL DateTime SubType $fld->max_length = $rs->fields[7]; > 15 Max length in Octets $fld->not_null = !empty($rs->fields[10]); > 16 Ordinal Position $fld->scale = $rs->fields[8]; > 17 Is NULLABLE
< while (!$rs->EOF) { < if (strtoupper(trim($rs->fields[2])) == $table && (!$schema || strtoupper($rs->fields[1]) == $schema)) {
> while (!$rs->EOF) > { > if ($rs->fields[2] == $table) > { >
} else if (sizeof($retarr)>0)
>
break;
>
< $retarr[strtoupper($fld->name)] = $fld; < } else if (sizeof($retarr)>0)
> > //$columnName = $this->getMetaCasedValue($fld->name); > $columnName = strtoupper($fld->name); > $retarr[$columnName] = $fld; > > } > else if (sizeof($retarr)>0)
$rs->Close();
>
if (empty($retarr)) $retarr = false;
>
>
< if (empty($retarr)) $retarr = false;
> if (empty($retarr)) > $retarr = false; > > /* > * Now we find out if the column is part of a primary key > */
< $qid = db2_primary_keys($this->_connectionID, "", $schema, $table); < if (empty($qid)) return $false;
> $qid = @db2_primary_keys($this->_connectionID, "", $schema, $table); > if (empty($qid)) > return false;
$rs = new ADORecordSet_db2($qid);
< $ADODB_FETCH_MODE = $savem;
< if (!$rs) return $retarr;
> if (!$rs) > { > $ADODB_FETCH_MODE = $savem; > return $retarr; > }
$rs->_fetch(); /* $rs->fields indices 0 TABLE_CAT 1 TABLE_SCHEM 2 TABLE_NAME 3 COLUMN_NAME 4 KEY_SEQ 5 PK_NAME */ while (!$rs->EOF) {
< if (strtoupper(trim($rs->fields[2])) == $table && (!$schema || strtoupper($rs->fields[1]) == $schema)) {
> if (strtoupper(trim($rs->fields[2])) == $table > && (!$schema || strtoupper($rs->fields[1]) == $schema)) > {
$retarr[strtoupper($rs->fields[3])]->primary_key = true;
< } else if (sizeof($retarr)>0)
> } > else if (sizeof($retarr)>0)
break;
>
$rs->MoveNext(); } $rs->Close();
< if (empty($retarr)) $retarr = false;
> $ADODB_FETCH_MODE = $savem; > > if (empty($retarr)) > return false; > > /* > * If the fetch mode is numeric, return as numeric array > */ > if ($ADODB_FETCH_MODE == ADODB_FETCH_NUM) > $retarr = array_values($retarr); >
return $retarr; }
> /** > * In this version if prepareSp, we just check to make sure function Prepare($sql) > * that the name of the stored procedure is correct { > * If true, we returns an array if (! $this->_bindInputArray) return $sql; // no binding > * else false $stmt = db2_prepare($this->_connectionID,$sql); > * if (!$stmt) { > * @param string $procedureName // we don't know whether db2 driver is parsing prepared stmts, so just return sql > * @param mixed $parameters (not used in db2 connections) return $sql; > * @return mixed[] } > */ return array($sql,$stmt,false); > function prepareSp($procedureName,$parameters=false) { } > > global $ADODB_FETCH_MODE; /* returns queryID or false */ > function _query($sql,$inputarr=false) > $this->storedProcedureParameters = array('name'=>'', { > 'resource'=>false, $last_php_error = $this->resetLastError(); > 'in'=>array(), $this->_errorMsg = ''; > 'out'=>array(), > 'index'=>array(), if ($inputarr) { > 'parameters'=>array(), if (is_array($sql)) { > 'keyvalue' => array()); $stmtid = $sql[1]; > } else { > //$procedureName = strtoupper($procedureName); $stmtid = db2_prepare($this->_connectionID,$sql); > //$procedureName = $this->getTableCasedValue($procedureName); > if ($stmtid == false) { > $savem = $ADODB_FETCH_MODE; $this->_errorMsg = $this->getChangedErrorMsg($last_php_error); > $ADODB_FETCH_MODE = ADODB_FETCH_NUM; return false; > } > $qid = db2_procedures($this->_connectionID, NULL , '%' , $procedureName );
< function Prepare($sql)
> $ADODB_FETCH_MODE = $savem; > > if (!$qid)
> if ($this->debug) if (! db2_execute($stmtid,$inputarr)) { > ADOConnection::outp(sprintf('No Procedure of name %s available',$procedureName)); if ($this->_haserrorfunctions) { > return false; $this->_errorMsg = db2_stmt_errormsg(); > } $this->_errorCode = db2_stmt_error(); > } > return false; > } > $this->storedProcedureParameters['name'] = $procedureName; > /* } else if (is_array($sql)) { > * Now we know we have a valid procedure name, lets see if it requires $stmtid = $sql[1]; > * parameters if (!db2_execute($stmtid)) { > */ if ($this->_haserrorfunctions) { > $savem = $ADODB_FETCH_MODE; $this->_errorMsg = db2_stmt_errormsg(); > $ADODB_FETCH_MODE = ADODB_FETCH_NUM; $this->_errorCode = db2_stmt_error(); > } > $qid = db2_procedure_columns($this->_connectionID, NULL , '%' , $procedureName , NULL ); return false; > } > $ADODB_FETCH_MODE = $savem; } else > $stmtid = @db2_exec($this->_connectionID,$sql); > if (!$qid) > { $this->_lastAffectedRows = 0; > if ($this->debug) if ($stmtid) { > ADOConnection::outp(sprintf('No columns of name %s available',$procedureName)); if (@db2_num_fields($stmtid) == 0) { > return false; $this->_lastAffectedRows = db2_num_rows($stmtid); > } $stmtid = true; > $rs = new ADORecordSet_db2($qid); } else { > if (!$rs) $this->_lastAffectedRows = 0; > return false; } > > $preparedStatement = 'CALL %s(%s)'; if ($this->_haserrorfunctions) { > $parameterMarkers = array(); $this->_errorMsg = ''; > while (!$rs->EOF) $this->_errorCode = 0; > { } else { > $parameterName = $rs->fields[3]; $this->_errorMsg = $this->getChangedErrorMsg($last_php_error); > if ($parameterName == '') } > { } else { > $rs->moveNext(); if ($this->_haserrorfunctions) { > continue; $this->_errorMsg = db2_stmt_errormsg(); > } $this->_errorCode = db2_stmt_error(); > $parameterType = $rs->fields[4]; } else { > $ordinalPosition = $rs->fields[17]; $this->_errorMsg = $this->getChangedErrorMsg($last_php_error); > switch($parameterType) } > { } > case DB2_PARAM_IN: return $stmtid; > case DB2_PARAM_INOUT: } > $this->storedProcedureParameters['in'][$parameterName] = ''; > break; /* > case DB2_PARAM_INOUT: Insert a null into the blob field of the table first. > case DB2_PARAM_OUT: Then use UpdateBlob to store the blob. > $this->storedProcedureParameters['out'][$parameterName] = ''; > break; Usage: > } > $this->storedProcedureParameters['index'][$parameterName] = $ordinalPosition; $conn->Execute('INSERT INTO blobtable (id, blobcol) VALUES (1, null)'); > $this->storedProcedureParameters['parameters'][$ordinalPosition] = $rs->fields; $conn->UpdateBlob('blobtable','blobcol',$blob,'id=1'); > $rs->moveNext(); */ > function UpdateBlob($table,$column,$val,$where,$blobtype='BLOB') > } { > $parameterCount = count($this->storedProcedureParameters['index']); return $this->Execute("UPDATE $table SET $column=? WHERE $where",array($val)) != false; > $parameterMarkers = array_fill(0,$parameterCount,'?'); } > > /* // returns true or false > * We now know how many parameters to bind to the stored procedure function _close() > */ { > $parameterList = implode(',',$parameterMarkers); $ret = @db2_close($this->_connectionID); > $this->_connectionID = false; > $sql = sprintf($preparedStatement,$procedureName,$parameterList); return $ret; > } > $spResource = @db2_prepare($this->_connectionID,$sql); > function _affectedrows() > if (!$spResource) { > { return $this->_lastAffectedRows; > $errorMessage = @db2_conn_errormsg($this->_connectionID); } > $this->_errorMsg = $errorMessage; > } > if ($this->debug) > ADOConnection::outp($errorMessage); /*-------------------------------------------------------------------------------------- > Class Name: Recordset > return false; --------------------------------------------------------------------------------------*/ > } > class ADORecordSet_db2 extends ADORecordSet { > $this->storedProcedureParameters['resource'] = $spResource; > var $bind = false; > if ($this->debug) var $databaseType = "db2"; > { var $dataProvider = "db2"; > var $useFetchArray; > ADOConnection::outp('The following parameters will be used in the SP call'); > ADOConnection::outp(print_r($this->storedProcedureParameters)); function __construct($id,$mode=false) > } { > /* if ($mode === false) { > * We now have a stored parameter resource global $ADODB_FETCH_MODE; > * to bind to. The spResource and sql that is returned are $mode = $ADODB_FETCH_MODE; > * not usable, its for dummy compatibility. Everything } > * will be handled by the storedProcedureParameters $this->fetchMode = $mode; > * array > */ $this->_queryID = $id; > return array($sql,$spResource); } > > } > // returns the field object > private function storedProcedureParameter(&$stmt, function FetchField($offset = -1) > &$var, { > $name, $o= new ADOFieldObject(); > $isOutput=false, $o->name = @db2_field_name($this->_queryID,$offset); > $maxLen=4000, $o->type = @db2_field_type($this->_queryID,$offset); > $type=false) $o->max_length = db2_field_width($this->_queryID,$offset); > { if (ADODB_ASSOC_CASE == 0) $o->name = strtolower($o->name); > else if (ADODB_ASSOC_CASE == 1) $o->name = strtoupper($o->name); > return $o; > $name = strtoupper($name); } > > /* /* Use associative array to get fields array */ > * Must exist in the list of parameter names for the type function Fields($colname) > */ { > if ($isOutput if ($this->fetchMode & ADODB_FETCH_ASSOC) return $this->fields[$colname]; > && !isset( $this->storedProcedureParameters['out'][$name])) if (!$this->bind) { > { $this->bind = array(); > $errorMessage = sprintf('%s is not a valid OUT parameter name',$name); for ($i=0; $i < $this->_numOfFields; $i++) { > $o = $this->FetchField($i); > $this->_errorMsg = $errorMessage; $this->bind[strtoupper($o->name)] = $i; > if ($this->debug) } > ADOConnection::outp($errorMessage); } > return false; > } return $this->fields[$this->bind[strtoupper($colname)]]; > } > if (!$isOutput > && !isset( $this->storedProcedureParameters['in'][$name])) > { function _initrs() > $errorMessage = sprintf('%s is not a valid IN parameter name',$name); { > global $ADODB_COUNTRECS; > $this->_errorMsg = $errorMessage; $this->_numOfRows = ($ADODB_COUNTRECS) ? @db2_num_rows($this->_queryID) : -1; > if ($this->debug) $this->_numOfFields = @db2_num_fields($this->_queryID); > ADOConnection::outp($errorMessage); // some silly drivers such as db2 as/400 and intersystems cache return _numOfRows = 0 > return false; if ($this->_numOfRows == 0) $this->_numOfRows = -1; > } } > > /* function _seek($row) > * We will use these values to bind to when we execute { > * the query return false; > */ } > $this->storedProcedureParameters['keyvalue'][$name] = &$var; > // speed up SelectLimit() by switching to ADODB_FETCH_NUM as ADODB_FETCH_ASSOC is emulated > return true; function GetArrayLimit($nrows,$offset=-1) > { > } if ($offset <= 0) { > $rs = $this->GetArray($nrows); > /** return $rs; > * Executes a prepared stored procedure. } > * $savem = $this->fetchMode; > * The function uses the previously accumulated information and $this->fetchMode = ADODB_FETCH_NUM; > * resources in the $storedProcedureParameters array $this->Move($offset); > * $this->fetchMode = $savem; > * @return mixed The statement id if successful, or false > */ if ($this->fetchMode & ADODB_FETCH_ASSOC) { > private function executeStoredProcedure() $this->fields = $this->GetRowAssoc(); > { } > > /* $results = array(); > * Get the previously built resource $cnt = 0; > */ while (!$this->EOF && $nrows != $cnt) { > $stmtid = $this->storedProcedureParameters['resource']; $results[$cnt++] = $this->fields; > $this->MoveNext(); > /* } > * Bind our variables to the DB2 procedure > */ return $results; > foreach ($this->storedProcedureParameters['keyvalue'] as $spName=>$spValue){ } > > /* > * Get the ordinal position, required for binding function MoveNext() > */ { > $ordinalPosition = $this->storedProcedureParameters['index'][$spName]; if ($this->_numOfRows != 0 && !$this->EOF) { > $this->_currentRow++; > /* > * Get the db2 column dictionary for the parameter $this->fields = @db2_fetch_array($this->_queryID); > */ if ($this->fields) { > $columnDictionary = $this->storedProcedureParameters['parameters'][$ordinalPosition]; if ($this->fetchMode & ADODB_FETCH_ASSOC) { > $parameterType = $columnDictionary[4]; $this->fields = $this->GetRowAssoc(); > $dataType = $columnDictionary[5]; } > $precision = $columnDictionary[10]; return true; > $scale = $columnDictionary[9]; } > } > $ok = @db2_bind_param ($this->storedProcedureParameters['resource'], $this->fields = false; > $ordinalPosition , $this->EOF = true; > $spName, return false; > $parameterType, } > $dataType, > $precision, function _fetch() > $scale { > ); > $this->fields = db2_fetch_array($this->_queryID); > if (!$ok) if ($this->fields) { > { if ($this->fetchMode & ADODB_FETCH_ASSOC) { > $this->_errorMsg = @db2_stmt_errormsg(); $this->fields = $this->GetRowAssoc(); > $this->_errorCode = @db2_stmt_error(); } > return true; > if ($this->debug) } > ADOConnection::outp($this->_errorMsg); $this->fields = false; > return false; return false; > } } > > if ($this->debug) function _close() > ADOConnection::outp("Correctly Bound parameter $spName to procedure"); { > return @db2_free_result($this->_queryID); > /* } > * Build a variable in the current environment that matches > * the parameter name } > */ > ${$spName} = $spValue; > > } > > /* > * All bound, execute > */ > > if (!@db2_execute($stmtid)) > { > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > > if ($this->debug) > ADOConnection::outp($this->_errorMsg); > return false; > } > > /* > * We now take the changed parameters back into the > * stored procedures array where we can query them later > * Remember that $spValue was passed in by reference, so we > * can access the value in the variable that was originally > * passed to inParameter or outParameter > */ > foreach ($this->storedProcedureParameters['keyvalue'] as $spName=>$spValue) > { > /* > * We make it available to the environment > */ > $spValue = ${$spName}; > $this->storedProcedureParameters['keyvalue'][$spName] = $spValue; > } > > return $stmtid; > } > > /** > * > * Accepts an input or output parameter to bind to either a stored > * or prepared statements. For DB2, this should not be called as an > * API. always wrap with inParameter and outParameter > * > * @param mixed[] $stmt Statement returned by Prepare() or PrepareSP(). > * @param mixed $var PHP variable to bind to. Can set to null (for isNull support). > * @param string $name Name of stored procedure variable name to bind to. > * @param int $isOutput optional) Indicates direction of parameter > * 0/false=IN 1=OUT 2= IN/OUT > * This is ignored for Stored Procedures > * @param int $maxLen (optional)Holds an maximum length of the variable. > * This is ignored for Stored Procedures > * @param int $type (optional) The data type of $var. > * This is ignored for Stored Procedures > * > * @return bool Success of the operation > */ > public function parameter(&$stmt, &$var, $name, $isOutput=false, $maxLen=4000, $type=false) > { > > /* > * If the $stmt is the name of a stored procedure we are > * setting up, we will process it one way, otherwise > * we assume we are setting up a prepared statement > */ > if (is_array($stmt)) > { > if ($this->debug) > ADOConnection::outp("Adding parameter to stored procedure"); > if ($stmt[1] == $this->storedProcedureParameters['resource']) > return $this->storedProcedureParameter($stmt[1], > $var, > $name, > $isOutput, > $maxLen, > $type); > > } > > /* > * We are going to add a parameter to a prepared statement > */ > if ($this->debug) > ADOConnection::outp("Adding parameter to prepared statement"); > } > > > /** > * Prepares a prepared SQL statement, not used for stored procedures > * > * @param string $sql > * > * @return mixed > */ > function prepare($sql) > { >
< $stmt = db2_prepare($this->_connectionID,$sql);
> > $stmt = @db2_prepare($this->_connectionID,$sql);
< /* returns queryID or false */ < function _query($sql,$inputarr=false)
> /** > * Executes a query > * > * @param mixed $sql > * @param mixed $inputarr An optional array of parameters > * > * @return mixed either the queryID or false > */ > function _query(&$sql,$inputarr=false)
< $last_php_error = $this->resetLastError(); < $this->_errorMsg = '';
< if ($inputarr) { < if (is_array($sql)) {
> $this->_error = ''; > > $db2Options = array(); > /* > * Use DB2 Internal case handling for best speed > */ > switch(ADODB_ASSOC_CASE) > { > case ADODB_ASSOC_CASE_UPPER: > $db2Options = array('db2_attr_case'=>DB2_CASE_UPPER); > $setOption = @db2_set_option($this->_connectionID,$db2Options,1); > break; > > case ADODB_ASSOC_CASE_LOWER: > $db2Options = array('db2_attr_case'=>DB2_CASE_LOWER); > $setOption = @db2_set_option($this->_connectionID,$db2Options,1); > break; > > default: > $db2Options = array('db2_attr_case'=>DB2_CASE_NATURAL); > $setOption = @db2_set_option($this->_connectionID,$db2Options,1); > } > > if ($inputarr) > { > if (is_array($sql)) > {
< } else { < $stmtid = db2_prepare($this->_connectionID,$sql);
> } > else > { > $stmtid = @db2_prepare($this->_connectionID,$sql); > > if ($stmtid == false) > { > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > > if ($this->debug) > ADOConnection::outp($this->_errorMsg);
< if ($stmtid == false) { < $this->_errorMsg = $this->getChangedErrorMsg($last_php_error);
< if (! db2_execute($stmtid,$inputarr)) { < if ($this->_haserrorfunctions) { < $this->_errorMsg = db2_stmt_errormsg(); < $this->_errorCode = db2_stmt_error(); < }
> if (! @db2_execute($stmtid,$inputarr)) > { > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > if ($this->debug) > ADOConnection::outp($this->_errorMsg);
< } else if (is_array($sql)) { < $stmtid = $sql[1]; < if (!db2_execute($stmtid)) { < if ($this->_haserrorfunctions) { < $this->_errorMsg = db2_stmt_errormsg(); < $this->_errorCode = db2_stmt_error();
> else if (is_array($sql)) > { > > /* > * Either a prepared statement or a stored procedure > */ > > if (is_array($this->storedProcedureParameters) > && is_resource($this->storedProcedureParameters['resource'] > )) > /* > * This is all handled in the separate method for > * readability > */ > return $this->executeStoredProcedure(); > > /* > * First, we prepare the statement > */ > $stmtid = @db2_prepare($this->_connectionID,$sql[0]); > if (!$stmtid){ > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > if ($this->debug) > ADOConnection::outp("Prepare failed: " . $this->_errorMsg); > > return false; > } > /* > * We next bind some input parameters > */ > $ordinal = 1; > foreach ($sql[1] as $psVar=>$psVal){ > ${$psVar} = $psVal; > $ok = @db2_bind_param($stmtid, $ordinal, $psVar, DB2_PARAM_IN); > if (!$ok) > { > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > if ($this->debug) > ADOConnection::outp("Bind failed: " . $this->_errorMsg);
< } else < $stmtid = @db2_exec($this->_connectionID,$sql);
> } > > if (!@db2_execute($stmtid)) > { > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); > if ($this->debug) > ADOConnection::outp($this->_errorMsg); > return false; > } > > return $stmtid; > } > else > {
> $stmtid = @db2_exec($this->_connectionID,$sql); > }
< if ($stmtid) { < if (@db2_num_fields($stmtid) == 0) {
> if ($stmtid) > { > if (@db2_num_fields($stmtid) == 0) > {
< } else {
> } > else > {
< if ($this->_haserrorfunctions) {
< } else { < $this->_errorMsg = $this->getChangedErrorMsg($last_php_error); < } < } else { < if ($this->_haserrorfunctions) { < $this->_errorMsg = db2_stmt_errormsg(); < $this->_errorCode = db2_stmt_error(); < } else { < $this->_errorMsg = $this->getChangedErrorMsg($last_php_error);
>
> else > { > > $this->_errorMsg = @db2_stmt_errormsg(); > $this->_errorCode = @db2_stmt_error(); >
< $conn->Execute('INSERT INTO blobtable (id, blobcol) VALUES (1, null)');
> $conn->execute('INSERT INTO blobtable (id, blobcol) VALUES (1, null)');
< function UpdateBlob($table,$column,$val,$where,$blobtype='BLOB')
> function updateBlob($table,$column,$val,$where,$blobtype='BLOB')
< return $this->Execute("UPDATE $table SET $column=? WHERE $where",array($val)) != false;
> return $this->execute("UPDATE $table SET $column=? WHERE $where",array($val)) != false;
> /** > * Gets a meta cased parameter > * > * Receives an input variable to be processed per the metaCasing > * rule, and returns the same value, processed > * > * @param string $value > * > * @return string > */ > final public function getMetaCasedValue($value) > { > global $ADODB_ASSOC_CASE; > > switch($ADODB_ASSOC_CASE) > { > case ADODB_ASSOC_CASE_LOWER: > $value = strtolower($value); > break; > case ADODB_ASSOC_CASE_UPPER: > $value = strtoupper($value); > break; > } > return $value; > } > > > const TABLECASE_LOWER = 0; > const TABLECASE_UPPER = 1; > const TABLECASE_DEFAULT = 2; > > /** > * Controls the casing of the table provided to the meta functions > */ > private $tableCase = 2; > > /** > * Sets the table case parameter > * > * @param int $caseOption > * @return null > */ > final public function setTableCasing($caseOption) > { > $this->tableCase = $caseOption; > } > > /** > * Gets the table casing parameter > * > * @return int $caseOption > */ > final public function getTableCasing() > { > return $this->tableCase; > } > > /** > * Gets a table cased parameter > * > * Receives an input variable to be processed per the tableCasing > * rule, and returns the same value, processed > * > * @param string $value > * > * @return string > */ > final public function getTableCasedValue($value) > { > switch($this->tableCase) > { > case self::TABLECASE_LOWER: > $value = strtolower($value); > break; > case self::TABLECASE_UPPER: > $value = strtoupper($value); > break; > } > return $value; > } >
< function FetchField($offset = -1)
> function fetchField($offset = 0)
< $o->max_length = db2_field_width($this->_queryID,$offset); < if (ADODB_ASSOC_CASE == 0) $o->name = strtolower($o->name); < else if (ADODB_ASSOC_CASE == 1) $o->name = strtoupper($o->name);
> $o->max_length = @db2_field_width($this->_queryID,$offset); > > /* > if (ADODB_ASSOC_CASE == 0) > $o->name = strtolower($o->name); > else if (ADODB_ASSOC_CASE == 1) > $o->name = strtoupper($o->name); > */
< function Fields($colname)
> function fields($colname)
< if ($this->fetchMode & ADODB_FETCH_ASSOC) return $this->fields[$colname];
> > if ($this->fetchMode & ADODB_FETCH_ASSOC) { > return $this->fields[$colname]; > } >
>
>
< if ($this->_numOfRows == 0) $this->_numOfRows = -1;
> > if ($this->_numOfRows == 0) > $this->_numOfRows = -1;
< // speed up SelectLimit() by switching to ADODB_FETCH_NUM as ADODB_FETCH_ASSOC is emulated < function GetArrayLimit($nrows,$offset=-1)
> function getArrayLimit($nrows,$offset=0)
< $savem = $this->fetchMode; < $this->fetchMode = ADODB_FETCH_NUM;
>
< $this->fetchMode = $savem;
< if ($this->fetchMode & ADODB_FETCH_ASSOC) { < $this->fields = $this->GetRowAssoc(); < }
< < function MoveNext()
> function moveNext()
< if ($this->_numOfRows != 0 && !$this->EOF) {
> if ($this->EOF || $this->_numOfRows == 0) > return false; >
< $this->fields = @db2_fetch_array($this->_queryID); < if ($this->fields) { < if ($this->fetchMode & ADODB_FETCH_ASSOC) { < $this->fields = $this->GetRowAssoc();
> $this->processCoreFetch(); > return $this->processMoveRecord(); >
< return true;
> > private function processCoreFetch() > { > switch ($this->fetchMode){ > case ADODB_FETCH_ASSOC: > > /* > * Associative array > */ > $this->fields = @db2_fetch_assoc($this->_queryID); > break; > > case ADODB_FETCH_BOTH: > /* > * Fetch both numeric and Associative array > */ > $this->fields = @db2_fetch_both($this->_queryID); > break; > default: > /* > * Numeric array > */ > $this->fields = @db2_fetch_array($this->_queryID); > break;
< $this->fields = false;
> > private function processMoveRecord() > { > if (!$this->fields){
> return true; > } >
< < $this->fields = db2_fetch_array($this->_queryID); < if ($this->fields) { < if ($this->fetchMode & ADODB_FETCH_ASSOC) { < $this->fields = $this->GetRowAssoc(); < }
> $this->processCoreFetch(); > if ($this->fields)
< }
>
< return @db2_free_result($this->_queryID);
> $ok = @db2_free_result($this->_queryID); > if (!$ok) > { > $this->_errorMsg = @db2_stmt_errormsg($this->_queryId); > $this->_errorCode = @db2_stmt_error(); > > if ($this->debug) > ADOConnection::outp($this->_errorMsg); > return false; > } >