1
0
mirror of synced 2025-01-22 08:11:40 +03:00
doctrine2/lib/Doctrine/Record.php

1364 lines
44 KiB
PHP
Raw Normal View History

2006-05-30 08:42:10 +00:00
<?php
/*
2006-07-26 17:09:00 +00:00
* $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>.
*/
2006-09-21 21:09:58 +00:00
Doctrine::autoload('Doctrine_Access');
2006-05-30 08:42:10 +00:00
/**
* Doctrine_Record
2006-07-26 17:09:00 +00:00
* All record classes should inherit this super class
*
* @author Konsta Vesterinen
* @license LGPL
* @package Doctrine
2006-05-30 08:42:10 +00:00
*/
2006-06-05 09:57:53 +00:00
abstract class Doctrine_Record extends Doctrine_Access implements Countable, IteratorAggregate, Serializable {
2006-05-30 08:42:10 +00:00
/**
* STATE CONSTANTS
*/
/**
* DIRTY STATE
* a Doctrine_Record is in dirty state when its properties are changed
*/
const STATE_DIRTY = 1;
/**
* TDIRTY STATE
* a Doctrine_Record is in transient dirty state when it is created and some of its fields are modified
* but it is NOT yet persisted into database
*/
const STATE_TDIRTY = 2;
/**
* CLEAN STATE
* a Doctrine_Record is in clean state when all of its properties are loaded from the database
* and none of its properties are changed
*/
const STATE_CLEAN = 3;
/**
* PROXY STATE
* a Doctrine_Record is in proxy state when its properties are not fully loaded
*/
const STATE_PROXY = 4;
/**
* NEW TCLEAN
* a Doctrine_Record is in transient clean state when it is created and none of its fields are modified
*/
const STATE_TCLEAN = 5;
/**
* DELETED STATE
* a Doctrine_Record turns into deleted state when it is deleted
*/
const STATE_DELETED = 6;
/**
2006-06-03 09:10:43 +00:00
* CALLBACK CONSTANTS
2006-05-30 08:42:10 +00:00
*/
2006-06-03 09:10:43 +00:00
/**
* RAW CALLBACK
*
* when using a raw callback and the property if a record is changed using this callback the
* record state remains untouched
*/
const CALLBACK_RAW = 1;
/**
* STATE-WISE CALLBACK
*
* state-wise callback means that when callback is used and the property is changed the
* record state is also updated
*/
const CALLBACK_STATEWISE = 2;
2006-05-30 08:42:10 +00:00
/**
* @var object Doctrine_Table $table the factory that created this data access object
*/
protected $table;
/**
* @var integer $id the primary keys of this object
2006-05-30 08:42:10 +00:00
*/
2006-07-21 23:22:15 +00:00
protected $id = array();
2006-05-30 08:42:10 +00:00
/**
* @var array $data the record data
*/
2006-07-21 23:22:15 +00:00
protected $data = array();
2006-05-30 08:42:10 +00:00
/**
* @var integer $state the state of this record
* @see STATE_* constants
*/
2006-06-01 11:58:05 +00:00
protected $state;
/**
* @var array $modified an array containing properties that have been modified
*/
2006-07-21 23:22:15 +00:00
protected $modified = array();
2006-05-30 08:42:10 +00:00
/**
* @var array $collections the collections this record is in
*/
2006-07-21 23:22:15 +00:00
private $collections = array();
2006-05-30 08:42:10 +00:00
/**
2006-07-21 23:22:15 +00:00
* @var array $references an array containing all the references
2006-05-30 08:42:10 +00:00
*/
2006-07-21 23:22:15 +00:00
private $references = array();
2006-05-30 08:42:10 +00:00
/**
2006-07-21 23:22:15 +00:00
* @var array $originals an array containing all the original references
2006-05-30 08:42:10 +00:00
*/
2006-07-21 23:22:15 +00:00
private $originals = array();
/**
* @var array $filters
*/
private $filters = array();
2006-05-30 08:42:10 +00:00
/**
* @var integer $index this index is used for creating object identifiers
*/
2006-07-21 23:22:15 +00:00
private static $index = 1;
2006-05-30 08:42:10 +00:00
/**
* @var Doctrine_Null $null a Doctrine_Null object used for extremely fast
* null value testing
2006-05-30 08:42:10 +00:00
*/
private static $null;
/**
* @var integer $oid object identifier
*/
private $oid;
/**
* constructor
* @param Doctrine_Table|null $table a Doctrine_Table object or null,
* if null the table object is retrieved from current connection
*
2006-08-21 23:19:15 +00:00
* @throws Doctrine_Connection_Exception if object is created using the new operator and there are no
* open connections
* @throws Doctrine_Record_Exception if the cleanData operation fails somehow
2006-05-30 08:42:10 +00:00
*/
public function __construct($table = null) {
if(isset($table) && $table instanceof Doctrine_Table) {
$this->table = $table;
$exists = ( ! $this->table->isNewEntry());
} else {
2006-08-21 23:19:15 +00:00
$this->table = Doctrine_Manager::getInstance()->getCurrentConnection()->getTable(get_class($this));
2006-05-30 08:42:10 +00:00
$exists = false;
}
2006-08-21 23:19:15 +00:00
// Check if the current connection has the records table in its registry
2006-05-30 08:42:10 +00:00
// If not this is record is only used for creating table definition and setting up
// relations.
2006-08-21 23:19:15 +00:00
if($this->table->getConnection()->hasTable($this->table->getComponentName())) {
2006-05-30 08:42:10 +00:00
$this->oid = self::$index;
2006-05-30 08:42:10 +00:00
self::$index++;
$keys = $this->table->getPrimaryKeys();
if( ! $exists) {
// listen the onPreCreate event
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onPreCreate($this);
} else {
2006-05-30 08:42:10 +00:00
// listen the onPreLoad event
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onPreLoad($this);
}
// get the data array
$this->data = $this->table->getData();
2006-06-05 10:24:14 +00:00
2006-05-30 08:42:10 +00:00
// get the column count
$count = count($this->data);
// clean data array
2006-06-01 11:58:05 +00:00
$this->cleanData();
2006-05-30 08:42:10 +00:00
$this->prepareIdentifiers($exists);
if( ! $exists) {
2006-06-01 11:58:05 +00:00
if($count > 0)
2006-05-30 08:42:10 +00:00
$this->state = Doctrine_Record::STATE_TDIRTY;
else
$this->state = Doctrine_Record::STATE_TCLEAN;
// set the default values for this record
$this->setDefaultValues();
2006-05-30 08:42:10 +00:00
// listen the onCreate event
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onCreate($this);
} else {
$this->state = Doctrine_Record::STATE_CLEAN;
if($count < $this->table->getColumnCount()) {
$this->state = Doctrine_Record::STATE_PROXY;
}
// listen the onLoad event
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onLoad($this);
}
2006-06-19 21:31:22 +00:00
2006-06-29 23:04:39 +00:00
$repository = $this->table->getRepository();
$repository->add($this);
2006-05-30 08:42:10 +00:00
}
}
/**
* initNullObject
2006-06-01 11:58:05 +00:00
*
* @param Doctrine_Null $null
2006-05-30 08:42:10 +00:00
*/
public static function initNullObject(Doctrine_Null $null) {
self::$null = $null;
}
2006-06-01 11:58:05 +00:00
/**
* @return Doctrine_Null
*/
public static function getNullObject() {
return self::$null;
}
/**
2006-05-30 08:42:10 +00:00
* setUp
2006-09-21 21:09:58 +00:00
* this method is used for setting up relations and attributes
* it should be implemented by child classes
*
* @return void
2006-05-30 08:42:10 +00:00
*/
public function setUp() { }
/**
2006-09-21 21:09:58 +00:00
* getOID
* returns the object identifier
2006-05-30 08:42:10 +00:00
*
* @return integer
*/
public function getOID() {
return $this->oid;
}
/**
* setDefaultValues
* sets the default values for records internal data
*
* @param boolean $overwrite whether or not to overwrite the already set values
* @return boolean
*/
public function setDefaultValues($overwrite = false) {
if( ! $this->table->hasDefaultValues())
return false;
foreach($this->data as $column => $value) {
$default = $this->table->getDefaultValueOf($column);
if($default === null)
$default = self::$null;
if($value === self::$null || $overwrite) {
$this->data[$column] = $default;
$this->modified[] = $column;
$this->state = Doctrine_Record::STATE_TDIRTY;
}
}
}
2006-05-30 08:42:10 +00:00
/**
* cleanData
* this method does several things to records internal data
*
* 1. It unserializes array and object typed columns
* 2. Uncompresses gzip typed columns
* 3. Gets the appropriate enum values for enum typed columns
* 4. Initializes special null object pointer for null values (for fast column existence checking purposes)
*
*
2006-05-30 08:42:10 +00:00
* example:
*
* $data = array("name"=>"John","lastname"=> null, "id" => 1,"unknown" => "unknown");
* $names = array("name", "lastname", "id");
2006-05-30 08:42:10 +00:00
* $data after operation:
* $data = array("name"=>"John","lastname" => Object(Doctrine_Null));
*
* here column 'id' is removed since its auto-incremented primary key (read-only)
*
* @throws Doctrine_Record_Exception if unserialization of array/object typed column fails or
* if uncompression of gzip typed column fails
2006-06-17 22:46:03 +00:00
*
* @return integer
2006-05-30 08:42:10 +00:00
*/
2006-09-20 15:46:25 +00:00
private function cleanData($debug = false) {
2006-09-03 19:20:02 +00:00
$tmp = $this->data;
2006-06-01 11:58:05 +00:00
2006-05-30 08:42:10 +00:00
$this->data = array();
$count = 0;
2006-05-30 08:42:10 +00:00
foreach($this->table->getColumnNames() as $name) {
$type = $this->table->getTypeOf($name);
if( ! isset($tmp[$name])) {
$this->data[$name] = self::$null;
2006-05-30 08:42:10 +00:00
} else {
switch($type):
2006-06-03 09:10:43 +00:00
case "array":
case "object":
2006-07-30 21:37:07 +00:00
if($tmp[$name] !== self::$null) {
if(is_string($tmp[$name])) {
$value = unserialize($tmp[$name]);
2006-07-30 21:37:54 +00:00
2006-08-25 17:17:55 +00:00
if($value === false)
throw new Doctrine_Record_Exception("Unserialization of $name failed. ".var_dump(substr($tmp[$lower],0,30)."...",true));
2006-08-25 17:17:55 +00:00
} else
$value = $tmp[$name];
2006-08-25 20:13:37 +00:00
2006-07-30 21:37:07 +00:00
$this->data[$name] = $value;
}
2006-06-17 22:46:03 +00:00
break;
2006-09-20 15:46:25 +00:00
case "gzip":
if($tmp[$name] !== self::$null) {
$value = gzuncompress($tmp[$name]);
2006-09-20 15:46:25 +00:00
if($value === false)
throw new Doctrine_Record_Exception("Uncompressing of $name failed.");
$this->data[$name] = $value;
}
break;
case "enum":
$this->data[$name] = $this->table->enumValue($name, $tmp[$name]);
2006-06-03 09:10:43 +00:00
break;
default:
$this->data[$name] = $tmp[$name];
2006-06-03 09:10:43 +00:00
endswitch;
2006-06-17 22:46:03 +00:00
$count++;
2006-05-30 08:42:10 +00:00
}
}
2006-07-30 21:37:07 +00:00
2006-09-20 15:46:25 +00:00
return $count;
2006-05-30 08:42:10 +00:00
}
/**
* prepareIdentifiers
* prepares identifiers for later use
2006-05-30 08:42:10 +00:00
*
* @param boolean $exists whether or not this record exists in persistent data store
2006-05-30 08:42:10 +00:00
* @return void
*/
private function prepareIdentifiers($exists = true) {
switch($this->table->getIdentifierType()):
case Doctrine_Identifier::AUTO_INCREMENT:
case Doctrine_Identifier::SEQUENCE:
2006-06-03 09:10:43 +00:00
$name = $this->table->getIdentifier();
if($exists) {
2006-06-05 09:57:53 +00:00
if(isset($this->data[$name]) && $this->data[$name] !== self::$null)
$this->id[$name] = $this->data[$name];
2006-05-30 08:42:10 +00:00
}
2006-06-05 09:57:53 +00:00
2006-06-03 09:10:43 +00:00
unset($this->data[$name]);
2006-05-30 08:42:10 +00:00
break;
2006-07-27 18:31:18 +00:00
case Doctrine_Identifier::NORMAL:
$this->id = array();
$name = $this->table->getIdentifier();
2006-07-27 18:31:18 +00:00
if(isset($this->data[$name]) && $this->data[$name] !== self::$null)
$this->id[$name] = $this->data[$name];
break;
2006-05-30 08:42:10 +00:00
case Doctrine_Identifier::COMPOSITE:
$names = $this->table->getIdentifier();
2006-07-27 18:31:18 +00:00
2006-05-30 08:42:10 +00:00
foreach($names as $name) {
if($this->data[$name] === self::$null)
$this->id[$name] = null;
else
2006-05-30 08:42:10 +00:00
$this->id[$name] = $this->data[$name];
}
break;
endswitch;
}
/**
* serialize
2006-05-30 08:42:10 +00:00
* this method is automatically called when this Doctrine_Record is serialized
*
* @return array
*/
2006-06-05 09:57:53 +00:00
public function serialize() {
2006-05-30 08:42:10 +00:00
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onSleep($this);
2006-06-05 09:57:53 +00:00
$vars = get_object_vars($this);
2006-06-05 09:57:53 +00:00
unset($vars['references']);
unset($vars['collections']);
unset($vars['originals']);
unset($vars['table']);
$name = $this->table->getIdentifier();
$this->data = array_merge($this->data, $this->id);
2006-05-30 08:42:10 +00:00
2006-06-03 09:10:43 +00:00
foreach($this->data as $k => $v) {
2006-05-30 08:42:10 +00:00
if($v instanceof Doctrine_Record)
2006-06-05 19:44:31 +00:00
unset($vars['data'][$k]);
2006-06-03 09:10:43 +00:00
elseif($v === self::$null) {
2006-06-05 19:44:31 +00:00
unset($vars['data'][$k]);
2006-06-03 09:10:43 +00:00
} else {
switch($this->table->getTypeOf($k)):
case "array":
case "object":
2006-06-05 19:44:31 +00:00
$vars['data'][$k] = serialize($vars['data'][$k]);
2006-06-03 09:10:43 +00:00
break;
endswitch;
}
2006-05-30 08:42:10 +00:00
}
2006-06-05 09:57:53 +00:00
return serialize($vars);
2006-05-30 08:42:10 +00:00
}
/**
* unseralize
* this method is automatically called everytime a Doctrine_Record object is unserialized
*
* @param string $serialized Doctrine_Record as serialized string
* @throws Doctrine_Record_Exception if the cleanData operation fails somehow
2006-05-30 08:42:10 +00:00
* @return void
*/
2006-06-05 09:57:53 +00:00
public function unserialize($serialized) {
2006-05-30 08:42:10 +00:00
$manager = Doctrine_Manager::getInstance();
2006-08-21 23:19:15 +00:00
$connection = $manager->getCurrentConnection();
2006-05-30 08:42:10 +00:00
$this->oid = self::$index;
self::$index++;
2006-08-21 23:19:15 +00:00
$this->table = $connection->getTable(get_class($this));
2006-06-05 09:57:53 +00:00
$array = unserialize($serialized);
foreach($array as $name => $values) {
$this->$name = $values;
}
2006-05-30 08:42:10 +00:00
$this->table->getRepository()->add($this);
$this->cleanData();
$this->prepareIdentifiers($this->exists());
2006-06-03 09:10:43 +00:00
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onWakeUp($this);
2006-05-30 08:42:10 +00:00
}
2006-06-05 09:57:53 +00:00
2006-05-30 08:42:10 +00:00
/**
* addCollection
*
2006-05-30 08:42:10 +00:00
* @param Doctrine_Collection $collection
* @param mixed $key
*/
final public function addCollection(Doctrine_Collection $collection,$key = null) {
if($key !== null) {
$this->collections[$key] = $collection;
} else {
$this->collections[] = $collection;
}
}
/**
* getCollection
* @param integer $key
* @return Doctrine_Collection
*/
final public function getCollection($key) {
return $this->collections[$key];
}
/**
* hasCollections
* whether or not this record is part of a collection
*
* @return boolean
*/
final public function hasCollections() {
return (! empty($this->collections));
}
/**
* getState
* returns the current state of the object
*
* @see Doctrine_Record::STATE_* constants
* @return integer
*/
final public function getState() {
return $this->state;
}
/**
* refresh
2006-05-30 08:42:10 +00:00
* refresh internal data from the database
*
* @throws Doctrine_Record_Exception When the refresh operation fails (when the database row
* this record represents does not exist anymore)
2006-05-30 08:42:10 +00:00
* @return boolean
*/
final public function refresh() {
$id = $this->obtainIdentifier();
2006-05-30 08:42:10 +00:00
if( ! is_array($id))
$id = array($id);
if(empty($id))
return false;
$id = array_values($id);
2006-06-25 18:34:53 +00:00
$query = $this->table->getQuery()." WHERE ".implode(" = ? AND ",$this->table->getPrimaryKeys())." = ?";
$stmt = $this->table->getConnection()->execute($query,$id);
$this->data = $stmt->fetch(PDO::FETCH_ASSOC);
2006-05-30 08:42:10 +00:00
2006-09-03 19:20:02 +00:00
if( ! $this->data)
throw new Doctrine_Record_Exception('Failed to refresh. Record does not exist anymore');
2006-09-03 19:20:02 +00:00
$this->data = array_change_key_case($this->data, CASE_LOWER);
2006-05-30 08:42:10 +00:00
$this->modified = array();
2006-09-20 15:46:25 +00:00
$this->cleanData(true);
2006-05-30 08:42:10 +00:00
$this->prepareIdentifiers();
$this->state = Doctrine_Record::STATE_CLEAN;
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onLoad($this);
2006-05-30 08:42:10 +00:00
return true;
}
/**
* factoryRefresh
2006-06-17 22:46:03 +00:00
* refreshes the data from outer source (Doctrine_Table)
*
* @throws Doctrine_Record_Exception When the primary key of this record doesn't match the primary key fetched from a collection
2006-05-30 08:42:10 +00:00
* @return void
*/
final public function factoryRefresh() {
2006-07-30 21:37:07 +00:00
$this->data = $this->table->getData();
$old = $this->id;
$this->cleanData();
2006-05-30 08:42:10 +00:00
$this->prepareIdentifiers();
if($this->id != $old)
throw new Doctrine_Record_Exception("The refreshed primary key doesn't match the one in the record memory.", Doctrine::ERR_REFRESH);
2006-05-30 08:42:10 +00:00
$this->state = Doctrine_Record::STATE_CLEAN;
$this->modified = array();
$this->table->getAttribute(Doctrine::ATTR_LISTENER)->onLoad($this);
2006-05-30 08:42:10 +00:00
}
/**
* getTable
* returns the table object for this record
*
2006-05-30 08:42:10 +00:00
* @return object Doctrine_Table a Doctrine_Table object
*/
final public function getTable() {
return $this->table;
}
/**
* getData
2006-05-30 08:42:10 +00:00
* return all the internal data
*
* @return array an array containing all the properties
2006-05-30 08:42:10 +00:00
*/
final public function getData() {
return $this->data;
}
2006-06-01 11:58:05 +00:00
/**
* rawGet
* returns the value of a property, if the property is not yet loaded
* this method does NOT load it
*
* @param $name name of the property
* @return mixed
*/
2006-09-27 21:34:32 +00:00
2006-06-01 11:58:05 +00:00
public function rawGet($name) {
if( ! isset($this->data[$name]))
throw new InvalidKeyException();
if($this->data[$name] === self::$null)
2006-06-01 11:58:05 +00:00
return null;
return $this->data[$name];
}
2006-09-27 21:34:32 +00:00
/**
* load
* loads all the unitialized properties from the database
*
* @return boolean
*/
public function load() {
// only load the data from database if the Doctrine_Record is in proxy state
if($this->state == Doctrine_Record::STATE_PROXY) {
if( ! empty($this->collections)) {
// delegate the loading operation to collections in which this record resides
foreach($this->collections as $collection) {
$collection->load($this);
2006-09-12 21:36:36 +00:00
}
} else {
2006-09-12 21:36:36 +00:00
$this->refresh();
}
$this->state = Doctrine_Record::STATE_CLEAN;
return true;
}
return false;
}
2006-05-30 08:42:10 +00:00
/**
* get
* returns a value of a property or a related component
2006-05-30 08:42:10 +00:00
*
* @param mixed $name name of the property or related component
* @param boolean $invoke whether or not to invoke the onGetProperty listener
* @throws Doctrine_Record_Exception if trying to get a value of unknown property / related component
2006-05-30 08:42:10 +00:00
* @return mixed
*/
public function get($name, $invoke = true) {
$listener = $this->table->getAttribute(Doctrine::ATTR_LISTENER);
$value = self::$null;
$lower = strtolower($name);
if(isset($this->data[$lower])) {
2006-05-30 08:42:10 +00:00
// check if the property is null (= it is the Doctrine_Null object located in self::$null)
if($this->data[$lower] === self::$null) {
$this->load();
2006-09-12 21:36:36 +00:00
}
if($this->data[$lower] === self::$null)
2006-09-12 21:36:36 +00:00
$value = null;
else
$value = $this->data[$lower];
2006-05-30 08:42:10 +00:00
2006-09-12 21:36:36 +00:00
}
2006-06-05 09:57:53 +00:00
if($value !== self::$null) {
2006-09-12 21:36:36 +00:00
if($invoke && $name !== $this->table->getIdentifier()) {
return $this->table->getAttribute(Doctrine::ATTR_LISTENER)->onGetProperty($this, $name, $value);
} else
return $value;
}
2006-05-30 08:42:10 +00:00
2006-09-12 21:36:36 +00:00
if(isset($this->id[$lower]))
return $this->id[$lower];
2006-09-12 21:36:36 +00:00
if($name === $this->table->getIdentifier())
return null;
$rel = $this->table->getRelation($name);
2006-05-30 08:42:10 +00:00
try {
if( ! isset($this->references[$name]))
$this->loadReference($name);
} catch(Doctrine_Table_Exception $e) {
throw new Doctrine_Record_Exception("Unknown property / related component '$name'.");
}
2006-05-30 08:42:10 +00:00
return $this->references[$name];
}
2006-05-30 08:42:10 +00:00
/**
* set
* method for altering properties and Doctrine_Record references
2006-09-27 21:21:33 +00:00
* if the load parameter is set to false this method will not try to load uninitialized record data
2006-05-30 08:42:10 +00:00
*
* @param mixed $name name of the property or reference
* @param mixed $value value of the property or reference
2006-09-27 21:21:33 +00:00
* @param boolean $load whether or not to refresh / load the uninitialized record data
*
* @throws Doctrine_Record_Exception if trying to set a value for unknown property / related component
2006-09-27 21:21:33 +00:00
* @throws Doctrine_Record_Exception if trying to set a value of wrong type for related component
*
* @return Doctrine_Record
2006-05-30 08:42:10 +00:00
*/
2006-09-27 21:21:33 +00:00
public function set($name, $value, $load = true) {
$lower = strtolower($name);
if(isset($this->data[$lower])) {
2006-05-30 08:42:10 +00:00
if($value instanceof Doctrine_Record) {
$id = $value->getIncremented();
if($id !== null)
$value = $id;
2006-05-30 08:42:10 +00:00
}
2006-09-27 21:21:33 +00:00
if($load)
$old = $this->get($lower, false);
else
$old = $this->data[$lower];
2006-05-30 08:42:10 +00:00
if($old !== $value) {
// invoke the onPreSetProperty listener
$value = $this->table->getAttribute(Doctrine::ATTR_LISTENER)->onSetProperty($this, $name, $value);
2006-08-25 20:13:37 +00:00
if($value === null)
$value = self::$null;
$this->data[$lower] = $value;
$this->modified[] = $lower;
2006-05-30 08:42:10 +00:00
switch($this->state):
case Doctrine_Record::STATE_CLEAN:
$this->state = Doctrine_Record::STATE_DIRTY;
break;
case Doctrine_Record::STATE_TCLEAN:
$this->state = Doctrine_Record::STATE_TDIRTY;
break;
endswitch;
}
} else {
try {
2006-09-28 20:03:29 +00:00
$this->coreSetRelated($name, $value);
} catch(Doctrine_Table_Exception $e) {
throw new Doctrine_Record_Exception("Unknown property / related component '$name'.");
}
2006-09-28 20:03:29 +00:00
}
}
public function coreSetRelated($name, $value) {
$rel = $this->table->getRelation($name);
// one-to-many or one-to-one relation
if($rel instanceof Doctrine_Relation_ForeignKey ||
$rel instanceof Doctrine_Relation_LocalKey) {
if( ! $rel->isOneToOne()) {
// one-to-many relation found
if( ! ($value instanceof Doctrine_Collection))
throw new Doctrine_Record_Exception("Couldn't call Doctrine::set(), second argument should be an instance of Doctrine_Collection when setting one-to-many references.");
$value->setReference($this,$rel);
} else {
// one-to-one relation found
if( ! ($value instanceof Doctrine_Record))
throw new Doctrine_Record_Exception("Couldn't call Doctrine::set(), second argument should be an instance of Doctrine_Record when setting one-to-one references.");
if($rel instanceof Doctrine_Relation_LocalKey) {
$this->set($rel->getLocal(), $value, false);
} else {
$value->set($rel->getForeign(), $this, false);
}
2006-05-30 08:42:10 +00:00
}
} elseif($rel instanceof Doctrine_Relation_Association) {
2006-09-28 20:03:29 +00:00
// join table relation found
if( ! ($value instanceof Doctrine_Collection))
throw new Doctrine_Record_Exception("Couldn't call Doctrine::set(), second argument should be an instance of Doctrine_Collection when setting many-to-many references.");
2006-09-28 20:03:29 +00:00
2006-05-30 08:42:10 +00:00
}
2006-09-28 20:03:29 +00:00
$this->references[$name] = $value;
2006-05-30 08:42:10 +00:00
}
/**
* contains
2006-05-30 08:42:10 +00:00
*
* @param string $name
* @return boolean
*/
public function contains($name) {
$lower = strtolower($name);
if(isset($this->data[$lower]))
2006-05-30 08:42:10 +00:00
return true;
if(isset($this->id[$lower]))
2006-09-28 20:03:29 +00:00
return true;
2006-05-30 08:42:10 +00:00
if(isset($this->references[$name]))
return true;
return false;
}
/**
* @param string $name
* @return void
*/
public function __unset($name) {
if(isset($this->data[$name]))
$this->data[$name] = array();
// todo: what to do with references ?
}
/**
* applies the changes made to this object into database
* this method is smart enough to know if any changes are made
* and whether to use INSERT or UPDATE statement
*
* this method also saves the related composites
*
* @return void
*/
final public function save(Doctrine_Connection $conn = null) {
if ($conn == null) {
$conn = $this->table->getConnection();
}
$conn->beginTransaction();
2006-05-30 08:42:10 +00:00
$saveLater = $conn->saveRelated($this);
2006-05-30 08:42:10 +00:00
$conn->save($this);
2006-05-30 08:42:10 +00:00
foreach($saveLater as $fk) {
$table = $fk->getTable();
$alias = $this->table->getAlias($table->getComponentName());
if(isset($this->references[$alias])) {
$obj = $this->references[$alias];
$obj->save();
}
}
// save the MANY-TO-MANY associations
$this->saveAssociations();
$conn->commit();
2006-05-30 08:42:10 +00:00
}
/**
* returns an array of modified fields and associated values
* @return array
*/
final public function getModified() {
$a = array();
2006-06-19 21:31:22 +00:00
foreach($this->modified as $k => $v) {
2006-05-30 08:42:10 +00:00
$a[$v] = $this->data[$v];
}
return $a;
}
/**
* returns an array of modified fields and values with data preparation
* adds column aggregation inheritance and converts Records into primary key values
*
* @return array
*/
2006-06-19 21:31:22 +00:00
final public function getPrepared(array $array = array()) {
2006-05-30 08:42:10 +00:00
$a = array();
2006-06-19 21:31:22 +00:00
if(empty($array))
$array = $this->modified;
foreach($array as $k => $v) {
2006-06-03 09:10:43 +00:00
$type = $this->table->getTypeOf($v);
2006-09-04 06:23:38 +00:00
switch($type) {
case 'array':
case 'object':
$a[$v] = serialize($this->data[$v]);
2006-09-20 15:46:25 +00:00
break;
case 'gzip':
$a[$v] = gzcompress($this->data[$v],5);
break;
case 'boolean':
$a[$v] = (int) $this->data[$v];
break;
2006-09-04 06:27:27 +00:00
case 'enum':
2006-09-04 06:23:38 +00:00
$a[$v] = $this->table->enumIndex($v,$this->data[$v]);
break;
default:
if($this->data[$v] instanceof Doctrine_Record)
$this->data[$v] = $this->data[$v]->getIncremented();
2006-06-03 09:10:43 +00:00
2006-09-04 06:23:38 +00:00
if($this->data[$v] === self::$null)
$a[$v] = null;
else
$a[$v] = $this->data[$v];
}
2006-05-30 08:42:10 +00:00
}
2006-06-08 22:11:36 +00:00
foreach($this->table->getInheritanceMap() as $k => $v) {
$old = $this->get($k);
if((string) $old !== (string) $v || $old === null) {
$a[$k] = $v;
$this->data[$k] = $v;
}
}
2006-05-30 08:42:10 +00:00
return $a;
}
/**
2006-09-27 21:21:33 +00:00
* count
2006-05-30 08:42:10 +00:00
* this class implements countable interface
2006-09-27 21:21:33 +00:00
*
2006-05-30 08:42:10 +00:00
* @return integer the number of columns
*/
public function count() {
return count($this->data);
}
/**
* alias for count()
2006-09-27 21:21:33 +00:00
*
* @return integer
*/
public function getColumnCount() {
return $this->count();
}
2006-09-07 21:28:47 +00:00
/**
* toArray
2006-09-27 21:21:33 +00:00
* returns the record as an array
2006-09-07 21:28:47 +00:00
*
* @return array
*/
public function toArray() {
$a = array();
foreach($this as $column => $value) {
$a[$column] = $value;
}
if($this->table->getIdentifierType() == Doctrine_Identifier::AUTO_INCREMENT) {
$i = $this->table->getIdentifier();
$a[$i] = $this->getIncremented();
}
return $a;
}
/**
2006-09-27 21:21:33 +00:00
* exists
* returns true if this record is persistent, otherwise false
*
* @return boolean
*/
public function exists() {
return ($this->state !== Doctrine_Record::STATE_TCLEAN &&
$this->state !== Doctrine_Record::STATE_TDIRTY);
}
/**
* method for checking existence of properties and Doctrine_Record references
* @param mixed $name name of the property or reference
* @return boolean
*/
public function hasRelation($name) {
if(isset($this->data[$name]) || isset($this->id[$name]))
return true;
return $this->table->hasRelation($name);
}
2006-05-30 08:42:10 +00:00
/**
* getIterator
2006-06-01 11:58:05 +00:00
* @return Doctrine_Record_Iterator a Doctrine_Record_Iterator that iterates through the data
2006-05-30 08:42:10 +00:00
*/
public function getIterator() {
2006-06-01 11:58:05 +00:00
return new Doctrine_Record_Iterator($this);
2006-05-30 08:42:10 +00:00
}
/**
* saveAssociations
*
2006-05-30 08:42:10 +00:00
* save the associations of many-to-many relations
* this method also deletes associations that do not exist anymore
*
2006-05-30 08:42:10 +00:00
* @return void
*/
final public function saveAssociations() {
foreach($this->table->getRelations() as $fk):
2006-05-30 08:42:10 +00:00
$table = $fk->getTable();
$name = $table->getComponentName();
$alias = $this->table->getAlias($name);
if($fk instanceof Doctrine_Relation_Association) {
2006-05-30 08:42:10 +00:00
switch($fk->getType()):
case Doctrine_Relation::MANY_AGGREGATE:
$asf = $fk->getAssociationFactory();
if(isset($this->references[$alias])) {
$new = $this->references[$alias];
if( ! isset($this->originals[$alias])) {
$this->loadReference($alias);
}
$r = Doctrine_Relation::getDeleteOperations($this->originals[$alias],$new);
2006-05-30 08:42:10 +00:00
foreach($r as $record) {
2006-05-30 08:42:10 +00:00
$query = "DELETE FROM ".$asf->getTableName()." WHERE ".$fk->getForeign()." = ?"
2006-06-25 18:34:53 +00:00
." AND ".$fk->getLocal()." = ?";
2006-08-21 23:19:15 +00:00
$this->table->getConnection()->execute($query, array($record->getIncremented(),$this->getIncremented()));
2006-05-30 08:42:10 +00:00
}
$r = Doctrine_Relation::getInsertOperations($this->originals[$alias],$new);
foreach($r as $record) {
2006-05-30 08:42:10 +00:00
$reldao = $asf->create();
$reldao->set($fk->getForeign(),$record);
$reldao->set($fk->getLocal(),$this);
$reldao->save();
}
2006-05-30 08:42:10 +00:00
$this->originals[$alias] = clone $this->references[$alias];
}
break;
endswitch;
} elseif($fk instanceof Doctrine_Relation_ForeignKey ||
$fk instanceof Doctrine_Relation_LocalKey) {
2006-05-30 08:42:10 +00:00
switch($fk->getType()):
case Doctrine_Relation::ONE_COMPOSITE:
if(isset($this->originals[$alias]) && $this->originals[$alias]->obtainIdentifier() != $this->references[$alias]->obtainIdentifier())
2006-05-30 08:42:10 +00:00
$this->originals[$alias]->delete();
2006-05-30 08:42:10 +00:00
break;
case Doctrine_Relation::MANY_COMPOSITE:
if(isset($this->references[$alias])) {
$new = $this->references[$alias];
if( ! isset($this->originals[$alias]))
$this->loadReference($alias);
$r = Doctrine_Relation::getDeleteOperations($this->originals[$alias], $new);
2006-05-30 08:42:10 +00:00
foreach($r as $record) {
2006-05-30 08:42:10 +00:00
$record->delete();
}
2006-05-30 08:42:10 +00:00
$this->originals[$alias] = clone $this->references[$alias];
}
break;
endswitch;
}
endforeach;
}
/**
* getOriginals
* returns an original collection of related component
*
* @return Doctrine_Collection
2006-05-30 08:42:10 +00:00
*/
final public function getOriginals($name) {
if( ! isset($this->originals[$name]))
throw new InvalidKeyException();
return $this->originals[$name];
}
/**
* deletes this data access object and all the related composites
* this operation is isolated by a transaction
*
* this event can be listened by the onPreDelete and onDelete listeners
*
* @return boolean true on success, false on failure
*/
public function delete(Doctrine_Connection $conn = null) {
if ($conn == null) {
$conn = $this->table->getConnection();
}
return $conn->delete($this);
2006-05-30 08:42:10 +00:00
}
/**
* copy
2006-05-30 08:42:10 +00:00
* returns a copy of this object
*
* @return Doctrine_Record
2006-05-30 08:42:10 +00:00
*/
public function copy() {
return $this->table->create($this->data);
}
/**
* assignIdentifier
*
2006-05-30 08:42:10 +00:00
* @param integer $id
* @return void
*/
final public function assignIdentifier($id = false) {
2006-05-30 08:42:10 +00:00
if($id === false) {
$this->id = array();
2006-05-30 08:42:10 +00:00
$this->cleanData();
$this->state = Doctrine_Record::STATE_TCLEAN;
$this->modified = array();
} elseif($id === true) {
$this->prepareIdentifiers(false);
$this->state = Doctrine_Record::STATE_CLEAN;
$this->modified = array();
} else {
2006-06-05 09:57:53 +00:00
$name = $this->table->getIdentifier();
$this->id[$name] = $id;
2006-06-05 09:57:53 +00:00
$this->state = Doctrine_Record::STATE_CLEAN;
$this->modified = array();
2006-05-30 08:42:10 +00:00
}
}
/**
* returns the primary keys of this object
*
* @return array
2006-05-30 08:42:10 +00:00
*/
final public function obtainIdentifier() {
2006-05-30 08:42:10 +00:00
return $this->id;
}
/**
* returns the value of autoincremented primary key of this object (if any)
*
* @return integer
*/
final public function getIncremented() {
$id = current($this->id);
if($id === false)
return null;
return $id;
}
2006-05-30 08:42:10 +00:00
/**
* getLast
* this method is used internally be Doctrine_Query
* it is needed to provide compatibility between
* records and collections
*
* @return Doctrine_Record
*/
public function getLast() {
return $this;
}
/**
* hasRefence
* @param string $name
* @return boolean
*/
public function hasReference($name) {
return isset($this->references[$name]);
}
/**
* obtainReference
*
* @param string $name
* @throws Doctrine_Record_Exception if trying to get an unknown related component
*/
public function obtainReference($name) {
if(isset($this->references[$name]))
return $this->references[$name];
throw new Doctrine_Record_Exception("Unknown reference $name");
}
2006-05-30 08:42:10 +00:00
/**
* initalizes a one-to-many / many-to-many relation
*
* @param Doctrine_Collection $coll
* @param Doctrine_Relation $connector
* @return boolean
2006-05-30 08:42:10 +00:00
*/
public function initReference(Doctrine_Collection $coll, Doctrine_Relation $connector) {
2006-06-26 18:55:42 +00:00
$alias = $connector->getAlias();
if(isset($this->references[$alias]))
return false;
if( ! $connector->isOneToOne()) {
if( ! ($connector instanceof Doctrine_Relation_Association))
$coll->setReference($this, $connector);
$this->references[$alias] = $coll;
$this->originals[$alias] = clone $coll;
return true;
}
return false;
2006-05-30 08:42:10 +00:00
}
public function lazyInitRelated(Doctrine_Collection $coll, Doctrine_Relation $connector) {
}
2006-05-30 08:42:10 +00:00
/**
* addReference
* @param Doctrine_Record $record
* @param mixed $key
* @return void
*/
2006-06-26 18:55:42 +00:00
public function addReference(Doctrine_Record $record, Doctrine_Relation $connector, $key = null) {
$alias = $connector->getAlias();
2006-05-30 08:42:10 +00:00
$this->references[$alias]->add($record, $key);
$this->originals[$alias]->add($record, $key);
2006-05-30 08:42:10 +00:00
}
/**
* getReferences
* @return array all references
*/
public function getReferences() {
return $this->references;
}
/**
* setRelated
*
* @param string $alias
* @param Doctrine_Access $coll
*/
final public function setRelated($alias, Doctrine_Access $coll) {
$this->references[$alias] = $coll;
$this->originals[$alias] = $coll;
}
/**
* loadReference
* loads a related component
*
* @throws Doctrine_Table_Exception if trying to load an unknown related component
* @param string $name
2006-05-30 08:42:10 +00:00
* @return void
*/
final public function loadReference($name) {
$fk = $this->table->getRelation($name);
2006-05-30 08:42:10 +00:00
2006-09-28 20:03:29 +00:00
if($fk->isOneToOne()) {
$this->references[$name] = $fk->fetchRelatedFor($this);
} else {
$coll = $fk->fetchRelatedFor($this);
2006-09-28 20:03:29 +00:00
$this->references[$name] = $coll;
$this->originals[$name] = clone $coll;
}
2006-05-30 08:42:10 +00:00
}
2006-07-21 23:22:15 +00:00
/**
* filterRelated
* lazy initializes a new filter instance for given related component
*
* @param $componentAlias alias of the related component
2006-07-21 23:22:15 +00:00
* @return Doctrine_Filter
*/
final public function filterRelated($componentAlias) {
if( ! isset($this->filters[$componentAlias])) {
$this->filters[$componentAlias] = new Doctrine_Filter($componentAlias);
2006-07-21 23:22:15 +00:00
}
return $this->filters[$componentAlias];
2006-07-21 23:22:15 +00:00
}
2006-05-30 08:42:10 +00:00
/**
* binds One-to-One composite relation
*
* @param string $objTableName
* @param string $fkField
* @return void
*/
final public function ownsOne($componentName,$foreignKey, $localKey = null) {
$this->table->bind($componentName,$foreignKey,Doctrine_Relation::ONE_COMPOSITE, $localKey);
}
/**
* binds One-to-Many composite relation
*
* @param string $objTableName
* @param string $fkField
* @return void
*/
final public function ownsMany($componentName,$foreignKey, $localKey = null) {
$this->table->bind($componentName,$foreignKey,Doctrine_Relation::MANY_COMPOSITE, $localKey);
}
/**
* binds One-to-One aggregate relation
*
* @param string $objTableName
* @param string $fkField
* @return void
*/
final public function hasOne($componentName,$foreignKey, $localKey = null) {
$this->table->bind($componentName,$foreignKey,Doctrine_Relation::ONE_AGGREGATE, $localKey);
}
/**
* binds One-to-Many aggregate relation
*
* @param string $objTableName
* @param string $fkField
* @return void
*/
final public function hasMany($componentName,$foreignKey, $localKey = null) {
$this->table->bind($componentName,$foreignKey,Doctrine_Relation::MANY_AGGREGATE, $localKey);
}
/**
* setPrimaryKey
* @param mixed $key
*/
final public function setPrimaryKey($key) {
$this->table->setPrimaryKey($key);
}
/**
* hasColumn
* sets a column definition
*
* @param string $name
* @param string $type
* @param integer $length
* @param mixed $options
* @return void
*/
2006-09-24 09:52:04 +00:00
final public function hasColumn($name, $type, $length = 2147483647, $options = "") {
2006-05-30 08:42:10 +00:00
$this->table->setColumn($name, $type, $length, $options);
}
2006-08-25 17:13:21 +00:00
/**
* countRelated
2006-08-25 23:50:55 +00:00
*
2006-08-29 20:37:55 +00:00
* @param string $name the name of the related component
2006-08-25 23:50:55 +00:00
* @return integer
2006-08-25 17:13:21 +00:00
*/
public function countRelated($name) {
$rel = $this->table->getRelation($name);
2006-08-25 23:50:55 +00:00
$componentName = $rel->getTable()->getComponentName();
$alias = $rel->getTable()->getAlias(get_class($this));
$query = new Doctrine_Query();
$query->from($componentName. '(' . 'COUNT(1)' . ')')->where($componentName. '.' .$alias. '.' . $this->getTable()->getIdentifier(). ' = ?');
$array = $query->execute(array($this->getIncremented()));
return $array[0]['COUNT(1)'];
2006-08-25 17:13:21 +00:00
}
2006-08-06 20:55:54 +00:00
/**
* merge
* merges this record with an array of values
2006-08-06 20:55:54 +00:00
*
* @param array $values
2006-09-27 21:21:33 +00:00
* @return void
2006-08-06 20:55:54 +00:00
*/
public function merge(array $values) {
foreach($this->table->getColumnNames() as $value) {
try {
if(isset($values[$value]))
$this->set($value, $values[$value]);
2006-09-27 21:21:33 +00:00
} catch(Exception $e) {
// silence all exceptions
}
2006-08-06 20:55:54 +00:00
}
}
2006-05-30 22:47:01 +00:00
/**
* __call
* @param string $m
* @param array $a
*/
public function __call($m,$a) {
2006-08-15 21:33:41 +00:00
if(method_exists($this->table, $m))
return call_user_func_array(array($this->table, $m), $a);
2006-05-30 22:47:01 +00:00
if( ! function_exists($m))
2006-05-31 08:46:44 +00:00
throw new Doctrine_Record_Exception("unknown callback '$m'");
2006-06-01 11:58:05 +00:00
2006-05-30 22:47:01 +00:00
if(isset($a[0])) {
$column = $a[0];
2006-05-31 08:46:44 +00:00
$a[0] = $this->get($column);
2006-06-03 09:10:43 +00:00
$newvalue = call_user_func_array($m, $a);
2006-08-15 21:33:41 +00:00
$this->data[$column] = $newvalue;
2006-05-30 22:47:01 +00:00
}
return $this;
}
2006-05-30 08:42:10 +00:00
/**
* returns a string representation of this object
*/
public function __toString() {
return Doctrine_Lib::getRecordAsString($this);
}
}