2006-09-21 21:09:58 +00:00
< ? php
/*
* $Id $
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT
* LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*
* This software consists of voluntary contributions made by many individuals
* and is licensed under the LGPL . For more information , see
* < http :// www . phpdoctrine . com >.
*/
/**
* Doctrine_DB
* A thin layer on top of PDO
*
* @ author Konsta Vesterinen
* @ license LGPL
* @ package Doctrine
*/
2006-09-22 21:20:21 +00:00
class Doctrine_DB2 implements Countable , IteratorAggregate {
2006-11-05 19:24:28 +00:00
/**
* A connection operation or selecting a database .
*/
const CONNECT = 1 ;
/**
* Any general database query that does not fit into the other constants .
*/
const QUERY = 2 ;
/**
* Adding new data to the database , such as SQL ' s INSERT .
*/
const INSERT = 4 ;
/**
* Updating existing information in the database , such as SQL ' s UPDATE .
*
*/
const UPDATE = 8 ;
/**
* An operation related to deleting data in the database ,
* such as SQL ' s DELETE .
*/
const DELETE = 16 ;
/**
* Retrieving information from the database , such as SQL ' s SELECT .
*/
const SELECT = 32 ;
/**
* Transactional operation , such as start transaction , commit , or rollback .
*/
const TRANSACTION = 64 ;
2006-09-21 21:09:58 +00:00
/**
* default DSN
*/
const DSN = " mysql://root:dc34@localhost/test " ;
/**
* @ var array $instances all the instances of this class
*/
protected static $instances = array ();
/**
* @ var array $isConnected whether or not a connection has been established
*/
protected $isConnected = false ;
/**
* @ var string $dsn data source name
*/
protected $dsn ;
/**
* @ var string $username database username
*/
protected $username ;
/**
* @ var string $password database password
*/
protected $password ;
2006-09-22 22:16:12 +00:00
/**
2006-09-21 21:09:58 +00:00
* @ var PDO $dbh the database handler
*/
2006-09-22 22:16:12 +00:00
protected $dbh ;
/**
2006-09-23 10:44:39 +00:00
* @ var Doctrine_DB_EventListener_Interface | Doctrine_Overloadable $listener listener for listening events
2006-09-22 22:16:12 +00:00
*/
protected $listener ;
2006-11-05 19:24:28 +00:00
/**
* @ var integer $querySequence
*/
protected $querySequence = 0 ;
private static $driverMap = array ( 'oracle' => 'oci8' ,
'postgres' => 'pgsql' ,
'oci' => 'oci8' ,
'sqlite2' => 'sqlite' ,
'sqlite3' => 'sqlite' );
2006-09-22 22:16:12 +00:00
2006-09-21 21:09:58 +00:00
/**
* constructor
*
* @ param string $dsn data source name
* @ param string $username database username
* @ param string $password database password
*/
public function __construct ( $dsn , $username , $password ) {
$this -> dsn = $dsn ;
$this -> username = $username ;
$this -> password = $password ;
2006-09-22 22:16:12 +00:00
$this -> listener = new Doctrine_DB_EventListener ();
2006-09-21 21:09:58 +00:00
}
2006-11-05 19:24:28 +00:00
/**
* getQuerySequence
*/
public function getQuerySequence () {
return $this -> querySequence ;
}
2006-09-23 10:44:39 +00:00
/**
* getDBH
*/
public function getDBH () {
return $this -> dbh ;
}
2006-09-21 21:09:58 +00:00
/**
* getDSN
2006-09-23 10:44:39 +00:00
* returns the data source name
2006-09-21 21:09:58 +00:00
*
* @ return string
*/
public function getDSN () {
return $this -> dsn ;
}
2006-09-23 10:44:39 +00:00
/**
2006-09-21 21:09:58 +00:00
* getUsername
2006-09-27 10:55:02 +00:00
*
* @ return string
2006-09-21 21:09:58 +00:00
*/
public function getUsername () {
return $this -> username ;
}
/**
* getPassword
2006-09-27 10:55:02 +00:00
*
* @ return string
2006-09-21 21:09:58 +00:00
*/
public function getPassword () {
return $this -> password ;
}
2006-09-23 10:44:39 +00:00
/**
* addListener
*
* @ param Doctrine_DB_EventListener_Interface | Doctrine_Overloadable $listener
* @ return Doctrine_DB
*/
public function addListener ( $listener , $name = null ) {
if ( ! ( $this -> listener instanceof Doctrine_DB_EventListener_Chain ))
$this -> listener = new Doctrine_DB_EventListener_Chain ();
$this -> listener -> add ( $listener , $name );
return $this ;
}
/**
* getListener
*
* @ return Doctrine_DB_EventListener_Interface | Doctrine_Overloadable
*/
public function getListener () {
return $this -> listener ;
}
/**
* setListener
*
* @ param Doctrine_DB_EventListener_Interface | Doctrine_Overloadable $listener
* @ return Doctrine_DB
*/
public function setListener ( $listener ) {
if ( ! ( $listener instanceof Doctrine_DB_EventListener_Interface ) &&
! ( $listener instanceof Doctrine_Overloadable ))
throw new Doctrine_DB_Exception ( " Couldn't set eventlistener for database handler. EventListeners should implement either Doctrine_DB_EventListener_Interface or Doctrine_Overloadable " );
$this -> listener = $listener ;
return $this ;
}
2006-09-21 21:09:58 +00:00
2006-09-23 10:44:39 +00:00
/**
2006-09-21 21:09:58 +00:00
* connect
* connects into database
*
* @ return boolean
*/
public function connect () {
2006-09-23 10:44:39 +00:00
if ( $this -> isConnected )
2006-09-21 21:09:58 +00:00
return false ;
$this -> dbh = new PDO ( $this -> dsn , $this -> username , $this -> password );
$this -> dbh -> setAttribute ( PDO :: ATTR_ERRMODE , PDO :: ERRMODE_EXCEPTION );
2006-09-23 10:44:39 +00:00
$this -> dbh -> setAttribute ( PDO :: ATTR_STATEMENT_CLASS , array ( " Doctrine_DB_Statement " , array ( $this )));
2006-09-27 10:55:02 +00:00
$this -> isConnected = true ;
2006-09-21 21:09:58 +00:00
return true ;
}
/**
* getConnection
*
2006-09-22 22:16:12 +00:00
* @ param string $dsn PEAR :: DB like DSN or PDO like DSN
* format for PEAR :: DB like DSN : schema :// user : password @ address / dbname
2006-09-21 21:09:58 +00:00
*
* @ return
*/
public static function getConnection ( $dsn = null , $username = null , $password = null ) {
$md5 = md5 ( $dsn );
if ( ! isset ( self :: $instances [ $md5 ])) {
2006-09-22 22:16:12 +00:00
if ( isset ( $username )) {
self :: $instances [ $md5 ] = new self ( $dsn , $username , $password );
2006-09-21 21:09:58 +00:00
} else {
2006-09-22 22:16:12 +00:00
if ( ! isset ( $dsn ))
$a = self :: parseDSN ( self :: DSN );
else
$a = self :: parseDSN ( $dsn );
2006-09-21 21:09:58 +00:00
2006-09-22 22:16:12 +00:00
extract ( $a );
self :: $instances [ $md5 ] = new self ( $dsn , $user , $pass );
}
2006-09-21 21:09:58 +00:00
}
return self :: $instances [ $md5 ];
}
2006-09-24 10:00:44 +00:00
/**
* driverName
* converts a driver name like ( oracle ) to appropriate PDO
* driver name ( oci8 in the case of oracle )
*
* @ param string $name
* @ return string
*/
public static function driverName ( $name ) {
if ( isset ( self :: $driverMap [ $name ]))
return self :: $driverMap [ $name ];
return $name ;
2006-09-23 10:44:39 +00:00
}
/**
* parseDSN
*
* @ param string $dsn
* @ return array Parsed contents of DSN
*/
function parseDSN ( $dsn ) {
2006-09-24 10:00:44 +00:00
// silence any warnings
2006-09-23 10:44:39 +00:00
$parts = @ parse_url ( $dsn );
$names = array ( 'scheme' , 'host' , 'port' , 'user' , 'pass' , 'path' , 'query' , 'fragment' );
foreach ( $names as $name ) {
if ( ! isset ( $parts [ $name ]))
$parts [ $name ] = null ;
}
if ( count ( $parts ) == 0 || ! isset ( $parts [ 'scheme' ]))
throw new Doctrine_DB_Exception ( 'Empty data source name' );
$drivers = self :: getAvailableDrivers ();
2006-09-24 10:00:44 +00:00
$parts [ 'scheme' ] = self :: driverName ( $parts [ 'scheme' ]);
2006-09-23 10:44:39 +00:00
if ( ! in_array ( $parts [ 'scheme' ], $drivers ))
throw new Doctrine_DB_Exception ( 'Driver ' . $parts [ 'scheme' ] . ' not availible or extension not loaded' );
switch ( $parts [ 'scheme' ]) {
case 'sqlite' :
if ( isset ( $parts [ 'host' ]) && $parts [ 'host' ] == ':memory' ) {
$parts [ 'database' ] = ':memory:' ;
$parts [ 'dsn' ] = 'sqlite::memory:' ;
}
break ;
case 'mysql' :
case 'informix' :
case 'oci8' :
case 'mssql' :
case 'firebird' :
case 'pgsql' :
case 'odbc' :
if ( ! isset ( $parts [ 'path' ]) || $parts [ 'path' ] == '/' )
throw new Doctrine_DB_Exception ( 'No database availible in data source name' );
if ( isset ( $parts [ 'path' ]))
$parts [ 'database' ] = substr ( $parts [ 'path' ], 1 );
if ( ! isset ( $parts [ 'host' ]))
throw new Doctrine_DB_Exception ( 'No hostname set in data source name' );
$parts [ 'dsn' ] = $parts [ " scheme " ] . " :host= " . $parts [ " host " ] . " ;dbname= " . $parts [ " database " ];
break ;
default :
throw new Doctrine_DB_Exception ( 'Unknown driver ' . $parts [ 'scheme' ]);
}
return $parts ;
}
2006-09-21 21:09:58 +00:00
/**
* clear
* clears all instances from the memory
*
* @ return void
*/
public static function clear () {
self :: $instances = array ();
}
/**
* errorCode
* Fetch the SQLSTATE associated with the last operation on the database handle
*
* @ return integer
*/
2006-09-22 22:16:12 +00:00
public function errorCode () {
2006-09-21 21:09:58 +00:00
return $this -> dbh -> errorCode ();
}
/**
* errorInfo
* Fetch extended error information associated with the last operation on the database handle
*
* @ return array
*/
public function errorInfo () {
return $this -> dbh -> errorInfo ();
}
/**
2006-09-22 22:16:12 +00:00
* prepare
2006-09-21 21:09:58 +00:00
*
* @ param string $statement
*/
2006-09-22 22:16:12 +00:00
public function prepare ( $statement ) {
2006-09-27 10:55:02 +00:00
$this -> connect ();
2006-09-23 10:44:39 +00:00
$args = func_get_args ();
2006-11-05 19:24:28 +00:00
$this -> listener -> onPrePrepare ( $this , $statement , $args );
2006-09-22 22:16:12 +00:00
$stmt = $this -> dbh -> prepare ( $statement );
2006-09-23 10:44:39 +00:00
2006-11-05 19:24:28 +00:00
$this -> listener -> onPrepare ( $this , $statement , $args , $this -> querySequence );
$this -> querySequence ++ ;
2006-09-22 22:16:12 +00:00
return $stmt ;
2006-09-21 21:09:58 +00:00
}
/**
* query
*
* @ param string $statement
2006-11-05 19:24:28 +00:00
* @ param array $params
2006-09-21 21:09:58 +00:00
* @ return Doctrine_DB_Statement | boolean
*/
2006-09-27 10:55:02 +00:00
public function query ( $statement , array $params = array ()) {
$this -> connect ();
2006-09-21 21:09:58 +00:00
2006-11-05 19:24:28 +00:00
$this -> listener -> onPreQuery ( $this , $statement , $params );
2006-09-27 10:55:02 +00:00
if ( ! empty ( $params ))
$stmt = $this -> dbh -> query ( $statement ) -> execute ( $params );
else
$stmt = $this -> dbh -> query ( $statement );
2006-09-22 22:16:12 +00:00
2006-11-05 19:24:28 +00:00
$this -> listener -> onQuery ( $this , $statement , $params , $this -> querySequence );
$this -> querySequence ++ ;
2006-09-21 21:09:58 +00:00
return $stmt ;
}
/**
* quote
* quotes a string for use in a query
*
* @ param string $input
* @ return string
*/
public function quote ( $input ) {
$this -> connect ();
2006-09-22 22:16:12 +00:00
2006-09-21 21:09:58 +00:00
return $this -> dbh -> quote ( $input );
}
/**
* exec
* executes an SQL statement and returns the number of affected rows
*
* @ param string $statement
* @ return integer
*/
public function exec ( $statement ) {
2006-09-27 10:55:02 +00:00
$this -> connect ();
2006-09-23 10:44:39 +00:00
$args = func_get_args ();
2006-11-05 19:24:28 +00:00
$this -> listener -> onPreExec ( $this , $statement , $args );
2006-09-22 22:16:12 +00:00
$rows = $this -> dbh -> exec ( $statement );
2006-11-05 19:24:28 +00:00
$this -> listener -> onExec ( $this , $statement , $args );
2006-09-22 22:16:12 +00:00
return $rows ;
2006-09-21 21:09:58 +00:00
}
2006-09-25 21:08:02 +00:00
/**
* fetchAll
2006-09-27 10:55:02 +00:00
*
* @ return array
2006-09-25 21:08:02 +00:00
*/
2006-09-27 10:55:02 +00:00
public function fetchAll ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetchAll ( PDO :: FETCH_ASSOC );
}
public function fetchOne ( $statement , array $params = array ()) {
return current ( $this -> query ( $statement , $params ) -> fetch ( PDO :: FETCH_NUM ));
}
public function fetchRow ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetch ( PDO :: FETCH_ASSOC );
}
public function fetchArray ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetch ( PDO :: FETCH_NUM );
}
public function fetchColumn ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetchAll ( PDO :: FETCH_COLUMN );
}
public function fetchAssoc ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetchAll ( PDO :: FETCH_ASSOC );
}
public function fetchBoth ( $statement , array $params = array ()) {
return $this -> query ( $statement , $params ) -> fetchAll ( PDO :: FETCH_BOTH );
2006-09-25 21:08:02 +00:00
}
2006-09-21 21:09:58 +00:00
/**
* lastInsertId
*
2006-09-27 10:55:02 +00:00
* @ return integer
2006-09-21 21:09:58 +00:00
*/
public function lastInsertId () {
$this -> connect ();
return $this -> dbh -> lastInsertId ();
}
/**
* begins a transaction
*
* @ return boolean
*/
public function beginTransaction () {
2006-09-22 22:16:12 +00:00
$this -> listener -> onPreBeginTransaction ( $this );
2006-09-21 21:09:58 +00:00
2006-09-22 22:16:12 +00:00
$return = $this -> dbh -> beginTransaction ();
$this -> listener -> onBeginTransaction ( $this );
return $return ;
2006-09-21 21:09:58 +00:00
}
/**
* commits a transaction
*
* @ return boolean
*/
public function commit () {
2006-09-22 22:16:12 +00:00
$this -> listener -> onPreCommit ( $this );
$return = $this -> dbh -> commit ();
$this -> listener -> onCommit ( $this );
return $return ;
2006-09-21 21:09:58 +00:00
}
/**
* rollBack
*
* @ return boolean
*/
public function rollBack () {
$this -> connect ();
$this -> dbh -> rollBack ();
}
/**
* getAttribute
* retrieves a database connection attribute
*
* @ param integer $attribute
* @ return mixed
*/
public function getAttribute ( $attribute ) {
$this -> connect ();
$this -> dbh -> getAttribute ( $attribute );
}
/**
* returns an array of available PDO drivers
*/
public static function getAvailableDrivers () {
2006-09-22 21:20:21 +00:00
return PDO :: getAvailableDrivers ();
2006-09-21 21:09:58 +00:00
}
/**
* setAttribute
* sets an attribute
*
* @ param integer $attribute
* @ param mixed $value
* @ return boolean
*/
public function setAttribute ( $attribute , $value ) {
$this -> connect ();
$this -> dbh -> setAttribute ( $attribute , $value );
}
/**
* getIterator
*
* @ return ArrayIterator
*/
public function getIterator () {
2006-11-05 19:24:28 +00:00
if ( $this -> listener instanceof Doctrine_DB_Profiler )
return $this -> listener ;
2006-09-21 21:09:58 +00:00
}
/**
* count
* returns the number of executed queries
*
* @ return integer
*/
public function count () {
2006-11-05 19:24:28 +00:00
return $this -> querySequence ;
2006-09-21 21:09:58 +00:00
}
}