2006-05-30 13:08:46 +04:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
V4.80 8 Mar 2006 (c) 2000-2006 John Lim (jlim@natsoft.com.my). All rights reserved.
|
|
|
|
Released under both BSD license and Lesser GPL library license.
|
|
|
|
Whenever there is any discrepancy between the two licenses,
|
|
|
|
the BSD license will take precedence.
|
|
|
|
|
|
|
|
Set tabs to 4 for best viewing.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
// security - hide paths
|
|
|
|
if (!defined('ADODB_DIR')) die();
|
|
|
|
|
|
|
|
class ADODB2_mysql extends ADODB_DataDict {
|
|
|
|
var $databaseType = 'mysql';
|
|
|
|
var $alterCol = ' MODIFY COLUMN';
|
|
|
|
var $alterTableAddIndex = true;
|
|
|
|
var $dropTable = 'DROP TABLE IF EXISTS %s'; // requires mysql 3.22 or later
|
|
|
|
|
|
|
|
var $dropIndex = 'DROP INDEX %s ON %s';
|
|
|
|
var $renameColumn = 'ALTER TABLE %s CHANGE COLUMN %s %s %s'; // needs column-definition!
|
|
|
|
var $metaColumnsSQL = "SHOW COLUMNS FROM %s";
|
|
|
|
|
|
|
|
function MetaType($t,$len=-1,$fieldobj=false)
|
|
|
|
{
|
|
|
|
if (is_object($t)) {
|
|
|
|
$fieldobj = $t;
|
|
|
|
$t = $fieldobj->type;
|
|
|
|
$len = $fieldobj->max_length;
|
|
|
|
}
|
|
|
|
$is_serial = is_object($fieldobj) && $fieldobj->primary_key && $fieldobj->auto_increment;
|
|
|
|
|
|
|
|
$len = -1; // mysql max_length is not accurate
|
|
|
|
switch (strtoupper($t)) {
|
|
|
|
case 'STRING':
|
|
|
|
case 'CHAR':
|
|
|
|
case 'VARCHAR':
|
|
|
|
case 'TINYBLOB':
|
|
|
|
case 'TINYTEXT':
|
|
|
|
case 'ENUM':
|
|
|
|
case 'SET':
|
|
|
|
if ($len <= $this->blobSize) return 'C';
|
|
|
|
|
|
|
|
case 'TEXT':
|
|
|
|
case 'LONGTEXT':
|
|
|
|
case 'MEDIUMTEXT':
|
|
|
|
return 'X';
|
|
|
|
|
|
|
|
// php_mysql extension always returns 'blob' even if 'text'
|
|
|
|
// so we have to check whether binary...
|
|
|
|
case 'IMAGE':
|
|
|
|
case 'LONGBLOB':
|
|
|
|
case 'BLOB':
|
|
|
|
case 'MEDIUMBLOB':
|
|
|
|
return !empty($fieldobj->binary) ? 'B' : 'X';
|
|
|
|
|
|
|
|
case 'YEAR':
|
|
|
|
case 'DATE': return 'D';
|
|
|
|
|
|
|
|
case 'TIME':
|
|
|
|
case 'DATETIME':
|
|
|
|
case 'TIMESTAMP': return 'T';
|
|
|
|
|
|
|
|
case 'FLOAT':
|
|
|
|
case 'DOUBLE':
|
|
|
|
return 'F';
|
|
|
|
|
|
|
|
case 'INT':
|
|
|
|
case 'INTEGER': return $is_serial ? 'R' : 'I';
|
|
|
|
case 'TINYINT': return $is_serial ? 'R' : 'I1';
|
|
|
|
case 'SMALLINT': return $is_serial ? 'R' : 'I2';
|
|
|
|
case 'MEDIUMINT': return $is_serial ? 'R' : 'I4';
|
|
|
|
case 'BIGINT': return $is_serial ? 'R' : 'I8';
|
|
|
|
default: return 'N';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function ActualType($meta)
|
|
|
|
{
|
|
|
|
switch(strtoupper($meta)) {
|
|
|
|
case 'C': return 'VARCHAR';
|
|
|
|
case 'XL':return 'LONGTEXT';
|
|
|
|
case 'X': return 'TEXT';
|
|
|
|
|
|
|
|
case 'C2': return 'VARCHAR';
|
|
|
|
case 'X2': return 'LONGTEXT';
|
|
|
|
|
|
|
|
case 'B': return 'LONGBLOB';
|
|
|
|
|
|
|
|
case 'D': return 'DATE';
|
|
|
|
case 'T': return 'DATETIME';
|
|
|
|
case 'L': return 'TINYINT';
|
|
|
|
|
|
|
|
case 'R':
|
|
|
|
case 'I4':
|
|
|
|
case 'I': return 'INTEGER';
|
|
|
|
case 'I1': return 'TINYINT';
|
|
|
|
case 'I2': return 'SMALLINT';
|
|
|
|
case 'I8': return 'BIGINT';
|
|
|
|
|
|
|
|
case 'F': return 'DOUBLE';
|
|
|
|
case 'N': return 'NUMERIC';
|
|
|
|
default:
|
|
|
|
return $meta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return string must begin with space
|
|
|
|
function _CreateSuffix($fname,$ftype,$fnotnull,$fdefault,$fautoinc,$fconstraint,$funsigned)
|
|
|
|
{
|
|
|
|
$suffix = '';
|
|
|
|
if ($funsigned) $suffix .= ' UNSIGNED';
|
|
|
|
if ($fnotnull) $suffix .= ' NOT NULL';
|
|
|
|
if (strlen($fdefault)) $suffix .= " DEFAULT $fdefault";
|
|
|
|
if ($fautoinc) $suffix .= ' AUTO_INCREMENT';
|
|
|
|
if ($fconstraint) $suffix .= ' '.$fconstraint;
|
|
|
|
return $suffix;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tbl_name [(create_definition,...)]
|
|
|
|
[table_options] [select_statement]
|
|
|
|
create_definition:
|
|
|
|
col_name type [NOT NULL | NULL] [DEFAULT default_value] [AUTO_INCREMENT]
|
|
|
|
[PRIMARY KEY] [reference_definition]
|
|
|
|
or PRIMARY KEY (index_col_name,...)
|
|
|
|
or KEY [index_name] (index_col_name,...)
|
|
|
|
or INDEX [index_name] (index_col_name,...)
|
|
|
|
or UNIQUE [INDEX] [index_name] (index_col_name,...)
|
|
|
|
or FULLTEXT [INDEX] [index_name] (index_col_name,...)
|
|
|
|
or [CONSTRAINT symbol] FOREIGN KEY [index_name] (index_col_name,...)
|
|
|
|
[reference_definition]
|
|
|
|
or CHECK (expr)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
CREATE [UNIQUE|FULLTEXT] INDEX index_name
|
|
|
|
ON tbl_name (col_name[(length)],... )
|
|
|
|
*/
|
|
|
|
|
|
|
|
function _IndexSQL($idxname, $tabname, $flds, $idxoptions)
|
|
|
|
{
|
|
|
|
$sql = array();
|
|
|
|
|
|
|
|
if ( isset($idxoptions['REPLACE']) || isset($idxoptions['DROP']) ) {
|
|
|
|
if ($this->alterTableAddIndex) $sql[] = "ALTER TABLE $tabname DROP INDEX $idxname";
|
|
|
|
else $sql[] = sprintf($this->dropIndex, $idxname, $tabname);
|
|
|
|
|
|
|
|
if ( isset($idxoptions['DROP']) )
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( empty ($flds) ) {
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($idxoptions['FULLTEXT'])) {
|
|
|
|
$unique = ' FULLTEXT';
|
|
|
|
} elseif (isset($idxoptions['UNIQUE'])) {
|
|
|
|
$unique = ' UNIQUE';
|
|
|
|
} else {
|
|
|
|
$unique = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( is_array($flds) ) $flds = implode(', ',$flds);
|
|
|
|
|
|
|
|
if ($this->alterTableAddIndex) $s = "ALTER TABLE $tabname ADD $unique INDEX $idxname ";
|
|
|
|
else $s = 'CREATE' . $unique . ' INDEX ' . $idxname . ' ON ' . $tabname;
|
|
|
|
|
|
|
|
$s .= ' (' . $flds . ')';
|
|
|
|
|
|
|
|
if ( isset($idxoptions[$this->upperName]) )
|
|
|
|
$s .= $idxoptions[$this->upperName];
|
|
|
|
|
|
|
|
$sql[] = $s;
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
2006-09-02 12:38:03 +04:00
|
|
|
/**
|
|
|
|
* @param ttype can either be 'VIEW' or 'TABLE' or false.
|
|
|
|
* If false, both views and tables are returned.
|
|
|
|
* "VIEW" returns only views
|
|
|
|
* "TABLE" returns only tables
|
|
|
|
* @param showSchema returns the schema/user with the table name, eg. USER.TABLE
|
|
|
|
* @param mask is the input mask - only supported by oci8 and postgresql
|
|
|
|
*
|
|
|
|
* @return array of tables for current database.
|
|
|
|
*/
|
|
|
|
function &MetaTables($ttype=false,$showSchema=false,$mask=false)
|
|
|
|
{
|
|
|
|
$save = $this->metaTablesSQL;
|
|
|
|
if ($showSchema && is_string($showSchema)) {
|
|
|
|
$this->metaTablesSQL .= " from $showSchema";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($mask) {
|
|
|
|
$mask = $this->connection->qstr($mask);
|
|
|
|
$this->metaTablesSQL .= " like $mask";
|
|
|
|
}
|
|
|
|
$ret =& $this->_MetaTables($ttype,$showSchema);
|
|
|
|
|
|
|
|
$this->metaTablesSQL = $save;
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function &_MetaTables($ttype=false,$showSchema=false,$mask=false)
|
|
|
|
{
|
|
|
|
$false = false;
|
|
|
|
if ($mask) {
|
|
|
|
return $false;
|
|
|
|
}
|
|
|
|
if ($this->metaTablesSQL) {
|
|
|
|
|
|
|
|
$rs = $this->connection->query($this->metaTablesSQL);
|
|
|
|
|
|
|
|
$arr =& $rs->fetchAll(PDO::FETCH_NUM);
|
|
|
|
$arr2 = array();
|
|
|
|
|
|
|
|
if ($hast = ($ttype && isset($arr[0][1]))) {
|
|
|
|
$showt = strncmp($ttype,'T',1);
|
|
|
|
}
|
|
|
|
|
|
|
|
for ($i=0; $i < sizeof($arr); $i++) {
|
|
|
|
if ($hast) {
|
|
|
|
if ($showt == 0) {
|
|
|
|
if (strncmp($arr[$i][1],'T',1) == 0) $arr2[] = trim($arr[$i][0]);
|
|
|
|
} else {
|
|
|
|
if (strncmp($arr[$i][1],'V',1) == 0) $arr2[] = trim($arr[$i][0]);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
$arr2[] = trim($arr[$i][0]);
|
|
|
|
}
|
|
|
|
$rs->closeCursor();
|
|
|
|
return $arr2;
|
|
|
|
}
|
|
|
|
return $false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List columns in a database as an array of ADOFieldObjects.
|
|
|
|
* See top of file for definition of object.
|
|
|
|
*
|
|
|
|
* @param table table name to query
|
|
|
|
* @param upper uppercase table name (required by some databases)
|
|
|
|
* @schema is optional database schema to use - not supported by all databases.
|
|
|
|
*
|
|
|
|
* @return array of ADOFieldObjects for current table.
|
|
|
|
*/
|
|
|
|
function MetaColumns($table)
|
|
|
|
{
|
|
|
|
$this->_findschema($table,$schema);
|
|
|
|
if ($schema) {
|
|
|
|
$dbName = $this->database;
|
|
|
|
$this->connection->SelectDB($schema);
|
|
|
|
}
|
|
|
|
|
|
|
|
$stmt = $this->connection->query(sprintf($this->metaColumnsSQL,$table));
|
|
|
|
|
|
|
|
if ($schema) {
|
|
|
|
$this->connection->SelectDB($dbName);
|
|
|
|
}
|
|
|
|
|
|
|
|
$retarr = array();
|
|
|
|
while ($rs = $stmt->fetch(PDO::FETCH_NUM)){
|
|
|
|
print_r($rs);
|
|
|
|
$fld = new ADOFieldObject();
|
|
|
|
$fld->name = $rs[0];
|
|
|
|
$type = $rs[1];
|
|
|
|
|
|
|
|
// split type into type(length):
|
|
|
|
$fld->scale = null;
|
|
|
|
if (preg_match("/^(.+)\((\d+),(\d+)/", $type, $query_array)) {
|
|
|
|
$fld->type = $query_array[1];
|
|
|
|
$fld->max_length = is_numeric($query_array[2]) ? $query_array[2] : -1;
|
|
|
|
$fld->scale = is_numeric($query_array[3]) ? $query_array[3] : -1;
|
|
|
|
} elseif (preg_match("/^(.+)\((\d+)/", $type, $query_array)) {
|
|
|
|
$fld->type = $query_array[1];
|
|
|
|
$fld->max_length = is_numeric($query_array[2]) ? $query_array[2] : -1;
|
|
|
|
} elseif (preg_match("/^(enum)\((.*)\)$/i", $type, $query_array)) {
|
|
|
|
$fld->type = $query_array[1];
|
|
|
|
$arr = explode(",",$query_array[2]);
|
|
|
|
$fld->enums = $arr;
|
|
|
|
$zlen = max(array_map("strlen",$arr)) - 2; // PHP >= 4.0.6
|
|
|
|
$fld->max_length = ($zlen > 0) ? $zlen : 1;
|
|
|
|
} else {
|
|
|
|
$fld->type = $type;
|
|
|
|
$fld->max_length = -1;
|
|
|
|
}
|
|
|
|
$fld->not_null = ($rs[2] != 'YES');
|
|
|
|
$fld->primary_key = ($rs[3] == 'PRI');
|
|
|
|
$fld->auto_increment = (strpos($rs[5], 'auto_increment') !== false);
|
|
|
|
$fld->binary = (strpos($type,'blob') !== false);
|
|
|
|
$fld->unsigned = (strpos($type,'unsigned') !== false);
|
|
|
|
|
|
|
|
if (!$fld->binary) {
|
|
|
|
$d = $rs[4];
|
|
|
|
if ($d != '' && $d != 'NULL') {
|
|
|
|
$fld->has_default = true;
|
|
|
|
$fld->default_value = $d;
|
|
|
|
} else {
|
|
|
|
$fld->has_default = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($save == ADODB_FETCH_NUM) {
|
|
|
|
$retarr[] = $fld;
|
|
|
|
} else {
|
|
|
|
$retarr[strtoupper($fld->name)] = $fld;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
$stmt->closeCursor();
|
|
|
|
return $retarr;
|
|
|
|
}
|
2006-05-30 13:08:46 +04:00
|
|
|
}
|
2006-09-04 02:46:30 +04:00
|
|
|
|