2008-05-30 11:40:08 +00:00
< ? php
/**
* Object - relational mapper .
*
* DBO - backed object data model , for mapping database tables to Cake objects .
*
* PHP versions 5
*
2009-05-01 14:05:46 -07:00
* CakePHP ( tm ) : Rapid Development Framework ( http :// cakephp . org )
2010-01-26 14:18:20 -05:00
* Copyright 2005 - 2010 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
2008-05-30 11:40:08 +00:00
*
* Licensed under The MIT License
* Redistributions of files must retain the above copyright notice .
*
2010-01-26 14:18:20 -05:00
* @ copyright Copyright 2005 - 2010 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
* @ link http :// cakephp . org CakePHP ( tm ) Project
2010-12-24 13:57:20 -05:00
* @ package cake . libs . model
2008-10-30 17:30:26 +00:00
* @ since CakePHP ( tm ) v 0.10 . 0.0
2009-05-01 14:05:46 -07:00
* @ license MIT License ( http :// www . opensource . org / licenses / mit - license . php )
2008-05-30 11:40:08 +00:00
*/
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Included libs
*/
2010-11-21 22:11:59 -05:00
App :: import ( 'Core' , 'ClassRegistry' , false );
App :: import ( 'Core' , 'Validation' , false );
App :: import ( 'Core' , 'String' , false );
2010-07-04 00:43:39 -04:00
App :: import ( 'Model' , 'BehaviorCollection' , false );
2009-07-28 08:36:55 +02:00
App :: import ( 'Model' , 'ModelBehavior' , false );
App :: import ( 'Model' , 'ConnectionManager' , false );
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Object - relational mapper .
*
* DBO - backed object data model .
* Automatically selects a database table name based on a pluralized lowercase object class name
* ( i . e . class ' User ' => table ' users '; class ' Man ' => table ' men ' )
2008-10-31 19:05:30 +00:00
* The table is required to have at least 'id auto_increment' primary key .
2008-05-30 11:40:08 +00:00
*
2010-12-24 13:57:20 -05:00
* @ package cake . libs . model
2010-04-04 23:31:50 -04:00
* @ link http :// book . cakephp . org / view / 1000 / Models
2008-05-30 11:40:08 +00:00
*/
2010-04-18 01:02:39 -04:00
class Model extends Object {
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* The name of the DataSource connection that this Model uses
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #useDbConfig-1058
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $useDbConfig = 'default' ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Custom database table name , or null / false if no table association is desired .
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #useTable-1059
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $useTable = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Custom display field name . Display fields are used by Scaffold , in SELECT boxes ' OPTION elements .
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #displayField-1062
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $displayField = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-11-08 02:54:07 +00:00
* Value of the primary key ID of the record that this model is currently pointing to .
2008-10-31 19:05:30 +00:00
* Automatically set after database insertions .
2008-05-30 11:40:08 +00:00
*
* @ var mixed
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $id = false ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Container for the data that this model gets from persistent storage ( usually , a database ) .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #data-1065
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $data = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Table name for this Model .
*
* @ var string
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $table = false ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* The name of the primary key field for this model .
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #primaryKey-1061
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $primaryKey = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Field - by - field table metadata .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access protected
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #_schema-1066
2008-05-30 11:40:08 +00:00
*/
2010-04-04 16:36:12 +10:00
protected $_schema = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* List of validation rules . Append entries for validation as ( 'field_name' => '/^perl_compat_regexp$/' )
* that have to match with preg_match () . Use these rules with Model :: validate ()
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #validate-1067
* @ link http :// book . cakephp . org / view / 1143 / Data - Validation
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $validate = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* List of validation errors .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1182 / Validating - Data - from - the - Controller
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $validationErrors = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Database table prefix for tables in model .
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #tablePrefix-1060
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $tablePrefix = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Name of the model .
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #name-1068
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $name = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Alias name for model .
*
* @ var string
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $alias = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* List of table names included in the model description . Used for associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $tableToModel = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Whether or not to log transactions for this model .
2008-05-30 11:40:08 +00:00
*
* @ var boolean
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $logTransactions = false ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Whether or not to cache queries for this model . This enables in - memory
2008-10-31 19:05:30 +00:00
* caching only , the results are not stored beyond the current request .
2008-05-30 11:40:08 +00:00
*
* @ var boolean
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #cacheQueries-1069
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $cacheQueries = false ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Detailed list of belongsTo associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1042 / belongsTo
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $belongsTo = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Detailed list of hasOne associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1041 / hasOne
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $hasOne = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Detailed list of hasMany associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1043 / hasMany
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $hasMany = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Detailed list of hasAndBelongsToMany associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1044 / hasAndBelongsToMany - HABTM
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $hasAndBelongsToMany = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* List of behaviors to load when the model object is initialized . Settings can be
* passed to behaviors by using the behavior name as index . Eg :
*
2010-04-04 17:14:00 +10:00
* public $actsAs = array ( 'Translate' , 'MyBehavior' => array ( 'setting1' => 'value1' ))
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1072 / Using - Behaviors
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $actsAs = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Holds the Behavior objects currently bound to this model .
2008-05-30 11:40:08 +00:00
*
2009-03-17 21:10:28 +00:00
* @ var BehaviorCollection
2008-05-30 11:40:08 +00:00
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $Behaviors = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Whitelist of fields allowed to be saved .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $whitelist = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Whether or not to cache sources for this model .
2008-05-30 11:40:08 +00:00
*
* @ var boolean
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $cacheSources = true ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Type of find query currently executing .
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $findQueryType = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-11-08 02:54:07 +00:00
* Number of associations to recurse through during find calls . Fetches only
2008-10-31 19:05:30 +00:00
* the first level by default .
2008-05-30 11:40:08 +00:00
*
* @ var integer
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #recursive-1063
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $recursive = 1 ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-11-08 02:54:07 +00:00
* The column name ( s ) and direction ( s ) to order find results by default .
*
2010-04-04 17:14:00 +10:00
* public $order = " Post.created DESC " ;
* public $order = array ( " Post.view_count DESC " , " Post.rating DESC " );
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1057 / Model - Attributes #order-1064
2008-05-30 11:40:08 +00:00
*/
2010-04-04 17:14:00 +10:00
public $order = null ;
2009-07-24 21:18:37 +02:00
2009-12-20 16:26:12 -05:00
/**
* Array of virtual fields this model has . Virtual fields are aliased
* SQL expressions . Fields added to this property will be read as other fields in a model
* but will not be saveable .
*
2010-04-04 17:14:00 +10:00
* `public $virtualFields = array('two' => '1 + 1');`
2009-12-20 16:26:12 -05:00
*
* Is a simplistic example of how to set virtualFields
*
* @ var array
* @ access public
*/
2010-04-04 17:14:00 +10:00
public $virtualFields = array ();
2009-12-20 16:26:12 -05:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Default list of association keys .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access private
*/
2010-04-04 16:33:39 +10:00
private $__associationKeys = array (
2009-03-08 18:05:18 +00:00
'belongsTo' => array ( 'className' , 'foreignKey' , 'conditions' , 'fields' , 'order' , 'counterCache' ),
'hasOne' => array ( 'className' , 'foreignKey' , 'conditions' , 'fields' , 'order' , 'dependent' ),
'hasMany' => array ( 'className' , 'foreignKey' , 'conditions' , 'fields' , 'order' , 'limit' , 'offset' , 'dependent' , 'exclusive' , 'finderQuery' , 'counterQuery' ),
'hasAndBelongsToMany' => array ( 'className' , 'joinTable' , 'with' , 'foreignKey' , 'associationForeignKey' , 'conditions' , 'fields' , 'order' , 'limit' , 'offset' , 'unique' , 'finderQuery' , 'deleteQuery' , 'insertQuery' )
);
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Holds provided / generated association key names and other data for all associations .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access private
*/
2010-04-04 16:33:39 +10:00
private $__associations = array ( 'belongsTo' , 'hasOne' , 'hasMany' , 'hasAndBelongsToMany' );
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Holds model associations temporarily to allow for dynamic ( un ) binding .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access private
*/
2010-12-12 15:48:36 -05:00
public $__backAssociation = array ();
public $__backInnerAssociation = array ();
public $__backOriginalAssociation = array ();
public $__backContainableAssociation = array ();
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* The ID of the model record that was last inserted .
2008-05-30 11:40:08 +00:00
*
* @ var integer
* @ access private
*/
2010-04-04 16:33:39 +10:00
private $__insertID = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* The number of records returned by the last query .
2008-05-30 11:40:08 +00:00
*
* @ var integer
* @ access private
*/
2010-04-04 16:33:39 +10:00
private $__numRows = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* The number of records affected by the last query .
2008-05-30 11:40:08 +00:00
*
* @ var integer
* @ access private
*/
2010-04-04 16:33:39 +10:00
private $__affectedRows = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* List of valid finder method options , supplied as the first parameter to find () .
2008-05-30 11:40:08 +00:00
*
* @ var array
2008-09-21 04:09:16 +00:00
* @ access protected
2008-05-30 11:40:08 +00:00
*/
2010-04-04 16:36:12 +10:00
protected $_findMethods = array (
2008-09-21 04:09:16 +00:00
'all' => true , 'first' => true , 'count' => true ,
'neighbors' => true , 'list' => true , 'threaded' => true
);
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Constructor . Binds the model ' s database table to the object .
2008-05-30 11:40:08 +00:00
*
2010-01-01 23:58:07 -05:00
* If `$id` is an array it can be used to pass several options into the model .
*
* - id - The id to start the model on .
* - table - The table to use for this model .
* - ds - The connection name this model is connected to .
* - name - The name of the model eg . Post .
* - alias - The alias of the model , this is used for registering the instance in the `ClassRegistry` .
* eg . `ParentThread`
*
* ### Overriding Model's __construct method.
2010-01-14 12:57:43 -05:00
*
* When overriding Model :: __construct () be careful to include and pass in all 3 of the
2010-01-01 23:58:07 -05:00
* arguments to `parent::__construct($id, $table, $ds);`
*
* ### Dynamically creating models
*
2010-07-19 15:29:18 -07:00
* You can dynamically create model instances using the $id array syntax .
2010-01-14 12:57:43 -05:00
*
2010-01-01 23:58:07 -05:00
* {{{
* $Post = new Model ( array ( 'table' => 'posts' , 'name' => 'Post' , 'ds' => 'connection2' ));
* }}}
*
* Would create a model attached to the posts table on connection2 . Dynamic model creation is useful
* when you want a model object that contains no associations or attached behaviors .
*
* @ param mixed $id Set this ID for this model on startup , can also be an array of options , see above .
2008-05-30 11:40:08 +00:00
* @ param string $table Name of database table to use .
2010-01-01 23:58:07 -05:00
* @ param string $ds DataSource connection name .
2008-05-30 11:40:08 +00:00
*/
function __construct ( $id = false , $table = null , $ds = null ) {
parent :: __construct ();
if ( is_array ( $id )) {
2008-08-25 22:33:36 +00:00
extract ( array_merge (
2008-09-21 04:09:16 +00:00
array (
'id' => $this -> id , 'table' => $this -> useTable , 'ds' => $this -> useDbConfig ,
'name' => $this -> name , 'alias' => $this -> alias
),
$id
));
2008-05-30 11:40:08 +00:00
}
if ( $this -> name === null ) {
2008-09-21 04:09:16 +00:00
$this -> name = ( isset ( $name ) ? $name : get_class ( $this ));
2008-05-30 11:40:08 +00:00
}
if ( $this -> alias === null ) {
2008-09-21 04:09:16 +00:00
$this -> alias = ( isset ( $alias ) ? $alias : $this -> name );
2008-05-30 11:40:08 +00:00
}
if ( $this -> primaryKey === null ) {
$this -> primaryKey = 'id' ;
}
2008-09-26 15:27:36 +00:00
2008-05-30 11:40:08 +00:00
ClassRegistry :: addObject ( $this -> alias , $this );
$this -> id = $id ;
unset ( $id );
if ( $table === false ) {
$this -> useTable = false ;
} elseif ( $table ) {
$this -> useTable = $table ;
}
2009-07-24 21:18:37 +02:00
2009-06-16 21:35:21 +00:00
if ( $ds !== null ) {
$this -> useDbConfig = $ds ;
}
2008-05-30 11:40:08 +00:00
2008-09-21 04:09:16 +00:00
if ( is_subclass_of ( $this , 'AppModel' )) {
$merge = array ( '_findMethods' );
if ( $this -> actsAs !== null || $this -> actsAs !== false ) {
$merge [] = 'actsAs' ;
}
2008-11-14 20:24:27 +00:00
$parentClass = get_parent_class ( $this );
2010-11-20 00:20:54 -05:00
if ( $parentClass !== 'AppModel' ) {
$this -> _mergeVars ( $merge , $parentClass );
2008-09-21 04:09:16 +00:00
}
2010-11-20 00:20:54 -05:00
$this -> _mergeVars ( $merge , 'AppModel' );
2008-09-21 04:09:16 +00:00
}
$this -> Behaviors = new BehaviorCollection ();
2008-05-30 11:40:08 +00:00
if ( $this -> useTable !== false ) {
if ( $this -> useTable === null ) {
$this -> useTable = Inflector :: tableize ( $this -> name );
}
if ( $this -> displayField == null ) {
2010-07-15 21:18:16 -04:30
unset ( $this -> displayField );
2008-05-30 11:40:08 +00:00
}
2010-07-15 21:16:52 -04:30
$this -> table = $this -> useTable ;
$this -> tableToModel [ $this -> table ] = $this -> alias ;
2008-10-15 17:30:08 +00:00
} elseif ( $this -> table === false ) {
$this -> table = Inflector :: tableize ( $this -> name );
2008-05-30 11:40:08 +00:00
}
2008-10-15 17:30:08 +00:00
$this -> __createLinks ();
2008-09-22 16:32:41 +00:00
$this -> Behaviors -> init ( $this -> alias , $this -> actsAs );
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Handles custom method calls , like findBy < field > for DB models ,
* and custom RPC calls for remote data sources .
*
* @ param string $method Name of method to call .
* @ param array $params Parameters for the method .
* @ return mixed Whatever is returned by called method
*/
2010-04-05 16:43:20 +10:00
public function __call ( $method , $params ) {
2008-05-30 11:40:08 +00:00
$result = $this -> Behaviors -> dispatchMethod ( $this , $method , $params );
if ( $result !== array ( 'unhandled' )) {
return $result ;
}
2010-07-14 23:19:38 -04:30
$return = $this -> getDataSource () -> query ( $method , $params , $this );
2008-05-30 11:40:08 +00:00
return $return ;
}
2009-07-24 21:18:37 +02:00
2010-07-14 16:58:12 -04:30
/**
* Handles the lazy loading of model associations by lookin in the association arrays for the requested variable
*
* @ param string $name variable tested for existance in class
* @ return boolean true if the variable exists ( if is a not loaded model association it will be created ), false otherwise
*/
public function __isset ( $name ) {
$className = false ;
foreach ( $this -> __associations as $type ) {
if ( isset ( $name , $this -> { $type }[ $name ])) {
$className = empty ( $this -> { $type }[ $name ][ 'className' ]) ? $name : $this -> { $type }[ $name ][ 'className' ];
break ;
} else if ( $type == 'hasAndBelongsToMany' ) {
foreach ( $this -> { $type } as $k => $relation ) {
2010-07-14 17:54:19 -04:30
if ( empty ( $relation [ 'with' ])) {
continue ;
}
2010-07-14 23:19:38 -04:30
if ( is_array ( $relation [ 'with' ])) {
if ( key ( $relation [ 'with' ]) === $name ) {
$className = $name ;
}
2010-07-14 17:54:19 -04:30
} else {
list ( $plugin , $class ) = pluginSplit ( $relation [ 'with' ]);
if ( $class === $name ) {
$className = $relation [ 'with' ];
}
}
if ( $className ) {
$assocKey = $k ;
2010-07-14 23:19:38 -04:30
$dynamic = ! empty ( $relation [ 'dynamicWith' ]);
2010-07-14 16:58:12 -04:30
break ( 2 );
}
}
}
}
if ( ! $className ) {
return false ;
}
list ( $plugin , $className ) = pluginSplit ( $className );
2010-07-14 23:19:38 -04:30
if ( ! ClassRegistry :: isKeySet ( $className ) && ! empty ( $dynamic )) {
$this -> { $className } = new AppModel ( array (
'name' => $className ,
'table' => $this -> hasAndBelongsToMany [ $assocKey ][ 'joinTable' ],
'ds' => $this -> useDbConfig
));
} else {
$this -> __constructLinkedModel ( $name , $className , $plugin );
}
2010-07-14 17:54:19 -04:30
if ( ! empty ( $assocKey )) {
$this -> hasAndBelongsToMany [ $assocKey ][ 'joinTable' ] = $this -> { $name } -> table ;
if ( count ( $this -> { $name } -> schema ()) <= 2 && $this -> { $name } -> primaryKey !== false ) {
$this -> { $name } -> primaryKey = $this -> hasAndBelongsToMany [ $assocKey ][ 'foreignKey' ];
}
}
2010-07-15 23:17:13 -04:30
return true ;
2010-07-14 16:58:12 -04:30
}
/**
* Returns the value of the requested variable if it can be set by __isset ()
*
* @ param string $name variable requested for it ' s value or reference
* @ return mixed value of requested variable if it is set
*/
function __get ( $name ) {
2010-07-15 21:18:16 -04:30
if ( $name === 'displayField' ) {
return $this -> displayField = $this -> hasField ( array ( 'title' , 'name' , $this -> primaryKey ));
}
2010-07-14 16:58:12 -04:30
if ( isset ( $this -> { $name })) {
return $this -> { $name };
}
}
2008-05-30 11:40:08 +00:00
/**
* Bind model associations on the fly .
*
2010-05-01 17:37:23 -04:00
* If `$reset` is false , association will not be reset
2008-05-30 11:40:08 +00:00
* to the originals defined in the model
*
* Example : Add a new hasOne binding to the Profile model not
* defined in the model source code :
2010-05-01 17:37:23 -04:00
*
* `$this->User->bindModel( array('hasOne' => array('Profile')) );`
*
* Bindings that are not made permanent will be reset by the next Model :: find () call on this
* model .
2008-05-30 11:40:08 +00:00
*
* @ param array $params Set of bindings ( indexed by binding type )
* @ param boolean $reset Set to false to make the binding permanent
* @ return boolean Success
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1045 / Creating - and - Destroying - Associations - on - the - Fly
2008-05-30 11:40:08 +00:00
*/
function bindModel ( $params , $reset = true ) {
foreach ( $params as $assoc => $model ) {
2010-07-03 18:05:14 -04:00
if ( $reset === true && ! isset ( $this -> __backAssociation [ $assoc ])) {
2008-05-30 11:40:08 +00:00
$this -> __backAssociation [ $assoc ] = $this -> { $assoc };
}
foreach ( $model as $key => $value ) {
$assocName = $key ;
if ( is_numeric ( $key )) {
$assocName = $value ;
$value = array ();
}
$modelName = $assocName ;
$this -> { $assoc }[ $assocName ] = $value ;
2010-07-14 16:58:12 -04:30
if ( property_exists ( $this , $assocName )) {
unset ( $this -> { $assocName });
}
2010-07-01 12:39:50 -04:00
if ( $reset === false && isset ( $this -> __backAssociation [ $assoc ])) {
$this -> __backAssociation [ $assoc ][ $assocName ] = $value ;
}
2008-05-30 11:40:08 +00:00
}
}
$this -> __createLinks ();
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Turn off associations on the fly .
*
* If $reset is false , association will not be reset
* to the originals defined in the model
*
* Example : Turn off the associated Model Support request ,
* to temporarily lighten the User model :
2010-11-03 18:55:42 +05:30
*
2010-05-01 17:37:23 -04:00
* `$this->User->unbindModel( array('hasMany' => array('Supportrequest')) );`
2010-11-03 18:55:42 +05:30
*
2010-05-01 17:37:23 -04:00
* unbound models that are not made permanent will reset with the next call to Model :: find ()
2008-05-30 11:40:08 +00:00
*
* @ param array $params Set of bindings to unbind ( indexed by binding type )
* @ param boolean $reset Set to false to make the unbinding permanent
* @ return boolean Success
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1045 / Creating - and - Destroying - Associations - on - the - Fly
2008-05-30 11:40:08 +00:00
*/
function unbindModel ( $params , $reset = true ) {
foreach ( $params as $assoc => $models ) {
2010-07-03 18:05:14 -04:00
if ( $reset === true && ! isset ( $this -> __backAssociation [ $assoc ])) {
2008-05-30 11:40:08 +00:00
$this -> __backAssociation [ $assoc ] = $this -> { $assoc };
}
foreach ( $models as $model ) {
2010-07-03 18:05:14 -04:00
if ( $reset === false && isset ( $this -> __backAssociation [ $assoc ][ $model ])) {
unset ( $this -> __backAssociation [ $assoc ][ $model ]);
}
2010-07-01 12:39:50 -04:00
unset ( $this -> { $assoc }[ $model ]);
2008-05-30 11:40:08 +00:00
}
}
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Create a set of associations .
2008-05-30 11:40:08 +00:00
*
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
* @ access private
*/
function __createLinks () {
foreach ( $this -> __associations as $type ) {
if ( ! is_array ( $this -> { $type })) {
$this -> { $type } = explode ( ',' , $this -> { $type });
foreach ( $this -> { $type } as $i => $className ) {
$className = trim ( $className );
unset ( $this -> { $type }[ $i ]);
$this -> { $type }[ $className ] = array ();
}
}
2008-06-11 08:54:27 +00:00
if ( ! empty ( $this -> { $type })) {
foreach ( $this -> { $type } as $assoc => $value ) {
$plugin = null ;
2008-05-30 11:40:08 +00:00
2008-06-11 08:54:27 +00:00
if ( is_numeric ( $assoc )) {
unset ( $this -> { $type }[ $assoc ]);
$assoc = $value ;
$value = array ();
2008-05-30 11:40:08 +00:00
$this -> { $type }[ $assoc ] = $value ;
2008-06-11 08:54:27 +00:00
if ( strpos ( $assoc , '.' ) !== false ) {
2010-07-13 23:28:48 -04:30
list ( $plugin , $assoc ) = pluginSplit ( $assoc );
$this -> { $type }[ $assoc ] = array ( 'className' => $plugin . '.' . $assoc );
2008-06-11 08:54:27 +00:00
}
2008-05-30 11:40:08 +00:00
}
2010-07-14 16:58:12 -04:30
$this -> __generateAssociation ( $type , $assoc );
2008-05-30 11:40:08 +00:00
}
}
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Private helper method to create associated models of a given class .
2008-05-30 11:40:08 +00:00
*
* @ param string $assoc Association name
* @ param string $className Class name
2010-07-13 23:28:48 -04:30
* @ param string $plugin name of the plugin where $className is located
2010-04-04 17:14:00 +10:00
* examples : public $hasMany = array ( 'Assoc' => array ( 'className' => 'ModelName' ));
2008-05-30 11:40:08 +00:00
* usage : $this -> Assoc -> modelMethods ();
*
2010-04-04 17:14:00 +10:00
* public $hasMany = array ( 'ModelName' );
2008-05-30 11:40:08 +00:00
* usage : $this -> ModelName -> modelMethods ();
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
* @ access private
*/
2010-07-13 23:28:48 -04:30
function __constructLinkedModel ( $assoc , $className = null , $plugin = null ) {
2008-10-23 00:10:44 +00:00
if ( empty ( $className )) {
2008-05-30 11:40:08 +00:00
$className = $assoc ;
}
2008-09-19 15:27:43 +00:00
if ( ! isset ( $this -> { $assoc }) || $this -> { $assoc } -> name !== $className ) {
2010-07-13 23:28:48 -04:30
$model = array ( 'class' => $plugin . '.' . $className , 'alias' => $assoc );
2010-07-05 22:19:22 -04:00
$this -> { $assoc } = ClassRegistry :: init ( $model );
2010-07-14 19:23:41 -04:30
if ( $plugin ) {
ClassRegistry :: addObject ( $plugin . '.' . $className , $this -> { $assoc });
2010-01-18 13:05:30 -04:30
}
2008-06-11 08:54:27 +00:00
if ( $assoc ) {
$this -> tableToModel [ $this -> { $assoc } -> table ] = $assoc ;
}
2008-05-30 11:40:08 +00:00
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Build an array - based association from string .
2008-05-30 11:40:08 +00:00
*
* @ param string $type 'belongsTo' , 'hasOne' , 'hasMany' , 'hasAndBelongsToMany'
2010-07-14 16:58:12 -04:30
* @ param string $assocKey
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
* @ access private
*/
2010-07-14 16:58:12 -04:30
function __generateAssociation ( $type , $assocKey ) {
$class = $assocKey ;
$dynamicWith = false ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
foreach ( $this -> __associationKeys [ $type ] as $key ) {
2008-10-12 22:21:55 +00:00
2010-07-14 16:58:12 -04:30
if ( ! isset ( $this -> { $type }[ $assocKey ][ $key ]) || $this -> { $type }[ $assocKey ][ $key ] === null ) {
$data = '' ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
switch ( $key ) {
case 'fields' :
$data = '' ;
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'foreignKey' :
$data = (( $type == 'belongsTo' ) ? Inflector :: underscore ( $assocKey ) : Inflector :: singularize ( $this -> table )) . '_id' ;
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'associationForeignKey' :
$data = Inflector :: singularize ( $this -> { $class } -> table ) . '_id' ;
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'with' :
$data = Inflector :: camelize ( Inflector :: singularize ( $this -> { $type }[ $assocKey ][ 'joinTable' ]));
$dynamicWith = true ;
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'joinTable' :
$tables = array ( $this -> table , $this -> { $class } -> table );
sort ( $tables );
$data = $tables [ 0 ] . '_' . $tables [ 1 ];
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'className' :
$data = $class ;
break ;
2008-05-30 11:40:08 +00:00
2010-07-14 16:58:12 -04:30
case 'unique' :
$data = true ;
break ;
2008-05-30 11:40:08 +00:00
}
2010-07-14 16:58:12 -04:30
$this -> { $type }[ $assocKey ][ $key ] = $data ;
2008-05-30 11:40:08 +00:00
}
2010-07-14 23:19:38 -04:30
if ( $dynamicWith ) {
$this -> { $type }[ $assocKey ][ 'dynamicWith' ] = true ;
2010-07-14 16:58:12 -04:30
}
2008-05-30 11:40:08 +00:00
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Sets a custom table for your controller class . Used by your controller to select a database table .
*
* @ param string $tableName Name of the custom table
2010-07-15 23:11:30 -04:30
* @ throws MissingTableException when database table $tableName is not found on data source
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
*/
2010-04-05 13:19:38 +10:00
public function setSource ( $tableName ) {
2008-05-30 11:40:08 +00:00
$this -> setDataSource ( $this -> useDbConfig );
2010-07-15 21:16:52 -04:30
$db = ConnectionManager :: getDataSource ( $this -> useDbConfig );
2008-08-27 04:55:15 +00:00
$db -> cacheSources = ( $this -> cacheSources && $db -> cacheSources );
2008-05-30 11:40:08 +00:00
if ( $db -> isInterfaceSupported ( 'listSources' )) {
$sources = $db -> listSources ();
if ( is_array ( $sources ) && ! in_array ( strtolower ( $this -> tablePrefix . $tableName ), array_map ( 'strtolower' , $sources ))) {
2010-08-29 21:37:25 -04:00
throw new MissingTableException ( array (
'table' => $this -> tablePrefix . $tableName ,
'class' => $this -> alias
));
2008-05-30 11:40:08 +00:00
}
$this -> _schema = null ;
}
$this -> table = $this -> useTable = $tableName ;
$this -> tableToModel [ $this -> table ] = $this -> alias ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2009-11-11 23:47:23 -04:30
* This function does two things :
2009-10-05 21:27:34 -04:00
*
* 1. it scans the array $one for the primary key ,
2008-05-30 11:40:08 +00:00
* and if that ' s found , it sets the current id to the value of $one [ id ] .
* For all other keys than 'id' the keys and values of $one are copied to the 'data' property of this object .
2009-10-05 21:27:34 -04:00
* 2. Returns an array with all of $one ' s keys and values .
2008-05-30 11:40:08 +00:00
* ( Alternative indata : two strings , which are mangled to
* a one - item , two - dimensional array using $one for a key and $two as its value . )
*
* @ param mixed $one Array or string of data
* @ param string $two Value string for the alternative indata method
* @ return array Data with all of $one ' s keys and values
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function set ( $one , $two = null ) {
if ( ! $one ) {
return ;
}
if ( is_object ( $one )) {
2010-10-17 22:47:43 -02:00
if ( $one instanceof SimpleXMLElement || $one instanceof DOMNode ) {
App :: import ( 'Core' , 'Xml' );
$one = $this -> _normalizeXmlData ( Xml :: toArray ( $one ));
} else {
$one = Set :: reverse ( $one );
}
2008-05-30 11:40:08 +00:00
}
if ( is_array ( $one )) {
2008-06-14 20:39:27 +00:00
$data = $one ;
if ( empty ( $one [ $this -> alias ])) {
2008-08-01 06:28:59 +00:00
if ( $this -> getAssociated ( key ( $one )) === null ) {
2008-06-14 20:39:27 +00:00
$data = array ( $this -> alias => $one );
}
2008-05-30 11:40:08 +00:00
}
} else {
$data = array ( $this -> alias => array ( $one => $two ));
}
2008-10-06 02:44:05 +00:00
foreach ( $data as $modelName => $fieldSet ) {
if ( is_array ( $fieldSet )) {
2008-05-30 11:40:08 +00:00
2008-10-06 02:44:05 +00:00
foreach ( $fieldSet as $fieldName => $fieldValue ) {
if ( isset ( $this -> validationErrors [ $fieldName ])) {
unset ( $this -> validationErrors [ $fieldName ]);
2008-05-30 11:40:08 +00:00
}
2008-10-06 02:44:05 +00:00
if ( $modelName === $this -> alias ) {
if ( $fieldName === $this -> primaryKey ) {
$this -> id = $fieldValue ;
2008-05-30 11:40:08 +00:00
}
}
2008-10-06 02:44:05 +00:00
if ( is_array ( $fieldValue ) || is_object ( $fieldValue )) {
$fieldValue = $this -> deconstruct ( $fieldName , $fieldValue );
2008-05-30 11:40:08 +00:00
}
2008-10-06 02:44:05 +00:00
$this -> data [ $modelName ][ $fieldName ] = $fieldValue ;
2008-05-30 11:40:08 +00:00
}
}
}
return $data ;
}
2009-07-24 21:18:37 +02:00
2010-10-17 22:47:43 -02:00
/**
* Normalize Xml :: toArray () to use in Model :: save ()
*
* @ param array $xml XML as array
* @ return array
*/
protected function _normalizeXmlData ( array $xml ) {
$return = array ();
foreach ( $xml as $key => $value ) {
if ( is_array ( $value )) {
$return [ Inflector :: camelize ( $key )] = $this -> _normalizeXmlData ( $value );
} elseif ( $key [ 0 ] === '@' ) {
$return [ substr ( $key , 1 )] = $value ;
} else {
$return [ $key ] = $value ;
}
}
return $return ;
}
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Deconstructs a complex data type ( array or object ) into a single field value .
2008-05-30 11:40:08 +00:00
*
* @ param string $field The name of the field to be deconstructed
* @ param mixed $data An array or object to be deconstructed into a field
* @ return mixed The resulting data that should be assigned to a field
*/
2010-04-05 13:19:38 +10:00
public function deconstruct ( $field , $data ) {
2009-06-02 03:23:15 +00:00
if ( ! is_array ( $data )) {
return $data ;
}
2008-05-30 11:40:08 +00:00
$copy = $data ;
$type = $this -> getColumnType ( $field );
if ( in_array ( $type , array ( 'datetime' , 'timestamp' , 'date' , 'time' ))) {
2009-07-24 21:18:37 +02:00
$useNewDate = ( isset ( $data [ 'year' ]) || isset ( $data [ 'month' ]) ||
2009-07-08 03:25:30 +00:00
isset ( $data [ 'day' ]) || isset ( $data [ 'hour' ]) || isset ( $data [ 'minute' ]));
2008-05-30 11:40:08 +00:00
$dateFields = array ( 'Y' => 'year' , 'm' => 'month' , 'd' => 'day' , 'H' => 'hour' , 'i' => 'min' , 's' => 'sec' );
2009-07-08 03:25:30 +00:00
$timeFields = array ( 'H' => 'hour' , 'i' => 'min' , 's' => 'sec' );
2008-05-30 11:40:08 +00:00
$date = array ();
if ( isset ( $data [ 'hour' ]) && isset ( $data [ 'meridian' ]) && $data [ 'hour' ] != 12 && 'pm' == $data [ 'meridian' ]) {
$data [ 'hour' ] = $data [ 'hour' ] + 12 ;
}
if ( isset ( $data [ 'hour' ]) && isset ( $data [ 'meridian' ]) && $data [ 'hour' ] == 12 && 'am' == $data [ 'meridian' ]) {
$data [ 'hour' ] = '00' ;
}
2009-07-08 03:25:30 +00:00
if ( $type == 'time' ) {
foreach ( $timeFields as $key => $val ) {
if ( ! isset ( $data [ $val ]) || $data [ $val ] === '0' || $data [ $val ] === '00' ) {
2008-05-30 11:40:08 +00:00
$data [ $val ] = '00' ;
2009-07-08 03:25:30 +00:00
} elseif ( $data [ $val ] === '' ) {
$data [ $val ] = '' ;
2008-05-30 11:40:08 +00:00
} else {
$data [ $val ] = sprintf ( '%02d' , $data [ $val ]);
}
2009-07-08 03:25:30 +00:00
if ( ! empty ( $data [ $val ])) {
$date [ $key ] = $data [ $val ];
} else {
return null ;
}
2008-05-30 11:40:08 +00:00
}
2009-07-08 03:25:30 +00:00
}
if ( $type == 'datetime' || $type == 'timestamp' || $type == 'date' ) {
foreach ( $dateFields as $key => $val ) {
if ( $val == 'hour' || $val == 'min' || $val == 'sec' ) {
if ( ! isset ( $data [ $val ]) || $data [ $val ] === '0' || $data [ $val ] === '00' ) {
$data [ $val ] = '00' ;
} else {
$data [ $val ] = sprintf ( '%02d' , $data [ $val ]);
}
}
if ( ! isset ( $data [ $val ]) || isset ( $data [ $val ]) && ( empty ( $data [ $val ]) || $data [ $val ][ 0 ] === '-' )) {
return null ;
}
if ( isset ( $data [ $val ]) && ! empty ( $data [ $val ])) {
$date [ $key ] = $data [ $val ];
}
2008-05-30 11:40:08 +00:00
}
}
2010-11-27 00:13:04 -04:30
$format = $this -> getDataSource () -> columns [ $type ][ 'format' ];
$day = empty ( $date [ 'Y' ]) ? null : $date [ 'Y' ] . '-' . $date [ 'm' ] . '-' . $date [ 'd' ] . ' ' ;
$hour = empty ( $date [ 'H' ]) ? null : $date [ 'H' ] . ':' . $date [ 'i' ] . ':' . $date [ 's' ];
$date = new DateTime ( $day . $hour );
2009-07-08 03:25:30 +00:00
if ( $useNewDate && ! empty ( $date )) {
2010-11-27 00:13:04 -04:30
return $date -> format ( $format );
2008-05-30 11:40:08 +00:00
}
}
return $data ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Returns an array of table metadata ( column names and types ) from the database .
* $field => keys ( type , null , default , key , length , extra )
*
* @ param mixed $field Set to true to reload schema , or a string to return a specific field
* @ return array Array of table metadata
*/
2010-04-05 13:19:38 +10:00
public function schema ( $field = false ) {
2008-05-30 11:40:08 +00:00
if ( ! is_array ( $this -> _schema ) || $field === true ) {
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-08-27 04:55:15 +00:00
$db -> cacheSources = ( $this -> cacheSources && $db -> cacheSources );
2008-05-30 11:40:08 +00:00
if ( $db -> isInterfaceSupported ( 'describe' ) && $this -> useTable !== false ) {
$this -> _schema = $db -> describe ( $this , $field );
} elseif ( $this -> useTable === false ) {
$this -> _schema = array ();
}
}
if ( is_string ( $field )) {
if ( isset ( $this -> _schema [ $field ])) {
return $this -> _schema [ $field ];
} else {
return null ;
}
}
return $this -> _schema ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Returns an associative array of field names and column types .
*
* @ return array Field types indexed by field name
*/
2010-04-05 13:19:38 +10:00
public function getColumnTypes () {
2008-05-30 11:40:08 +00:00
$columns = $this -> schema ();
if ( empty ( $columns )) {
2010-04-16 02:00:25 +10:00
trigger_error ( __ ( '(Model::getColumnTypes) Unable to build model field data. If you are using a model without a database table, try implementing schema()' ), E_USER_WARNING );
2008-05-30 11:40:08 +00:00
}
$cols = array ();
foreach ( $columns as $field => $values ) {
$cols [ $field ] = $values [ 'type' ];
}
return $cols ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the column type of a column in the model .
2008-05-30 11:40:08 +00:00
*
* @ param string $column The name of the model column
* @ return string Column type
*/
2010-04-05 13:19:38 +10:00
public function getColumnType ( $column ) {
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-05-30 11:40:08 +00:00
$cols = $this -> schema ();
$model = null ;
2008-11-10 19:53:04 +00:00
$column = str_replace ( array ( $db -> startQuote , $db -> endQuote ), '' , $column );
2008-05-30 11:40:08 +00:00
if ( strpos ( $column , '.' )) {
list ( $model , $column ) = explode ( '.' , $column );
}
if ( $model != $this -> alias && isset ( $this -> { $model })) {
return $this -> { $model } -> getColumnType ( $column );
}
if ( isset ( $cols [ $column ]) && isset ( $cols [ $column ][ 'type' ])) {
return $cols [ $column ][ 'type' ];
}
return null ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns true if the supplied field exists in the model ' s database table .
2008-05-30 11:40:08 +00:00
*
* @ param mixed $name Name of field to look for , or an array of names
2009-11-12 00:39:11 -04:30
* @ param boolean $checkVirtual checks if the field is declared as virtual
2008-05-30 11:40:08 +00:00
* @ return mixed If $name is a string , returns a boolean indicating whether the field exists .
* If $name is an array of field names , returns the first field that exists ,
* or false if none exist .
*/
2010-04-05 13:19:38 +10:00
public function hasField ( $name , $checkVirtual = false ) {
2008-05-30 11:40:08 +00:00
if ( is_array ( $name )) {
foreach ( $name as $n ) {
2009-12-17 00:15:42 -05:00
if ( $this -> hasField ( $n , $checkVirtual )) {
2008-05-30 11:40:08 +00:00
return $n ;
}
}
return false ;
}
2009-11-12 00:39:11 -04:30
if ( $checkVirtual && ! empty ( $this -> virtualFields )) {
2009-12-06 22:16:00 -04:30
if ( $this -> isVirtualField ( $name )) {
2009-11-12 00:39:11 -04:30
return true ;
}
}
2008-05-30 11:40:08 +00:00
if ( empty ( $this -> _schema )) {
$this -> schema ();
}
if ( $this -> _schema != null ) {
return isset ( $this -> _schema [ $name ]);
}
return false ;
}
2009-07-24 21:18:37 +02:00
2010-12-26 17:35:22 -05:00
/**
* Check that a method is callable on a model . This will check both the model ' s own methods , its
* inherited methods and methods that could be callable through behaviors .
*
* @ param string $method The method to be called .
* @ return boolean True on method being callable .
*/
public function hasMethod ( $method ) {
if ( method_exists ( $this , $method )) {
return true ;
}
if ( $this -> Behaviors -> hasMethod ( $method )) {
return true ;
}
return false ;
}
2009-12-06 22:16:00 -04:30
/**
* Returns true if the supplied field is a model Virtual Field
*
* @ param mixed $name Name of field to look for
* @ return boolean indicating whether the field exists as a model virtual field .
*/
2010-04-05 13:19:38 +10:00
public function isVirtualField ( $field ) {
2010-01-19 09:22:13 -05:00
if ( empty ( $this -> virtualFields ) || ! is_string ( $field )) {
return false ;
}
if ( isset ( $this -> virtualFields [ $field ])) {
return true ;
}
if ( strpos ( $field , '.' ) !== false ) {
list ( $model , $field ) = explode ( '.' , $field );
if ( isset ( $this -> virtualFields [ $field ])) {
return true ;
}
}
return false ;
2009-12-06 22:16:00 -04:30
}
/**
2010-01-14 12:57:43 -05:00
* Returns the expression for a model virtual field
2009-12-06 22:16:00 -04:30
*
* @ param mixed $name Name of field to look for
* @ return mixed If $field is string expression bound to virtual field $field
2009-12-20 16:26:12 -05:00
* If $field is null , returns an array of all model virtual fields
* or false if none $field exist .
2009-12-06 22:16:00 -04:30
*/
2010-04-05 13:19:38 +10:00
public function getVirtualField ( $field = null ) {
2009-12-06 22:16:00 -04:30
if ( $field == null ) {
return empty ( $this -> virtualFields ) ? false : $this -> virtualFields ;
}
if ( $this -> isVirtualField ( $field )) {
2010-01-19 09:22:13 -05:00
if ( strpos ( $field , '.' ) !== false ) {
list ( $model , $field ) = explode ( '.' , $field );
}
2009-12-06 22:16:00 -04:30
return $this -> virtualFields [ $field ];
}
return false ;
}
2008-05-30 11:40:08 +00:00
/**
* Initializes the model for writing a new record , loading the default values
2009-11-11 23:47:23 -04:30
* for those fields that are not defined in $data , and clearing previous validation errors .
2009-10-12 23:38:55 -04:00
* Especially helpful for saving data in loops .
2008-05-30 11:40:08 +00:00
*
* @ param mixed $data Optional data array to assign to the model after it is created . If null or false ,
2009-10-12 23:38:55 -04:00
* schema data defaults are not merged .
2008-05-30 11:40:08 +00:00
* @ param boolean $filterKey If true , overwrites any primary key input with an empty value
* @ return array The current Model :: data ; after merging $data and / or defaults from database
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function create ( $data = array (), $filterKey = false ) {
$defaults = array ();
$this -> id = false ;
$this -> data = array ();
$this -> validationErrors = array ();
if ( $data !== null && $data !== false ) {
foreach ( $this -> schema () as $field => $properties ) {
if ( $this -> primaryKey !== $field && isset ( $properties [ 'default' ])) {
$defaults [ $field ] = $properties [ 'default' ];
}
}
$this -> set ( Set :: filter ( $defaults ));
$this -> set ( $data );
}
if ( $filterKey ) {
$this -> set ( $this -> primaryKey , false );
}
return $this -> data ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Returns a list of fields from the database , and sets the current model
* data ( Model :: $data ) with the record found .
*
* @ param mixed $fields String of single fieldname , or an array of fieldnames .
* @ param mixed $id The ID of the record to read
* @ return array Array of database fields , or false if not found
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1017 / Retrieving - Your - Data #read-1029
2008-05-30 11:40:08 +00:00
*/
function read ( $fields = null , $id = null ) {
$this -> validationErrors = array ();
if ( $id != null ) {
$this -> id = $id ;
}
$id = $this -> id ;
if ( is_array ( $this -> id )) {
$id = $this -> id [ 0 ];
}
if ( $id !== null && $id !== false ) {
2009-05-04 15:57:10 -07:00
$this -> data = $this -> find ( 'first' , array (
'conditions' => array ( $this -> alias . '.' . $this -> primaryKey => $id ),
'fields' => $fields
));
2008-05-30 11:40:08 +00:00
return $this -> data ;
} else {
return false ;
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the contents of a single field given the supplied conditions , in the
* supplied order .
2008-05-30 11:40:08 +00:00
*
* @ param string $name Name of field to get
* @ param array $conditions SQL conditions ( defaults to NULL )
* @ param string $order SQL ORDER BY fragment
* @ return string field contents , or false if not found
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1017 / Retrieving - Your - Data #field-1028
2008-05-30 11:40:08 +00:00
*/
function field ( $name , $conditions = null , $order = null ) {
if ( $conditions === null && $this -> id !== false ) {
$conditions = array ( $this -> alias . '.' . $this -> primaryKey => $this -> id );
}
if ( $this -> recursive >= 1 ) {
$recursive = - 1 ;
} else {
$recursive = $this -> recursive ;
}
2009-11-11 23:43:37 -04:30
$fields = $name ;
2009-12-17 00:15:42 -05:00
if ( $data = $this -> find ( 'first' , compact ( 'conditions' , 'fields' , 'order' , 'recursive' ))) {
2008-05-30 11:40:08 +00:00
if ( strpos ( $name , '.' ) === false ) {
if ( isset ( $data [ $this -> alias ][ $name ])) {
return $data [ $this -> alias ][ $name ];
}
} else {
$name = explode ( '.' , $name );
if ( isset ( $data [ $name [ 0 ]][ $name [ 1 ]])) {
return $data [ $name [ 0 ]][ $name [ 1 ]];
}
}
2009-12-10 20:13:09 -04:30
if ( isset ( $data [ 0 ]) && count ( $data [ 0 ]) > 0 ) {
return array_shift ( $data [ 0 ]);
2008-05-30 11:40:08 +00:00
}
} else {
return false ;
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Saves the value of a single field to the database , based on the current
* model ID .
2008-05-30 11:40:08 +00:00
*
* @ param string $name Name of the table field
* @ param mixed $value Value of the field
2008-07-05 11:02:09 +00:00
* @ param array $validate See $options param in Model :: save () . Does not respect 'fieldList' key if passed
2008-06-20 20:17:23 +00:00
* @ return boolean See Model :: save ()
2008-05-30 11:40:08 +00:00
* @ access public
* @ see Model :: save ()
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function saveField ( $name , $value , $validate = false ) {
$id = $this -> id ;
$this -> create ( false );
2008-06-20 20:17:23 +00:00
if ( is_array ( $validate )) {
2008-07-05 11:02:09 +00:00
$options = array_merge ( array ( 'validate' => false , 'fieldList' => array ( $name )), $validate );
2008-06-20 20:17:23 +00:00
} else {
$options = array ( 'validate' => $validate , 'fieldList' => array ( $name ));
}
2008-07-05 11:02:09 +00:00
return $this -> save ( array ( $this -> alias => array ( $this -> primaryKey => $id , $name => $value )), $options );
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-11-08 02:54:07 +00:00
* Saves model data ( based on white - list , if supplied ) to the database . By
2008-10-31 19:05:30 +00:00
* default , validation occurs before save .
2008-05-30 11:40:08 +00:00
*
* @ param array $data Data to save .
2008-08-23 21:31:38 +00:00
* @ param mixed $validate Either a boolean , or an array .
2009-09-03 11:59:57 -04:00
* If a boolean , indicates whether or not to validate before saving .
* If an array , allows control of validate , callbacks , and fieldList
2008-05-30 11:40:08 +00:00
* @ param array $fieldList List of fields to allow to be written
* @ return mixed On success Model :: $data if its not empty or true , false on failure
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function save ( $data = null , $validate = true , $fieldList = array ()) {
$defaults = array ( 'validate' => true , 'fieldList' => array (), 'callbacks' => true );
2008-09-04 01:00:08 +00:00
$_whitelist = $this -> whitelist ;
2008-05-30 11:40:08 +00:00
$fields = array ();
if ( ! is_array ( $validate )) {
$options = array_merge ( $defaults , compact ( 'validate' , 'fieldList' , 'callbacks' ));
} else {
$options = array_merge ( $defaults , $validate );
}
if ( ! empty ( $options [ 'fieldList' ])) {
$this -> whitelist = $options [ 'fieldList' ];
} elseif ( $options [ 'fieldList' ] === null ) {
$this -> whitelist = array ();
}
$this -> set ( $data );
if ( empty ( $this -> data ) && ! $this -> hasField ( array ( 'created' , 'updated' , 'modified' ))) {
return false ;
}
foreach ( array ( 'created' , 'updated' , 'modified' ) as $field ) {
2008-09-04 01:00:08 +00:00
$keyPresentAndEmpty = (
isset ( $this -> data [ $this -> alias ]) &&
array_key_exists ( $field , $this -> data [ $this -> alias ]) &&
$this -> data [ $this -> alias ][ $field ] === null
);
if ( $keyPresentAndEmpty ) {
2008-05-30 11:40:08 +00:00
unset ( $this -> data [ $this -> alias ][ $field ]);
}
}
2008-06-15 02:58:23 +00:00
2010-01-14 13:47:38 -05:00
$exists = $this -> exists ();
2008-05-30 11:40:08 +00:00
$dateFields = array ( 'modified' , 'updated' );
2010-01-14 13:47:38 -05:00
if ( ! $exists ) {
2008-05-30 11:40:08 +00:00
$dateFields [] = 'created' ;
}
if ( isset ( $this -> data [ $this -> alias ])) {
$fields = array_keys ( $this -> data [ $this -> alias ]);
}
if ( $options [ 'validate' ] && ! $this -> validates ( $options )) {
$this -> whitelist = $_whitelist ;
return false ;
}
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-09-04 01:00:08 +00:00
2008-05-30 11:40:08 +00:00
foreach ( $dateFields as $updateCol ) {
if ( $this -> hasField ( $updateCol ) && ! in_array ( $updateCol , $fields )) {
2008-12-17 04:13:45 +00:00
$default = array ( 'formatter' => 'date' );
$colType = array_merge ( $default , $db -> columns [ $this -> getColumnType ( $updateCol )]);
if ( ! array_key_exists ( 'format' , $colType )) {
2008-05-30 11:40:08 +00:00
$time = strtotime ( 'now' );
} else {
$time = $colType [ 'formatter' ]( $colType [ 'format' ]);
}
if ( ! empty ( $this -> whitelist )) {
$this -> whitelist [] = $updateCol ;
}
$this -> set ( $updateCol , $time );
}
}
2008-06-20 20:17:23 +00:00
if ( $options [ 'callbacks' ] === true || $options [ 'callbacks' ] === 'before' ) {
2010-12-12 15:03:23 -05:00
$result = $this -> Behaviors -> trigger ( 'beforeSave' , array ( & $this , $options ), array (
2010-12-12 17:40:13 -05:00
'break' => true , 'breakOn' => array ( false , null )
2008-12-17 04:13:45 +00:00
));
if ( ! $result || ! $this -> beforeSave ( $options )) {
2008-05-30 11:40:08 +00:00
$this -> whitelist = $_whitelist ;
return false ;
}
}
2010-05-11 22:40:56 -04:00
if ( empty ( $this -> data [ $this -> alias ][ $this -> primaryKey ])) {
2008-05-30 11:40:08 +00:00
unset ( $this -> data [ $this -> alias ][ $this -> primaryKey ]);
}
2010-01-14 13:47:38 -05:00
$fields = $values = array ();
2008-05-30 11:40:08 +00:00
foreach ( $this -> data as $n => $v ) {
if ( isset ( $this -> hasAndBelongsToMany [ $n ])) {
if ( isset ( $v [ $n ])) {
$v = $v [ $n ];
}
$joined [ $n ] = $v ;
} else {
if ( $n === $this -> alias ) {
foreach ( array ( 'created' , 'updated' , 'modified' ) as $field ) {
if ( array_key_exists ( $field , $v ) && empty ( $v [ $field ])) {
unset ( $v [ $field ]);
}
}
foreach ( $v as $x => $y ) {
if ( $this -> hasField ( $x ) && ( empty ( $this -> whitelist ) || in_array ( $x , $this -> whitelist ))) {
list ( $fields [], $values []) = array ( $x , $y );
}
}
}
}
}
$count = count ( $fields );
2010-01-14 13:47:38 -05:00
if ( ! $exists && $count > 0 ) {
2008-05-30 11:40:08 +00:00
$this -> id = false ;
}
$success = true ;
$created = false ;
if ( $count > 0 ) {
2008-12-17 04:13:45 +00:00
$cache = $this -> _prepareUpdateFields ( array_combine ( $fields , $values ));
2008-05-30 11:40:08 +00:00
if ( ! empty ( $this -> id )) {
2008-12-17 04:13:45 +00:00
$success = ( bool ) $db -> update ( $this , $fields , $values );
2008-05-30 11:40:08 +00:00
} else {
2010-07-15 21:19:23 -04:30
$fInfo = $this -> schema ( $this -> primaryKey );
2010-03-15 23:36:20 -04:00
$isUUID = ( $fInfo [ 'length' ] == 36 &&
( $fInfo [ 'type' ] === 'string' || $fInfo [ 'type' ] === 'binary' )
);
if ( empty ( $this -> data [ $this -> alias ][ $this -> primaryKey ]) && $isUUID ) {
if ( array_key_exists ( $this -> primaryKey , $this -> data [ $this -> alias ])) {
$j = array_search ( $this -> primaryKey , $fields );
$values [ $j ] = String :: uuid ();
} else {
list ( $fields [], $values []) = array ( $this -> primaryKey , String :: uuid ());
2008-05-30 11:40:08 +00:00
}
}
if ( ! $db -> create ( $this , $fields , $values )) {
$success = $created = false ;
} else {
$created = true ;
}
}
2008-12-17 04:13:45 +00:00
if ( $success && ! empty ( $this -> belongsTo )) {
$this -> updateCounterCache ( $cache , $created );
}
2008-05-30 11:40:08 +00:00
}
if ( ! empty ( $joined ) && $success === true ) {
2010-01-14 12:57:43 -05:00
$this -> __saveMulti ( $joined , $this -> id , $db );
2008-05-30 11:40:08 +00:00
}
if ( $success && $count > 0 ) {
if ( ! empty ( $this -> data )) {
$success = $this -> data ;
}
2008-06-20 20:17:23 +00:00
if ( $options [ 'callbacks' ] === true || $options [ 'callbacks' ] === 'after' ) {
2010-12-12 15:03:23 -05:00
$this -> Behaviors -> trigger ( 'afterSave' , array ( & $this , $created , $options ));
2008-05-30 11:40:08 +00:00
$this -> afterSave ( $created );
}
if ( ! empty ( $this -> data )) {
$success = Set :: merge ( $success , $this -> data );
}
$this -> data = false ;
$this -> _clearCache ();
$this -> validationErrors = array ();
}
$this -> whitelist = $_whitelist ;
return $success ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Saves model hasAndBelongsToMany data to the database .
*
* @ param array $joined Data to save
* @ param mixed $id ID of record in this model
* @ access private
*/
2010-01-14 12:57:43 -05:00
function __saveMulti ( $joined , $id , & $db ) {
2008-10-29 07:19:26 +00:00
foreach ( $joined as $assoc => $data ) {
2008-05-30 11:40:08 +00:00
if ( isset ( $this -> hasAndBelongsToMany [ $assoc ])) {
list ( $join ) = $this -> joinModel ( $this -> hasAndBelongsToMany [ $assoc ][ 'with' ]);
2008-10-29 07:19:26 +00:00
2010-07-15 21:19:23 -04:30
$keyInfo = $this -> { $join } -> schema ( $this -> { $join } -> primaryKey );
2009-03-04 01:36:53 +00:00
$isUUID = ! empty ( $this -> { $join } -> primaryKey ) && (
2010-07-15 21:19:23 -04:30
$keyInfo [ 'length' ] == 36 && (
$keyInfo [ 'type' ] === 'string' ||
$keyInfo [ 'type' ] === 'binary'
2009-03-04 01:36:53 +00:00
)
);
2008-10-23 13:29:32 +00:00
2008-10-29 07:19:26 +00:00
$newData = $newValues = array ();
2008-12-30 16:09:25 +00:00
$primaryAdded = false ;
$fields = array (
$db -> name ( $this -> hasAndBelongsToMany [ $assoc ][ 'foreignKey' ]),
$db -> name ( $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ])
);
$idField = $db -> name ( $this -> { $join } -> primaryKey );
if ( $isUUID && ! in_array ( $idField , $fields )) {
$fields [] = $idField ;
$primaryAdded = true ;
}
2008-10-29 07:19:26 +00:00
2008-10-30 00:11:32 +00:00
foreach (( array ) $data as $row ) {
2008-11-09 20:55:51 +00:00
if (( is_string ( $row ) && ( strlen ( $row ) == 36 || strlen ( $row ) == 16 )) || is_numeric ( $row )) {
2010-10-24 19:59:54 -04:30
$values = array ( $id , $row );
2008-12-30 16:09:25 +00:00
if ( $isUUID && $primaryAdded ) {
2010-10-24 19:59:54 -04:30
$values [] = String :: uuid ();
2008-05-30 11:40:08 +00:00
}
2010-10-24 19:59:54 -04:30
$newValues [] = $values ;
2008-10-29 07:19:26 +00:00
unset ( $values );
2008-11-09 20:55:51 +00:00
} elseif ( isset ( $row [ $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ]])) {
2008-10-29 07:19:26 +00:00
$newData [] = $row ;
2009-07-23 19:28:46 +00:00
} elseif ( isset ( $row [ $join ]) && isset ( $row [ $join ][ $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ]])) {
2009-02-11 03:53:04 +00:00
$newData [] = $row [ $join ];
2008-10-29 07:19:26 +00:00
}
}
2009-01-16 04:09:44 +00:00
if ( $this -> hasAndBelongsToMany [ $assoc ][ 'unique' ]) {
2010-03-15 23:14:23 -04:00
$conditions = array (
$join . '.' . $this -> hasAndBelongsToMany [ $assoc ][ 'foreignKey' ] => $id
2009-11-22 17:56:46 -05:00
);
2010-03-15 23:14:23 -04:00
if ( ! empty ( $this -> hasAndBelongsToMany [ $assoc ][ 'conditions' ])) {
$conditions = array_merge ( $conditions , ( array ) $this -> hasAndBelongsToMany [ $assoc ][ 'conditions' ]);
}
2009-11-22 17:56:46 -05:00
$links = $this -> { $join } -> find ( 'all' , array (
'conditions' => $conditions ,
'recursive' => empty ( $this -> hasAndBelongsToMany [ $assoc ][ 'conditions' ]) ? - 1 : 0 ,
'fields' => $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ]
));
2008-10-29 07:19:26 +00:00
$associationForeignKey = " { $join } . " . $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ];
$oldLinks = Set :: extract ( $links , " { n}. { $associationForeignKey } " );
if ( ! empty ( $oldLinks )) {
$conditions [ $associationForeignKey ] = $oldLinks ;
$db -> delete ( $this -> { $join }, $conditions );
2008-05-30 11:40:08 +00:00
}
}
2009-01-16 04:09:44 +00:00
if ( ! empty ( $newData )) {
foreach ( $newData as $data ) {
$data [ $this -> hasAndBelongsToMany [ $assoc ][ 'foreignKey' ]] = $id ;
$this -> { $join } -> create ( $data );
$this -> { $join } -> save ();
}
}
2008-05-30 11:40:08 +00:00
if ( ! empty ( $newValues )) {
$db -> insertMulti ( $this -> { $join }, $fields , $newValues );
}
}
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Updates the counter cache of belongsTo associations after a save or delete operation
*
* @ param array $keys Optional foreign key data , defaults to the information $this -> data
* @ param boolean $created True if a new record was created , otherwise only associations with
2009-10-05 21:27:34 -04:00
* 'counterScope' defined get updated
2008-05-30 11:40:08 +00:00
* @ return void
*/
2010-04-05 13:19:38 +10:00
public function updateCounterCache ( $keys = array (), $created = false ) {
2008-12-17 04:13:45 +00:00
$keys = empty ( $keys ) ? $this -> data [ $this -> alias ] : $keys ;
$keys [ 'old' ] = isset ( $keys [ 'old' ]) ? $keys [ 'old' ] : array ();
2008-05-30 11:40:08 +00:00
foreach ( $this -> belongsTo as $parent => $assoc ) {
2008-12-17 04:13:45 +00:00
$foreignKey = $assoc [ 'foreignKey' ];
$fkQuoted = $this -> escapeField ( $assoc [ 'foreignKey' ]);
2008-05-30 11:40:08 +00:00
if ( ! empty ( $assoc [ 'counterCache' ])) {
if ( $assoc [ 'counterCache' ] === true ) {
$assoc [ 'counterCache' ] = Inflector :: underscore ( $this -> alias ) . '_count' ;
}
2008-12-17 04:13:45 +00:00
if ( ! $this -> { $parent } -> hasField ( $assoc [ 'counterCache' ])) {
continue ;
2008-05-30 11:40:08 +00:00
}
2008-12-17 04:13:45 +00:00
if ( ! array_key_exists ( $foreignKey , $keys )) {
$keys [ $foreignKey ] = $this -> field ( $foreignKey );
}
$recursive = ( isset ( $assoc [ 'counterScope' ]) ? 1 : - 1 );
$conditions = ( $recursive == 1 ) ? ( array ) $assoc [ 'counterScope' ] : array ();
if ( isset ( $keys [ 'old' ][ $foreignKey ])) {
2009-04-25 01:17:21 +00:00
if ( $keys [ 'old' ][ $foreignKey ] != $keys [ $foreignKey ]) {
$conditions [ $fkQuoted ] = $keys [ 'old' ][ $foreignKey ];
$count = intval ( $this -> find ( 'count' , compact ( 'conditions' , 'recursive' )));
2009-05-01 14:05:46 -07:00
2009-04-25 01:17:21 +00:00
$this -> { $parent } -> updateAll (
array ( $assoc [ 'counterCache' ] => $count ),
array ( $this -> { $parent } -> escapeField () => $keys [ 'old' ][ $foreignKey ])
);
2008-05-30 11:40:08 +00:00
}
}
2008-12-17 04:13:45 +00:00
$conditions [ $fkQuoted ] = $keys [ $foreignKey ];
if ( $recursive == 1 ) {
$conditions = array_merge ( $conditions , ( array ) $assoc [ 'counterScope' ]);
}
$count = intval ( $this -> find ( 'count' , compact ( 'conditions' , 'recursive' )));
$this -> { $parent } -> updateAll (
array ( $assoc [ 'counterCache' ] => $count ),
array ( $this -> { $parent } -> escapeField () => $keys [ $foreignKey ])
);
}
}
}
2009-07-24 21:18:37 +02:00
2008-12-17 04:13:45 +00:00
/**
* Helper method for Model :: updateCounterCache () . Checks the fields to be updated for
*
* @ param array $data The fields of the record that will be updated
* @ return array Returns updated foreign key values , along with an 'old' key containing the old
2010-03-05 22:07:39 -05:00
* values , or empty if no foreign keys are updated .
2008-12-17 04:13:45 +00:00
*/
2010-04-05 13:21:28 +10:00
protected function _prepareUpdateFields ( $data ) {
2008-12-17 04:13:45 +00:00
$foreignKeys = array ();
foreach ( $this -> belongsTo as $assoc => $info ) {
if ( $info [ 'counterCache' ]) {
$foreignKeys [ $assoc ] = $info [ 'foreignKey' ];
2008-05-30 11:40:08 +00:00
}
}
2008-12-17 04:13:45 +00:00
$included = array_intersect ( $foreignKeys , array_keys ( $data ));
if ( empty ( $included ) || empty ( $this -> id )) {
return array ();
}
$old = $this -> find ( 'first' , array (
2009-04-13 14:15:58 +00:00
'conditions' => array ( $this -> primaryKey => $this -> id ),
2008-12-17 04:13:45 +00:00
'fields' => array_values ( $included ),
'recursive' => - 1
));
return array_merge ( $data , array ( 'old' => $old [ $this -> alias ]));
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Saves multiple individual records for a single model ; Also works with a single record , as well as
* all its associated records .
2008-05-30 11:40:08 +00:00
*
2009-09-03 11:59:57 -04:00
* #### Options
*
2010-03-28 05:29:42 +05:30
* - validate : Set to false to disable validation , true to validate each record before saving ,
* 'first' to validate * all * records before any are saved ( default ),
* or 'only' to only validate the records , but not save them .
2009-09-03 11:59:57 -04:00
* - atomic : If true ( default ), will attempt to save all records in a single transaction .
* Should be set to false if database / table does not support transactions .
* - fieldList : Equivalent to the $fieldList parameter in Model :: save ()
*
2008-05-30 11:40:08 +00:00
* @ param array $data Record data to save . This can be either a numerically - indexed array ( for saving multiple
2009-09-03 11:59:57 -04:00
* records of the same type ), or an array indexed by association name .
* @ param array $options Options to use when saving record data , See $options above .
2010-02-23 16:05:30 +01:00
* @ return mixed If atomic : True on success , or false on failure .
2010-03-05 22:07:39 -05:00
* Otherwise : array similar to the $data array passed , but values are set to true / false
* depending on whether each record saved successfully .
2008-05-30 11:40:08 +00:00
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1032 / Saving - Related - Model - Data - hasOne - hasMany - belongsTo
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function saveAll ( $data = null , $options = array ()) {
if ( empty ( $data )) {
$data = $this -> data ;
}
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-05-30 11:40:08 +00:00
2010-03-28 05:29:42 +05:30
$options = array_merge ( array ( 'validate' => 'first' , 'atomic' => true ), $options );
2008-05-30 11:40:08 +00:00
$this -> validationErrors = $validationErrors = array ();
$validates = true ;
$return = array ();
2010-03-17 15:19:30 -04:30
if ( empty ( $data ) && $options [ 'validate' ] !== false ) {
$result = $this -> save ( $data , $options );
return ! empty ( $result );
}
2008-05-30 11:40:08 +00:00
if ( $options [ 'atomic' ] && $options [ 'validate' ] !== 'only' ) {
2010-11-03 18:55:42 +05:30
$transactionBegun = $db -> begin ( $this );
2008-05-30 11:40:08 +00:00
}
if ( Set :: numeric ( array_keys ( $data ))) {
while ( $validates ) {
2010-08-26 22:21:39 -04:00
$return = array ();
2008-05-31 22:01:47 +00:00
foreach ( $data as $key => $record ) {
2009-06-27 15:25:07 +00:00
if ( ! $currentValidates = $this -> __save ( $record , $options )) {
2008-08-26 13:31:07 +00:00
$validationErrors [ $key ] = $this -> validationErrors ;
2008-05-30 11:40:08 +00:00
}
2008-09-19 15:27:43 +00:00
2008-09-03 16:44:48 +00:00
if ( $options [ 'validate' ] === 'only' || $options [ 'validate' ] === 'first' ) {
$validating = true ;
if ( $options [ 'atomic' ]) {
$validates = $validates && $currentValidates ;
} else {
$validates = $currentValidates ;
}
} else {
$validating = false ;
$validates = $currentValidates ;
}
2008-09-19 15:27:43 +00:00
2008-05-30 11:40:08 +00:00
if ( ! $options [ 'atomic' ]) {
$return [] = $validates ;
} elseif ( ! $validates && ! $validating ) {
break ;
}
}
$this -> validationErrors = $validationErrors ;
switch ( true ) {
case ( $options [ 'validate' ] === 'only' ) :
2008-06-27 06:07:00 +00:00
return ( $options [ 'atomic' ] ? $validates : $return );
2008-05-30 11:40:08 +00:00
break ;
case ( $options [ 'validate' ] === 'first' ) :
$options [ 'validate' ] = true ;
break ;
default :
if ( $options [ 'atomic' ]) {
2010-11-03 18:55:42 +05:30
if ( $validates ) {
if ( $transactionBegun ) {
return $db -> commit ( $this ) !== false ;
} else {
return true ;
}
2008-05-30 11:40:08 +00:00
}
$db -> rollback ( $this );
return false ;
}
return $return ;
break ;
}
}
2010-06-09 13:48:54 -04:00
if ( $options [ 'atomic' ] && ! $validates ) {
$db -> rollback ( $this );
return false ;
}
2008-05-30 11:40:08 +00:00
return $return ;
}
$associations = $this -> getAssociated ();
while ( $validates ) {
foreach ( $data as $association => $values ) {
if ( isset ( $associations [ $association ])) {
switch ( $associations [ $association ]) {
case 'belongsTo' :
2009-06-27 15:25:07 +00:00
if ( $this -> { $association } -> __save ( $values , $options )) {
2008-05-30 11:40:08 +00:00
$data [ $this -> alias ][ $this -> belongsTo [ $association ][ 'foreignKey' ]] = $this -> { $association } -> id ;
} else {
$validationErrors [ $association ] = $this -> { $association } -> validationErrors ;
$validates = false ;
}
if ( ! $options [ 'atomic' ]) {
$return [ $association ][] = $validates ;
}
break ;
}
}
}
2010-09-28 23:37:28 -04:00
2009-06-27 15:25:07 +00:00
if ( ! $this -> __save ( $data , $options )) {
2008-05-30 11:40:08 +00:00
$validationErrors [ $this -> alias ] = $this -> validationErrors ;
$validates = false ;
}
if ( ! $options [ 'atomic' ]) {
2008-06-27 06:07:00 +00:00
$return [ $this -> alias ] = $validates ;
2008-05-30 11:40:08 +00:00
}
$validating = ( $options [ 'validate' ] === 'only' || $options [ 'validate' ] === 'first' );
foreach ( $data as $association => $values ) {
if ( ! $validates && ! $validating ) {
break ;
}
if ( isset ( $associations [ $association ])) {
$type = $associations [ $association ];
switch ( $type ) {
case 'hasOne' :
$values [ $this -> { $type }[ $association ][ 'foreignKey' ]] = $this -> id ;
2009-06-27 15:25:07 +00:00
if ( ! $this -> { $association } -> __save ( $values , $options )) {
2008-05-30 11:40:08 +00:00
$validationErrors [ $association ] = $this -> { $association } -> validationErrors ;
$validates = false ;
}
if ( ! $options [ 'atomic' ]) {
$return [ $association ][] = $validates ;
}
break ;
case 'hasMany' :
foreach ( $values as $i => $value ) {
$values [ $i ][ $this -> { $type }[ $association ][ 'foreignKey' ]] = $this -> id ;
}
$_options = array_merge ( $options , array ( 'atomic' => false ));
if ( $_options [ 'validate' ] === 'first' ) {
$_options [ 'validate' ] = 'only' ;
}
$_return = $this -> { $association } -> saveAll ( $values , $_options );
if ( $_return === false || ( is_array ( $_return ) && in_array ( false , $_return , true ))) {
$validationErrors [ $association ] = $this -> { $association } -> validationErrors ;
$validates = false ;
}
if ( is_array ( $_return )) {
foreach ( $_return as $val ) {
if ( ! isset ( $return [ $association ])) {
$return [ $association ] = array ();
} elseif ( ! is_array ( $return [ $association ])) {
$return [ $association ] = array ( $return [ $association ]);
}
$return [ $association ][] = $val ;
}
} else {
2008-06-20 20:17:23 +00:00
$return [ $association ] = $_return ;
2008-05-30 11:40:08 +00:00
}
break ;
}
}
}
$this -> validationErrors = $validationErrors ;
2008-06-27 06:07:00 +00:00
2008-05-30 11:40:08 +00:00
if ( isset ( $validationErrors [ $this -> alias ])) {
$this -> validationErrors = $validationErrors [ $this -> alias ];
}
switch ( true ) {
case ( $options [ 'validate' ] === 'only' ) :
2008-06-27 06:07:00 +00:00
return ( $options [ 'atomic' ] ? $validates : $return );
2008-05-30 11:40:08 +00:00
break ;
case ( $options [ 'validate' ] === 'first' ) :
$options [ 'validate' ] = true ;
2010-03-28 05:29:42 +05:30
$return = array ();
2008-05-30 11:40:08 +00:00
break ;
default :
if ( $options [ 'atomic' ]) {
if ( $validates ) {
2010-11-03 18:55:42 +05:30
if ( $transactionBegun ) {
return $db -> commit ( $this ) !== false ;
} else {
return true ;
}
2008-05-30 11:40:08 +00:00
} else {
$db -> rollback ( $this );
}
}
return $return ;
break ;
}
2010-09-28 23:37:28 -04:00
if ( $options [ 'atomic' ] && ! $validates ) {
$db -> rollback ( $this );
return false ;
}
2008-05-30 11:40:08 +00:00
}
2010-06-01 00:19:11 -04:30
return $return ;
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Private helper method used by saveAll .
2008-05-30 11:40:08 +00:00
*
2008-09-25 16:49:56 +00:00
* @ return boolean Success
2008-05-30 11:40:08 +00:00
* @ access private
* @ see Model :: saveAll ()
*/
2009-06-27 15:25:07 +00:00
function __save ( $data , $options ) {
2008-05-30 11:40:08 +00:00
if ( $options [ 'validate' ] === 'first' || $options [ 'validate' ] === 'only' ) {
2009-06-27 15:25:07 +00:00
if ( ! ( $this -> create ( $data ) && $this -> validates ( $options ))) {
2008-05-30 11:40:08 +00:00
return false ;
}
2009-06-27 15:25:07 +00:00
} elseif ( ! ( $this -> create ( null ) !== null && $this -> save ( $data , $options ))) {
2008-05-30 11:40:08 +00:00
return false ;
}
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Updates multiple model records based on a set of conditions .
2008-05-30 11:40:08 +00:00
*
2008-07-31 15:38:23 +00:00
* @ param array $fields Set of fields and values , indexed by fields .
2009-09-03 11:59:57 -04:00
* Fields are treated as SQL snippets , to insert literal values manually escape your data .
2008-05-30 11:40:08 +00:00
* @ param mixed $conditions Conditions to match , true for all records
* @ return boolean True on success , false on failure
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1031 / Saving - Your - Data
2008-05-30 11:40:08 +00:00
*/
function updateAll ( $fields , $conditions = true ) {
2010-07-14 23:19:38 -04:30
return $this -> getDataSource () -> update ( $this , $fields , null , $conditions );
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Removes record for given ID . If no ID is given , the current ID is used . Returns true on success .
2008-05-30 11:40:08 +00:00
*
* @ param mixed $id ID of record to delete
* @ param boolean $cascade Set to true to delete records that depend on this record
* @ return boolean True on success
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1036 / delete
2008-05-30 11:40:08 +00:00
*/
2009-05-01 14:05:46 -07:00
function delete ( $id = null , $cascade = true ) {
2008-05-30 11:40:08 +00:00
if ( ! empty ( $id )) {
$this -> id = $id ;
}
$id = $this -> id ;
2010-09-29 23:31:41 -04:00
if ( $this -> beforeDelete ( $cascade )) {
2010-12-12 15:03:23 -05:00
$filters = $this -> Behaviors -> trigger (
'beforeDelete' ,
array ( & $this , $cascade ),
2010-12-12 17:40:13 -05:00
array ( 'break' => true , 'breakOn' => array ( false , null ))
2010-12-12 15:03:23 -05:00
);
2010-09-29 23:31:41 -04:00
if ( ! $filters || ! $this -> exists ()) {
2008-05-30 11:40:08 +00:00
return false ;
}
2010-12-12 15:03:23 -05:00
$db = ConnectionManager :: getDataSource ( $this -> useDbConfig );
2010-09-29 23:31:41 -04:00
2008-05-30 11:40:08 +00:00
$this -> _deleteDependent ( $id , $cascade );
$this -> _deleteLinks ( $id );
$this -> id = $id ;
if ( ! empty ( $this -> belongsTo )) {
2010-03-13 00:18:51 -05:00
$keys = $this -> find ( 'first' , array (
'fields' => $this -> __collectForeignKeys (),
'conditions' => array ( $this -> alias . '.' . $this -> primaryKey => $id )
));
2008-05-30 11:40:08 +00:00
}
2010-11-08 21:50:45 -05:00
if ( $db -> delete ( $this , array ( $this -> alias . '.' . $this -> primaryKey => $id ))) {
2008-05-30 11:40:08 +00:00
if ( ! empty ( $this -> belongsTo )) {
$this -> updateCounterCache ( $keys [ $this -> alias ]);
}
2010-12-12 17:40:13 -05:00
$this -> Behaviors -> trigger ( 'afterDelete' , array ( & $this ));
2008-05-30 11:40:08 +00:00
$this -> afterDelete ();
$this -> _clearCache ();
$this -> id = false ;
return true ;
}
}
return false ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Cascades model deletes through associated hasMany and hasOne child records .
2008-05-30 11:40:08 +00:00
*
* @ param string $id ID of record that was deleted
* @ param boolean $cascade Set to true to delete records that depend on this record
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
*/
2010-04-05 13:21:28 +10:00
protected function _deleteDependent ( $id , $cascade ) {
2008-05-30 11:40:08 +00:00
if ( ! empty ( $this -> __backAssociation )) {
$savedAssociatons = $this -> __backAssociation ;
$this -> __backAssociation = array ();
}
foreach ( array_merge ( $this -> hasMany , $this -> hasOne ) as $assoc => $data ) {
if ( $data [ 'dependent' ] === true && $cascade === true ) {
2010-07-15 22:16:19 -04:30
$model = $this -> { $assoc };
2008-06-16 03:11:32 +00:00
$conditions = array ( $model -> escapeField ( $data [ 'foreignKey' ]) => $id );
if ( $data [ 'conditions' ]) {
2010-03-18 02:51:20 +05:30
$conditions = array_merge (( array ) $data [ 'conditions' ], $conditions );
2008-06-16 03:11:32 +00:00
}
2008-05-30 11:40:08 +00:00
$model -> recursive = - 1 ;
if ( isset ( $data [ 'exclusive' ]) && $data [ 'exclusive' ]) {
2008-06-16 03:11:32 +00:00
$model -> deleteAll ( $conditions );
2008-05-30 11:40:08 +00:00
} else {
2009-05-01 14:05:46 -07:00
$records = $model -> find ( 'all' , array (
'conditions' => $conditions , 'fields' => $model -> primaryKey
));
2008-05-30 11:40:08 +00:00
if ( ! empty ( $records )) {
foreach ( $records as $record ) {
$model -> delete ( $record [ $model -> alias ][ $model -> primaryKey ]);
}
}
}
}
}
if ( isset ( $savedAssociatons )) {
$this -> __backAssociation = $savedAssociatons ;
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Cascades model deletes through HABTM join keys .
2008-05-30 11:40:08 +00:00
*
* @ param string $id ID of record that was deleted
2008-09-25 16:49:56 +00:00
* @ return void
2008-05-30 11:40:08 +00:00
*/
2010-04-05 13:21:28 +10:00
protected function _deleteLinks ( $id ) {
2008-05-30 11:40:08 +00:00
foreach ( $this -> hasAndBelongsToMany as $assoc => $data ) {
2010-01-29 11:55:18 -05:00
$joinModel = $data [ 'with' ];
$records = $this -> { $joinModel } -> find ( 'all' , array (
'conditions' => array_merge ( array ( $this -> { $joinModel } -> escapeField ( $data [ 'foreignKey' ]) => $id )),
'fields' => $this -> { $joinModel } -> primaryKey ,
2008-05-30 11:40:08 +00:00
'recursive' => - 1
));
if ( ! empty ( $records )) {
foreach ( $records as $record ) {
2010-01-29 11:55:18 -05:00
$this -> { $joinModel } -> delete ( $record [ $this -> { $joinModel } -> alias ][ $this -> { $joinModel } -> primaryKey ]);
2008-05-30 11:40:08 +00:00
}
}
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Deletes multiple model records based on a set of conditions .
2008-05-30 11:40:08 +00:00
*
* @ param mixed $conditions Conditions to match
* @ param boolean $cascade Set to true to delete records that depend on this record
2010-04-16 21:45:44 +10:00
* @ param boolean $callbacks Run callbacks
2008-05-30 11:40:08 +00:00
* @ return boolean True on success , false on failure
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1038 / deleteAll
2008-05-30 11:40:08 +00:00
*/
function deleteAll ( $conditions , $cascade = true , $callbacks = false ) {
if ( empty ( $conditions )) {
return false ;
}
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-05-30 11:40:08 +00:00
if ( ! $cascade && ! $callbacks ) {
return $db -> delete ( $this , $conditions );
} else {
2010-04-25 02:34:18 +05:30
$ids = $this -> find ( 'all' , array_merge ( array (
'fields' => " { $this -> alias } . { $this -> primaryKey } " ,
'recursive' => 0 ), compact ( 'conditions' ))
2008-05-30 11:40:08 +00:00
);
2010-04-25 02:34:18 +05:30
if ( $ids === false ) {
return false ;
}
2008-05-30 11:40:08 +00:00
2010-04-25 02:34:18 +05:30
$ids = Set :: extract ( $ids , " { n}. { $this -> alias } . { $this -> primaryKey } " );
2008-05-30 11:40:08 +00:00
if ( empty ( $ids )) {
2009-07-03 00:26:58 +00:00
return true ;
2008-05-30 11:40:08 +00:00
}
if ( $callbacks ) {
$_id = $this -> id ;
2008-10-22 02:37:05 +00:00
$result = true ;
2008-05-30 11:40:08 +00:00
foreach ( $ids as $id ) {
2008-10-22 02:37:05 +00:00
$result = ( $result && $this -> delete ( $id , $cascade ));
2008-05-30 11:40:08 +00:00
}
$this -> id = $_id ;
2008-10-22 02:37:05 +00:00
return $result ;
2008-05-30 11:40:08 +00:00
} else {
foreach ( $ids as $id ) {
$this -> _deleteLinks ( $id );
if ( $cascade ) {
$this -> _deleteDependent ( $id , $cascade );
}
}
return $db -> delete ( $this , array ( $this -> alias . '.' . $this -> primaryKey => $ids ));
}
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Collects foreign keys from associations .
2008-05-30 11:40:08 +00:00
*
2008-09-25 16:49:56 +00:00
* @ return array
2008-05-30 11:40:08 +00:00
* @ access private
*/
function __collectForeignKeys ( $type = 'belongsTo' ) {
$result = array ();
foreach ( $this -> { $type } as $assoc => $data ) {
if ( isset ( $data [ 'foreignKey' ]) && is_string ( $data [ 'foreignKey' ])) {
$result [ $assoc ] = $data [ 'foreignKey' ];
}
}
return $result ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns true if a record with the currently set ID exists .
2008-05-30 11:40:08 +00:00
*
2010-01-14 16:42:16 -05:00
* Internally calls Model :: getID () to obtain the current record ID to verify ,
* and then performs a Model :: find ( 'count' ) on the currently configured datasource
* to ascertain the existence of the record in persistent storage .
*
2008-05-30 11:40:08 +00:00
* @ return boolean True if such a record exists
*/
2010-04-05 13:19:38 +10:00
public function exists () {
2010-01-14 16:42:16 -05:00
if ( $this -> getID () === false ) {
2008-05-30 11:40:08 +00:00
return false ;
}
2008-06-20 20:17:23 +00:00
$conditions = array ( $this -> alias . '.' . $this -> primaryKey => $this -> getID ());
2008-06-21 02:33:04 +00:00
$query = array ( 'conditions' => $conditions , 'recursive' => - 1 , 'callbacks' => false );
2010-01-14 13:47:38 -05:00
return ( $this -> find ( 'count' , $query ) > 0 );
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns true if a record that meets given conditions exists .
2008-05-30 11:40:08 +00:00
*
* @ param array $conditions SQL conditions array
* @ return boolean True if such a record exists
*/
2010-04-05 13:19:38 +10:00
public function hasAny ( $conditions = null ) {
2008-05-30 11:40:08 +00:00
return ( $this -> find ( 'count' , array ( 'conditions' => $conditions , 'recursive' => - 1 )) != false );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2010-07-27 21:25:31 -04:00
* Queries the datasource and returns a result set array .
2008-05-30 11:40:08 +00:00
*
* Also used to perform new - notation finds , where the first argument is type of find operation to perform
2008-10-03 21:23:13 +00:00
* ( all / first / count / neighbors / list / threaded ),
2008-10-03 18:06:15 +00:00
* second parameter options for finding ( indexed array , including : 'conditions' , 'limit' ,
2008-05-30 11:40:08 +00:00
* 'recursive' , 'page' , 'fields' , 'offset' , 'order' )
*
2009-11-11 23:47:23 -04:30
* Eg :
2009-10-05 21:01:31 -04:00
* {{{
2010-07-27 21:25:31 -04:00
* find ( 'all' , array (
* 'conditions' => array ( 'name' => 'Thomas Anderson' ),
2009-10-05 21:01:31 -04:00
* 'fields' => array ( 'name' , 'email' ),
* 'order' => 'field3 DESC' ,
* 'recursive' => 2 ,
* 'group' => 'type'
* ));
* }}}
2008-05-30 11:40:08 +00:00
*
2010-07-27 21:25:31 -04:00
* In addition to the standard query keys above , you can provide Datasource , and behavior specific
* keys . For example , when using a SQL based datasource you can use the joins key to specify additional
* joins that should be part of the query .
*
* {{{
* find ( 'all' , array (
* 'conditions' => array ( 'name' => 'Thomas Anderson' ),
* 'joins' => array (
2010-09-24 22:23:24 -04:00
* array (
* 'alias' => 'Thought' ,
* 'table' => 'thoughts' ,
* 'type' => 'LEFT' ,
* 'conditions' => '`Thought`.`person_id` = `Person`.`id`'
* )
2010-07-27 21:25:31 -04:00
* )
* ));
* }}}
*
* Behaviors and find types can also define custom finder keys which are passed into find () .
*
2008-05-30 11:40:08 +00:00
* Specifying 'fields' for new - notation 'list' :
2009-10-05 21:01:31 -04:00
*
2008-05-30 11:40:08 +00:00
* - If no fields are specified , then 'id' is used for key and 'model->displayField' is used for value .
* - If a single field is specified , 'id' is used for key and specified field is used for value .
* - If three fields are specified , they are used ( in order ) for key , value and group .
* - Otherwise , first and second fields are used for key and value .
*
2009-05-01 14:05:46 -07:00
* @ param array $conditions SQL conditions array , or type of find operation ( all / first / count /
2010-07-27 21:25:31 -04:00
* neighbors / list / threaded )
2009-05-01 14:05:46 -07:00
* @ param mixed $fields Either a single string of a field name , or an array of field names , or
2010-07-27 21:25:31 -04:00
* options for matching
2008-05-30 11:40:08 +00:00
* @ param string $order SQL ORDER BY conditions ( e . g . " price DESC " or " name ASC " )
* @ param integer $recursive The number of levels deep to fetch associated records
* @ return array Array of records
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1018 / find
2008-05-30 11:40:08 +00:00
*/
function find ( $conditions = null , $fields = array (), $order = null , $recursive = null ) {
2008-09-21 04:09:16 +00:00
if ( ! is_string ( $conditions ) || ( is_string ( $conditions ) && ! array_key_exists ( $conditions , $this -> _findMethods ))) {
2008-05-30 11:40:08 +00:00
$type = 'first' ;
$query = array_merge ( compact ( 'conditions' , 'fields' , 'order' , 'recursive' ), array ( 'limit' => 1 ));
} else {
list ( $type , $query ) = array ( $conditions , $fields );
}
$this -> findQueryType = $type ;
$this -> id = $this -> getID ();
$query = array_merge (
array (
2008-06-21 02:33:04 +00:00
'conditions' => null , 'fields' => null , 'joins' => array (), 'limit' => null ,
'offset' => null , 'order' => null , 'page' => null , 'group' => null , 'callbacks' => true
2008-05-30 11:40:08 +00:00
),
2008-06-19 17:06:50 +00:00
( array ) $query
2008-05-30 11:40:08 +00:00
);
if ( $type != 'all' ) {
2008-09-21 04:09:16 +00:00
if ( $this -> _findMethods [ $type ] === true ) {
2008-05-30 11:40:08 +00:00
$query = $this -> { '_find' . ucfirst ( $type )}( 'before' , $query );
}
}
if ( ! is_numeric ( $query [ 'page' ]) || intval ( $query [ 'page' ]) < 1 ) {
$query [ 'page' ] = 1 ;
}
if ( $query [ 'page' ] > 1 && ! empty ( $query [ 'limit' ])) {
$query [ 'offset' ] = ( $query [ 'page' ] - 1 ) * $query [ 'limit' ];
}
if ( $query [ 'order' ] === null && $this -> order !== null ) {
$query [ 'order' ] = $this -> order ;
}
$query [ 'order' ] = array ( $query [ 'order' ]);
2008-06-21 02:33:04 +00:00
if ( $query [ 'callbacks' ] === true || $query [ 'callbacks' ] === 'before' ) {
2010-12-12 15:03:23 -05:00
$return = $this -> Behaviors -> trigger (
'beforeFind' ,
2010-12-12 17:40:13 -05:00
array ( & $this , $query ),
array ( 'break' => true , 'breakOn' => array ( false , null ), 'modParams' => 1 )
2010-12-12 15:03:23 -05:00
);
2010-12-12 15:48:36 -05:00
2008-07-30 16:26:48 +00:00
$query = ( is_array ( $return )) ? $return : $query ;
2008-05-30 11:40:08 +00:00
2008-06-21 02:33:04 +00:00
if ( $return === false ) {
return null ;
}
2008-05-30 11:40:08 +00:00
2008-06-21 02:33:04 +00:00
$return = $this -> beforeFind ( $query );
2008-07-30 16:26:48 +00:00
$query = ( is_array ( $return )) ? $return : $query ;
2008-05-30 11:40:08 +00:00
2008-06-21 02:33:04 +00:00
if ( $return === false ) {
return null ;
}
2008-05-30 11:40:08 +00:00
}
2010-07-14 23:19:38 -04:30
if ( ! $db = $this -> getDataSource ()) {
2009-09-10 09:28:55 -04:00
return false ;
}
2010-01-19 09:22:13 -05:00
2008-05-30 11:40:08 +00:00
$results = $db -> read ( $this , $query );
$this -> resetAssociations ();
2008-09-04 01:00:08 +00:00
if ( $query [ 'callbacks' ] === true || $query [ 'callbacks' ] === 'after' ) {
$results = $this -> __filterResults ( $results );
}
2009-11-01 15:08:39 +01:00
$this -> findQueryType = null ;
2008-05-30 11:40:08 +00:00
if ( $type === 'all' ) {
2008-06-21 02:33:04 +00:00
return $results ;
2008-05-30 11:40:08 +00:00
} else {
2008-09-21 04:09:16 +00:00
if ( $this -> _findMethods [ $type ] === true ) {
2008-05-30 11:40:08 +00:00
return $this -> { '_find' . ucfirst ( $type )}( 'after' , $query , $results );
}
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Handles the before / after filter logic for find ( 'first' ) operations . Only called by Model :: find () .
*
* @ param string $state Either " before " or " after "
* @ param array $query
* @ param array $data
* @ return array
* @ access protected
2008-10-31 19:05:30 +00:00
* @ see Model :: find ()
2008-05-30 11:40:08 +00:00
*/
function _findFirst ( $state , $query , $results = array ()) {
if ( $state == 'before' ) {
$query [ 'limit' ] = 1 ;
return $query ;
} elseif ( $state == 'after' ) {
if ( empty ( $results [ 0 ])) {
return false ;
}
return $results [ 0 ];
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Handles the before / after filter logic for find ( 'count' ) operations . Only called by Model :: find () .
*
* @ param string $state Either " before " or " after "
* @ param array $query
* @ param array $data
* @ return int The number of records found , or false
* @ access protected
2008-10-31 19:05:30 +00:00
* @ see Model :: find ()
2008-05-30 11:40:08 +00:00
*/
function _findCount ( $state , $query , $results = array ()) {
if ( $state == 'before' ) {
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-05-30 11:40:08 +00:00
if ( empty ( $query [ 'fields' ])) {
$query [ 'fields' ] = $db -> calculate ( $this , 'count' );
2008-08-14 16:52:29 +00:00
} elseif ( is_string ( $query [ 'fields' ]) && ! preg_match ( '/count/i' , $query [ 'fields' ])) {
2008-10-01 15:27:29 +00:00
$query [ 'fields' ] = $db -> calculate ( $this , 'count' , array (
$db -> expression ( $query [ 'fields' ]), 'count'
));
2008-05-30 11:40:08 +00:00
}
$query [ 'order' ] = false ;
return $query ;
} elseif ( $state == 'after' ) {
if ( isset ( $results [ 0 ][ 0 ][ 'count' ])) {
return intval ( $results [ 0 ][ 0 ][ 'count' ]);
} elseif ( isset ( $results [ 0 ][ $this -> alias ][ 'count' ])) {
return intval ( $results [ 0 ][ $this -> alias ][ 'count' ]);
}
return false ;
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Handles the before / after filter logic for find ( 'list' ) operations . Only called by Model :: find () .
*
* @ param string $state Either " before " or " after "
* @ param array $query
* @ param array $data
* @ return array Key / value pairs of primary keys / display field values of all records found
* @ access protected
2008-10-31 19:05:30 +00:00
* @ see Model :: find ()
2008-05-30 11:40:08 +00:00
*/
function _findList ( $state , $query , $results = array ()) {
if ( $state == 'before' ) {
if ( empty ( $query [ 'fields' ])) {
$query [ 'fields' ] = array ( " { $this -> alias } . { $this -> primaryKey } " , " { $this -> alias } . { $this -> displayField } " );
$list = array ( " { n}. { $this -> alias } . { $this -> primaryKey } " , " { n}. { $this -> alias } . { $this -> displayField } " , null );
} else {
if ( ! is_array ( $query [ 'fields' ])) {
$query [ 'fields' ] = String :: tokenize ( $query [ 'fields' ]);
}
2008-06-10 17:27:12 +00:00
2008-05-30 11:40:08 +00:00
if ( count ( $query [ 'fields' ]) == 1 ) {
2008-06-10 17:27:12 +00:00
if ( strpos ( $query [ 'fields' ][ 0 ], '.' ) === false ) {
$query [ 'fields' ][ 0 ] = $this -> alias . '.' . $query [ 'fields' ][ 0 ];
}
2008-05-30 11:40:08 +00:00
$list = array ( " { n}. { $this -> alias } . { $this -> primaryKey } " , '{n}.' . $query [ 'fields' ][ 0 ], null );
$query [ 'fields' ] = array ( " { $this -> alias } . { $this -> primaryKey } " , $query [ 'fields' ][ 0 ]);
} elseif ( count ( $query [ 'fields' ]) == 3 ) {
2008-06-11 08:54:27 +00:00
for ( $i = 0 ; $i < 3 ; $i ++ ) {
2008-06-10 17:27:12 +00:00
if ( strpos ( $query [ 'fields' ][ $i ], '.' ) === false ) {
$query [ 'fields' ][ $i ] = $this -> alias . '.' . $query [ 'fields' ][ $i ];
}
}
2008-05-30 11:40:08 +00:00
$list = array ( '{n}.' . $query [ 'fields' ][ 0 ], '{n}.' . $query [ 'fields' ][ 1 ], '{n}.' . $query [ 'fields' ][ 2 ]);
} else {
2008-06-11 08:54:27 +00:00
for ( $i = 0 ; $i < 2 ; $i ++ ) {
2008-06-10 17:27:12 +00:00
if ( strpos ( $query [ 'fields' ][ $i ], '.' ) === false ) {
$query [ 'fields' ][ $i ] = $this -> alias . '.' . $query [ 'fields' ][ $i ];
}
}
2008-05-30 11:40:08 +00:00
$list = array ( '{n}.' . $query [ 'fields' ][ 0 ], '{n}.' . $query [ 'fields' ][ 1 ], null );
}
}
if ( ! isset ( $query [ 'recursive' ]) || $query [ 'recursive' ] === null ) {
$query [ 'recursive' ] = - 1 ;
}
list ( $query [ 'list' ][ 'keyPath' ], $query [ 'list' ][ 'valuePath' ], $query [ 'list' ][ 'groupPath' ]) = $list ;
return $query ;
} elseif ( $state == 'after' ) {
if ( empty ( $results )) {
return array ();
}
2008-09-04 01:00:08 +00:00
$lst = $query [ 'list' ];
return Set :: combine ( $results , $lst [ 'keyPath' ], $lst [ 'valuePath' ], $lst [ 'groupPath' ]);
2008-05-30 11:40:08 +00:00
}
}
2009-05-01 14:05:46 -07:00
2008-06-26 19:33:44 +00:00
/**
2008-10-31 19:05:30 +00:00
* Detects the previous field 's value, then uses logic to find the ' wrapping '
* rows and return them .
2008-06-26 19:42:07 +00:00
*
2008-06-26 19:33:44 +00:00
* @ param string $state Either " before " or " after "
2008-06-26 19:42:07 +00:00
* @ param mixed $query
* @ param array $results
2008-09-25 16:49:56 +00:00
* @ return array
2008-06-26 19:33:44 +00:00
*/
2010-04-05 13:21:28 +10:00
protected function _findNeighbors ( $state , $query , $results = array ()) {
2008-06-26 19:33:44 +00:00
if ( $state == 'before' ) {
$query = array_merge ( array ( 'recursive' => 0 ), $query );
extract ( $query );
$conditions = ( array ) $conditions ;
if ( isset ( $field ) && isset ( $value )) {
if ( strpos ( $field , '.' ) === false ) {
2008-06-26 19:42:07 +00:00
$field = $this -> alias . '.' . $field ;
}
2008-06-26 19:33:44 +00:00
} else {
$field = $this -> alias . '.' . $this -> primaryKey ;
$value = $this -> id ;
}
$query [ 'conditions' ] = array_merge ( $conditions , array ( $field . ' <' => $value ));
$query [ 'order' ] = $field . ' DESC' ;
$query [ 'limit' ] = 1 ;
$query [ 'field' ] = $field ;
$query [ 'value' ] = $value ;
2008-06-26 19:42:07 +00:00
return $query ;
2008-06-26 19:33:44 +00:00
} elseif ( $state == 'after' ) {
extract ( $query );
unset ( $query [ 'conditions' ][ $field . ' <' ]);
2008-06-26 19:42:07 +00:00
$return = array ();
if ( isset ( $results [ 0 ])) {
2008-06-26 19:33:44 +00:00
$prevVal = Set :: extract ( '/' . str_replace ( '.' , '/' , $field ), $results [ 0 ]);
$query [ 'conditions' ][ $field . ' >=' ] = $prevVal [ 0 ];
$query [ 'conditions' ][ $field . ' !=' ] = $value ;
$query [ 'limit' ] = 2 ;
} else {
$return [ 'prev' ] = null ;
$query [ 'conditions' ][ $field . ' >' ] = $value ;
$query [ 'limit' ] = 1 ;
2008-06-26 19:42:07 +00:00
}
2008-06-26 19:33:44 +00:00
$query [ 'order' ] = $field . ' ASC' ;
$return2 = $this -> find ( 'all' , $query );
if ( ! array_key_exists ( 'prev' , $return )) {
2008-06-26 19:42:07 +00:00
$return [ 'prev' ] = $return2 [ 0 ];
2008-06-26 19:33:44 +00:00
}
if ( count ( $return2 ) == 2 ) {
2008-06-26 19:42:07 +00:00
$return [ 'next' ] = $return2 [ 1 ];
} elseif ( count ( $return2 ) == 1 && ! $return [ 'prev' ]) {
$return [ 'next' ] = $return2 [ 0 ];
2008-06-26 19:33:44 +00:00
} else {
$return [ 'next' ] = null ;
}
return $return ;
}
}
2009-05-01 14:05:46 -07:00
2008-06-26 19:33:44 +00:00
/**
2008-08-01 16:49:56 +00:00
* In the event of ambiguous results returned ( multiple top level results , with different parent_ids )
* top level results with different parent_ids to the first result will be dropped
*
2008-06-26 19:42:07 +00:00
* @ param mixed $state
* @ param mixed $query
* @ param array $results
2008-06-26 19:33:44 +00:00
* @ return array Threaded results
*/
2010-04-05 13:21:28 +10:00
protected function _findThreaded ( $state , $query , $results = array ()) {
2008-06-26 19:33:44 +00:00
if ( $state == 'before' ) {
return $query ;
} elseif ( $state == 'after' ) {
$return = $idMap = array ();
2008-07-05 14:10:42 +00:00
$ids = Set :: extract ( $results , '{n}.' . $this -> alias . '.' . $this -> primaryKey );
2008-06-26 19:33:44 +00:00
foreach ( $results as $result ) {
$result [ 'children' ] = array ();
2008-07-05 12:37:18 +00:00
$id = $result [ $this -> alias ][ $this -> primaryKey ];
2008-06-26 19:33:44 +00:00
$parentId = $result [ $this -> alias ][ 'parent_id' ];
if ( isset ( $idMap [ $id ][ 'children' ])) {
2008-07-05 12:37:18 +00:00
$idMap [ $id ] = array_merge ( $result , ( array ) $idMap [ $id ]);
2008-06-26 19:33:44 +00:00
} else {
2008-07-05 12:37:18 +00:00
$idMap [ $id ] = array_merge ( $result , array ( 'children' => array ()));
2008-06-26 19:33:44 +00:00
}
2008-07-05 14:10:42 +00:00
if ( ! $parentId || ! in_array ( $parentId , $ids )) {
2008-06-26 19:33:44 +00:00
$return [] =& $idMap [ $id ];
2008-07-05 14:10:42 +00:00
} else {
$idMap [ $parentId ][ 'children' ][] =& $idMap [ $id ];
2008-06-26 19:33:44 +00:00
}
}
2008-08-01 16:49:56 +00:00
if ( count ( $return ) > 1 ) {
$ids = array_unique ( Set :: extract ( '/' . $this -> alias . '/parent_id' , $return ));
if ( count ( $ids ) > 1 ) {
$root = $return [ 0 ][ $this -> alias ][ 'parent_id' ];
foreach ( $return as $key => $value ) {
if ( $value [ $this -> alias ][ 'parent_id' ] != $root ) {
2008-08-07 15:36:26 +00:00
unset ( $return [ $key ]);
2008-08-01 16:49:56 +00:00
}
2008-08-07 15:36:26 +00:00
}
2008-08-01 16:49:56 +00:00
}
}
2008-06-26 19:33:44 +00:00
return $return ;
}
2008-06-26 19:42:07 +00:00
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Passes query results through model and behavior afterFilter () methods .
2008-05-30 11:40:08 +00:00
*
* @ param array Results to filter
* @ param boolean $primary If this is the primary model results ( results from model where the find operation was performed )
* @ return array Set of filtered results
* @ access private
*/
function __filterResults ( $results , $primary = true ) {
2010-12-12 15:03:23 -05:00
$return = $this -> Behaviors -> trigger (
'afterFind' ,
2010-12-12 15:48:36 -05:00
array ( & $this , $results , $primary ),
2010-12-12 15:03:23 -05:00
array ( 'modParams' => 1 )
);
2008-05-30 11:40:08 +00:00
if ( $return !== true ) {
$results = $return ;
}
return $this -> afterFind ( $results , $primary );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* This resets the association arrays for the model back
2010-07-01 12:39:50 -04:00
* to those originally defined in the model . Normally called at the end
* of each call to Model :: find ()
2008-05-30 11:40:08 +00:00
*
* @ return boolean Success
*/
2010-04-05 13:19:38 +10:00
public function resetAssociations () {
2008-05-30 11:40:08 +00:00
if ( ! empty ( $this -> __backAssociation )) {
foreach ( $this -> __associations as $type ) {
if ( isset ( $this -> __backAssociation [ $type ])) {
$this -> { $type } = $this -> __backAssociation [ $type ];
}
}
$this -> __backAssociation = array ();
}
foreach ( $this -> __associations as $type ) {
foreach ( $this -> { $type } as $key => $name ) {
2010-07-14 16:58:12 -04:30
if ( property_exists ( $this , $key ) && ! empty ( $this -> { $key } -> __backAssociation )) {
2008-05-30 11:40:08 +00:00
$this -> { $key } -> resetAssociations ();
}
}
}
$this -> __backAssociation = array ();
return true ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns false if any fields passed match any ( by default , all if $or = false ) of their matching values .
2008-05-30 11:40:08 +00:00
*
* @ param array $fields Field / value pairs to search ( if no values specified , they are pulled from $this -> data )
* @ param boolean $or If false , all fields specified must match in order for a false return value
* @ return boolean False if any records matching any fields are found
*/
2010-04-05 13:19:38 +10:00
public function isUnique ( $fields , $or = true ) {
2008-05-30 11:40:08 +00:00
if ( ! is_array ( $fields )) {
$fields = func_get_args ();
if ( is_bool ( $fields [ count ( $fields ) - 1 ])) {
$or = $fields [ count ( $fields ) - 1 ];
unset ( $fields [ count ( $fields ) - 1 ]);
}
}
foreach ( $fields as $field => $value ) {
if ( is_numeric ( $field )) {
unset ( $fields [ $field ]);
$field = $value ;
if ( isset ( $this -> data [ $this -> alias ][ $field ])) {
$value = $this -> data [ $this -> alias ][ $field ];
} else {
$value = null ;
}
}
if ( strpos ( $field , '.' ) === false ) {
unset ( $fields [ $field ]);
$fields [ $this -> alias . '.' . $field ] = $value ;
}
}
if ( $or ) {
$fields = array ( 'or' => $fields );
}
if ( ! empty ( $this -> id )) {
2008-05-31 12:36:38 +00:00
$fields [ $this -> alias . '.' . $this -> primaryKey . ' !=' ] = $this -> id ;
2008-05-30 11:40:08 +00:00
}
2008-09-26 00:46:53 +00:00
return ( $this -> find ( 'count' , array ( 'conditions' => $fields , 'recursive' => - 1 )) == 0 );
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns a resultset for a given SQL statement . Custom SQL queries should be performed with this method .
2008-05-30 11:40:08 +00:00
*
* @ param string $sql SQL statement
* @ return array Resultset
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1027 / query
2008-05-30 11:40:08 +00:00
*/
function query () {
$params = func_get_args ();
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2008-05-30 11:40:08 +00:00
return call_user_func_array ( array ( & $db , 'query' ), $params );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2009-12-14 23:21:26 -05:00
* Returns true if all fields pass validation . Will validate hasAndBelongsToMany associations
* that use the 'with' key as well . Since __saveMulti is incapable of exiting a save operation .
2008-05-30 11:40:08 +00:00
*
2009-12-03 00:53:31 -05:00
* Will validate the currently set data . Use Model :: set () or Model :: create () to set the active data .
*
2008-05-30 11:40:08 +00:00
* @ param string $options An optional array of custom options to be made available in the beforeValidate callback
* @ return boolean True if there are no errors
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1182 / Validating - Data - from - the - Controller
2008-05-30 11:40:08 +00:00
*/
function validates ( $options = array ()) {
$errors = $this -> invalidFields ( $options );
2009-12-14 23:21:26 -05:00
if ( empty ( $errors ) && $errors !== false ) {
$errors = $this -> __validateWithModels ( $options );
}
2008-05-30 11:40:08 +00:00
if ( is_array ( $errors )) {
return count ( $errors ) === 0 ;
}
return $errors ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2009-12-14 23:21:26 -05:00
* Returns an array of fields that have failed validation . On the current model .
2008-05-30 11:40:08 +00:00
*
* @ param string $options An optional array of custom options to be made available in the beforeValidate callback
* @ return array Array of invalid fields
2009-12-03 00:53:31 -05:00
* @ see Model :: validates ()
2008-05-30 11:40:08 +00:00
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1182 / Validating - Data - from - the - Controller
2008-05-30 11:40:08 +00:00
*/
function invalidFields ( $options = array ()) {
2008-09-04 01:00:08 +00:00
if (
! $this -> Behaviors -> trigger (
'beforeValidate' ,
2010-12-12 15:03:23 -05:00
array ( & $this , $options ),
2008-09-04 01:00:08 +00:00
array ( 'break' => true , 'breakOn' => false )
) ||
$this -> beforeValidate ( $options ) === false
) {
2009-11-04 12:36:17 -05:00
return false ;
2008-05-30 11:40:08 +00:00
}
if ( ! isset ( $this -> validate ) || empty ( $this -> validate )) {
return $this -> validationErrors ;
}
2008-09-04 01:00:08 +00:00
2008-05-30 11:40:08 +00:00
$data = $this -> data ;
$methods = array_map ( 'strtolower' , get_class_methods ( $this ));
$behaviorMethods = array_keys ( $this -> Behaviors -> methods ());
if ( isset ( $data [ $this -> alias ])) {
$data = $data [ $this -> alias ];
} elseif ( ! is_array ( $data )) {
$data = array ();
}
2010-01-14 13:47:38 -05:00
$exists = $this -> exists ();
2008-05-30 11:40:08 +00:00
2008-08-27 08:01:00 +00:00
$_validate = $this -> validate ;
2009-03-17 18:19:12 +00:00
$whitelist = $this -> whitelist ;
2010-08-21 22:48:59 -04:00
if ( ! empty ( $options [ 'fieldList' ])) {
2009-03-17 18:19:12 +00:00
$whitelist = $options [ 'fieldList' ];
}
if ( ! empty ( $whitelist )) {
2008-08-27 16:43:15 +00:00
$validate = array ();
2009-03-17 18:19:12 +00:00
foreach (( array ) $whitelist as $f ) {
2008-08-27 08:01:00 +00:00
if ( ! empty ( $this -> validate [ $f ])) {
$validate [ $f ] = $this -> validate [ $f ];
}
}
$this -> validate = $validate ;
}
2008-08-27 16:43:15 +00:00
2008-05-30 11:40:08 +00:00
foreach ( $this -> validate as $fieldName => $ruleSet ) {
if ( ! is_array ( $ruleSet ) || ( is_array ( $ruleSet ) && isset ( $ruleSet [ 'rule' ]))) {
$ruleSet = array ( $ruleSet );
}
2008-09-04 01:00:08 +00:00
$default = array (
'allowEmpty' => null ,
'required' => null ,
'rule' => 'blank' ,
'last' => false ,
'on' => null
);
2008-05-30 11:40:08 +00:00
foreach ( $ruleSet as $index => $validator ) {
if ( ! is_array ( $validator )) {
$validator = array ( 'rule' => $validator );
}
$validator = array_merge ( $default , $validator );
if ( isset ( $validator [ 'message' ])) {
$message = $validator [ 'message' ];
} else {
2010-04-16 02:00:25 +10:00
$message = __ ( 'This field cannot be left blank' );
2008-05-30 11:40:08 +00:00
}
2008-09-04 01:00:08 +00:00
if (
empty ( $validator [ 'on' ]) || ( $validator [ 'on' ] == 'create' &&
2010-01-14 13:47:38 -05:00
! $exists ) || ( $validator [ 'on' ] == 'update' && $exists
2008-09-04 01:00:08 +00:00
)) {
$required = (
( ! isset ( $data [ $fieldName ]) && $validator [ 'required' ] === true ) ||
(
isset ( $data [ $fieldName ]) && ( empty ( $data [ $fieldName ]) &&
! is_numeric ( $data [ $fieldName ])) && $validator [ 'allowEmpty' ] === false
)
);
2008-09-19 15:27:43 +00:00
2008-09-04 01:00:08 +00:00
if ( $required ) {
2008-05-30 11:40:08 +00:00
$this -> invalidate ( $fieldName , $message );
if ( $validator [ 'last' ]) {
break ;
}
} elseif ( array_key_exists ( $fieldName , $data )) {
if ( empty ( $data [ $fieldName ]) && $data [ $fieldName ] != '0' && $validator [ 'allowEmpty' ] === true ) {
break ;
}
if ( is_array ( $validator [ 'rule' ])) {
$rule = $validator [ 'rule' ][ 0 ];
unset ( $validator [ 'rule' ][ 0 ]);
$ruleParams = array_merge ( array ( $data [ $fieldName ]), array_values ( $validator [ 'rule' ]));
} else {
$rule = $validator [ 'rule' ];
$ruleParams = array ( $data [ $fieldName ]);
}
$valid = true ;
if ( in_array ( strtolower ( $rule ), $methods )) {
2008-08-01 06:28:59 +00:00
$ruleParams [] = $validator ;
2008-05-30 11:40:08 +00:00
$ruleParams [ 0 ] = array ( $fieldName => $ruleParams [ 0 ]);
$valid = $this -> dispatchMethod ( $rule , $ruleParams );
} elseif ( in_array ( $rule , $behaviorMethods ) || in_array ( strtolower ( $rule ), $behaviorMethods )) {
2008-09-04 01:00:08 +00:00
$ruleParams [] = $validator ;
2008-05-30 11:40:08 +00:00
$ruleParams [ 0 ] = array ( $fieldName => $ruleParams [ 0 ]);
$valid = $this -> Behaviors -> dispatchMethod ( $this , $rule , $ruleParams );
2010-04-23 20:45:13 -04:00
} elseif ( method_exists ( 'Validation' , $rule )) {
2010-04-23 20:47:40 -04:00
$valid = call_user_func_array ( array ( 'Validation' , $rule ), $ruleParams );
2008-05-30 11:40:08 +00:00
} elseif ( ! is_array ( $validator [ 'rule' ])) {
$valid = preg_match ( $rule , $data [ $fieldName ]);
2009-09-30 22:34:29 -04:00
} elseif ( Configure :: read ( 'debug' ) > 0 ) {
2010-12-04 23:37:13 -02:00
trigger_error ( __ ( 'Could not find validation handler %s for %s' , $rule , $fieldName ), E_USER_WARNING );
2008-05-30 11:40:08 +00:00
}
2008-08-01 06:28:59 +00:00
if ( ! $valid || ( is_string ( $valid ) && strlen ( $valid ) > 0 )) {
if ( is_string ( $valid ) && strlen ( $valid ) > 0 ) {
$validator [ 'message' ] = $valid ;
} elseif ( ! isset ( $validator [ 'message' ])) {
2008-05-30 11:40:08 +00:00
if ( is_string ( $index )) {
$validator [ 'message' ] = $index ;
2008-07-30 16:26:48 +00:00
} elseif ( is_numeric ( $index ) && count ( $ruleSet ) > 1 ) {
$validator [ 'message' ] = $index + 1 ;
2008-05-30 11:40:08 +00:00
} else {
2008-07-30 16:26:48 +00:00
$validator [ 'message' ] = $message ;
2008-05-30 11:40:08 +00:00
}
}
$this -> invalidate ( $fieldName , $validator [ 'message' ]);
if ( $validator [ 'last' ]) {
break ;
}
}
}
}
}
}
2008-08-27 08:01:00 +00:00
$this -> validate = $_validate ;
2008-05-30 11:40:08 +00:00
return $this -> validationErrors ;
}
2009-05-01 14:05:46 -07:00
2009-12-14 23:21:26 -05:00
/**
* Runs validation for hasAndBelongsToMany associations that have 'with' keys
* set . And data in the set () data set .
*
* @ param array $options Array of options to use on Valdation of with models
* @ return boolean Failure of validation on with models .
* @ access private
* @ see Model :: validates ()
*/
function __validateWithModels ( $options ) {
$valid = true ;
2009-12-15 09:25:05 -05:00
foreach ( $this -> hasAndBelongsToMany as $assoc => $association ) {
if ( empty ( $association [ 'with' ]) || ! isset ( $this -> data [ $assoc ])) {
continue ;
}
list ( $join ) = $this -> joinModel ( $this -> hasAndBelongsToMany [ $assoc ][ 'with' ]);
$data = $this -> data [ $assoc ];
$newData = array ();
foreach (( array ) $data as $row ) {
if ( isset ( $row [ $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ]])) {
$newData [] = $row ;
} elseif ( isset ( $row [ $join ]) && isset ( $row [ $join ][ $this -> hasAndBelongsToMany [ $assoc ][ 'associationForeignKey' ]])) {
$newData [] = $row [ $join ];
2009-12-14 23:21:26 -05:00
}
}
2009-12-15 09:25:05 -05:00
if ( empty ( $newData )) {
continue ;
}
foreach ( $newData as $data ) {
$data [ $this -> hasAndBelongsToMany [ $assoc ][ 'foreignKey' ]] = $this -> id ;
$this -> { $join } -> create ( $data );
$valid = ( $valid && $this -> { $join } -> validates ( $options ));
}
2009-12-14 23:21:26 -05:00
}
return $valid ;
2008-05-30 11:40:08 +00:00
}
/**
2008-10-31 19:05:30 +00:00
* Marks a field as invalid , optionally setting the name of validation
* rule ( in case of multiple validation for field ) that was broken .
2008-05-30 11:40:08 +00:00
*
* @ param string $field The name of the field to invalidate
2009-08-02 19:18:54 +00:00
* @ param mixed $value Name of validation rule that was not failed , or validation message to
2010-03-05 22:07:39 -05:00
* be returned . If no validation key is provided , defaults to true .
2008-05-30 11:40:08 +00:00
*/
2010-04-05 13:19:38 +10:00
public function invalidate ( $field , $value = true ) {
2008-05-30 11:40:08 +00:00
if ( ! is_array ( $this -> validationErrors )) {
$this -> validationErrors = array ();
}
$this -> validationErrors [ $field ] = $value ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns true if given field name is a foreign key in this model .
2008-05-30 11:40:08 +00:00
*
* @ param string $field Returns true if the input string ends in " _id "
* @ return boolean True if the field is a foreign key listed in the belongsTo array .
*/
2010-04-05 13:19:38 +10:00
public function isForeignKey ( $field ) {
2008-05-30 11:40:08 +00:00
$foreignKeys = array ();
if ( ! empty ( $this -> belongsTo )) {
foreach ( $this -> belongsTo as $assoc => $data ) {
$foreignKeys [] = $data [ 'foreignKey' ];
}
}
return in_array ( $field , $foreignKeys );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2010-03-15 11:55:47 +01:00
* Escapes the field name and prepends the model name . Escaping is done according to the
2010-03-05 22:07:39 -05:00
* current database driver ' s rules .
2008-05-30 11:40:08 +00:00
*
* @ param string $field Field to escape ( e . g : id )
* @ param string $alias Alias for the model ( e . g : Post )
* @ return string The name of the escaped field for this Model ( i . e . id becomes `Post` . `id` ) .
*/
2010-04-05 13:19:38 +10:00
public function escapeField ( $field = null , $alias = null ) {
2008-05-30 11:40:08 +00:00
if ( empty ( $alias )) {
$alias = $this -> alias ;
}
if ( empty ( $field )) {
$field = $this -> primaryKey ;
}
2010-07-14 23:19:38 -04:30
$db = $this -> getDataSource ();
2010-03-17 10:32:36 -04:30
if ( strpos ( $field , $db -> name ( $alias ) . '.' ) === 0 ) {
2008-05-30 11:40:08 +00:00
return $field ;
}
return $db -> name ( $alias . '.' . $field );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Returns the current record ' s ID
*
* @ param integer $list Index on which the composed ID is located
* @ return mixed The ID of the current record , false if no ID
*/
2010-04-05 13:19:38 +10:00
public function getID ( $list = 0 ) {
2008-05-30 11:40:08 +00:00
if ( empty ( $this -> id ) || ( is_array ( $this -> id ) && isset ( $this -> id [ 0 ]) && empty ( $this -> id [ 0 ]))) {
return false ;
}
if ( ! is_array ( $this -> id )) {
return $this -> id ;
}
if ( empty ( $this -> id )) {
return false ;
}
if ( isset ( $this -> id [ $list ]) && ! empty ( $this -> id [ $list ])) {
return $this -> id [ $list ];
} elseif ( isset ( $this -> id [ $list ])) {
return false ;
}
foreach ( $this -> id as $id ) {
return $id ;
}
return false ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the ID of the last record this model inserted .
2008-05-30 11:40:08 +00:00
*
* @ return mixed Last inserted ID
*/
2010-04-05 13:19:38 +10:00
public function getLastInsertID () {
2008-05-30 11:40:08 +00:00
return $this -> getInsertID ();
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the ID of the last record this model inserted .
2008-05-30 11:40:08 +00:00
*
* @ return mixed Last inserted ID
*/
2010-04-05 13:19:38 +10:00
public function getInsertID () {
2008-05-30 11:40:08 +00:00
return $this -> __insertID ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Sets the ID of the last record this model inserted
2008-05-30 11:40:08 +00:00
*
* @ param mixed Last inserted ID
*/
2010-04-05 13:19:38 +10:00
public function setInsertID ( $id ) {
2008-05-30 11:40:08 +00:00
$this -> __insertID = $id ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the number of rows returned from the last query .
2008-05-30 11:40:08 +00:00
*
* @ return int Number of rows
*/
2010-04-05 13:19:38 +10:00
public function getNumRows () {
2010-07-14 23:19:38 -04:30
return $this -> getDataSource () -> lastNumRows ();
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Returns the number of rows affected by the last query .
2008-05-30 11:40:08 +00:00
*
* @ return int Number of rows
*/
2010-04-05 13:19:38 +10:00
public function getAffectedRows () {
2010-07-14 23:19:38 -04:30
return $this -> getDataSource () -> lastAffected ();
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Sets the DataSource to which this model is bound .
2008-05-30 11:40:08 +00:00
*
2008-06-18 17:46:56 +00:00
* @ param string $dataSource The name of the DataSource , as defined in app / config / database . php
2008-05-30 11:40:08 +00:00
* @ return boolean True on success
*/
2010-04-05 13:19:38 +10:00
public function setDataSource ( $dataSource = null ) {
2008-05-30 11:40:08 +00:00
$oldConfig = $this -> useDbConfig ;
if ( $dataSource != null ) {
$this -> useDbConfig = $dataSource ;
}
2010-07-15 21:16:52 -04:30
$db = ConnectionManager :: getDataSource ( $this -> useDbConfig );
2008-05-30 11:40:08 +00:00
if ( ! empty ( $oldConfig ) && isset ( $db -> config [ 'prefix' ])) {
2010-07-15 22:16:19 -04:30
$oldDb = ConnectionManager :: getDataSource ( $oldConfig );
2008-05-30 11:40:08 +00:00
2008-08-19 01:18:33 +00:00
if ( ! isset ( $this -> tablePrefix ) || ( ! isset ( $oldDb -> config [ 'prefix' ]) || $this -> tablePrefix == $oldDb -> config [ 'prefix' ])) {
2008-05-30 11:40:08 +00:00
$this -> tablePrefix = $db -> config [ 'prefix' ];
}
} elseif ( isset ( $db -> config [ 'prefix' ])) {
$this -> tablePrefix = $db -> config [ 'prefix' ];
}
2009-10-12 23:38:55 -04:00
if ( empty ( $db ) || ! is_object ( $db )) {
2010-08-29 21:37:25 -04:00
throw new MissingConnectionException ( array ( 'class' => $this -> name ));
2008-05-30 11:40:08 +00:00
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Gets the DataSource to which this model is bound .
* Not safe for use with some versions of PHP4 , because this class is overloaded .
*
* @ return object A DataSource object
*/
2010-07-15 23:17:13 -04:30
public function getDataSource () {
2010-07-15 21:16:52 -04:30
static $configured = false ;
if ( ! $configured && $this -> useTable !== false ) {
$configured = true ;
$this -> setSource ( $this -> useTable );
}
2010-07-14 23:19:38 -04:30
return ConnectionManager :: getDataSource ( $this -> useDbConfig );
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2010-04-04 18:17:43 +10:00
/**
* Get associations
*
* @ return array
*/
2010-04-05 13:19:38 +10:00
public function associations () {
2010-04-04 18:17:43 +10:00
return $this -> __associations ;
}
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Gets all the models with which this model is associated .
2008-05-30 11:40:08 +00:00
*
* @ param string $type Only result associations of this type
* @ return array Associations
*/
2010-04-05 13:19:38 +10:00
public function getAssociated ( $type = null ) {
2008-05-30 11:40:08 +00:00
if ( $type == null ) {
$associated = array ();
foreach ( $this -> __associations as $assoc ) {
if ( ! empty ( $this -> { $assoc })) {
$models = array_keys ( $this -> { $assoc });
foreach ( $models as $m ) {
$associated [ $m ] = $assoc ;
}
}
}
return $associated ;
} elseif ( in_array ( $type , $this -> __associations )) {
if ( empty ( $this -> { $type })) {
return array ();
}
return array_keys ( $this -> { $type });
} else {
2009-05-01 14:05:46 -07:00
$assoc = array_merge (
$this -> hasOne ,
$this -> hasMany ,
$this -> belongsTo ,
$this -> hasAndBelongsToMany
);
2008-05-30 11:40:08 +00:00
if ( array_key_exists ( $type , $assoc )) {
foreach ( $this -> __associations as $a ) {
if ( isset ( $this -> { $a }[ $type ])) {
$assoc [ $type ][ 'association' ] = $a ;
break ;
}
}
return $assoc [ $type ];
}
return null ;
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2009-05-01 14:05:46 -07:00
* Gets the name and fields to be used by a join model . This allows specifying join fields
* in the association definition .
2008-05-30 11:40:08 +00:00
*
* @ param object $model The model to be joined
* @ param mixed $with The 'with' key of the model association
* @ param array $keys Any join keys which must be merged with the keys queried
* @ return array
*/
2010-04-05 13:19:38 +10:00
public function joinModel ( $assoc , $keys = array ()) {
2008-05-30 11:40:08 +00:00
if ( is_string ( $assoc )) {
2010-07-14 17:54:19 -04:30
list (, $assoc ) = pluginSplit ( $assoc );
2008-05-30 11:40:08 +00:00
return array ( $assoc , array_keys ( $this -> { $assoc } -> schema ()));
} elseif ( is_array ( $assoc )) {
$with = key ( $assoc );
return array ( $with , array_unique ( array_merge ( $assoc [ $with ], $keys )));
}
2009-05-01 14:05:46 -07:00
trigger_error (
2010-12-04 23:37:13 -02:00
__ ( 'Invalid join model settings in %s' , $model -> alias ),
2009-05-01 14:05:46 -07:00
E_USER_WARNING
);
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called before each find operation . Return false if you want to halt the find
* call , otherwise return the ( modified ) query data .
2008-05-30 11:40:08 +00:00
*
* @ param array $queryData Data used to execute this query , i . e . conditions , order , etc .
2009-05-01 14:05:46 -07:00
* @ return mixed true if the operation should continue , false if it should abort ; or , modified
* $queryData to continue with new $queryData
2008-05-30 11:40:08 +00:00
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #beforeFind-1049
2008-05-30 11:40:08 +00:00
*/
function beforeFind ( $queryData ) {
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called after each find operation . Can be used to modify any results returned by find () .
2008-11-08 02:54:07 +00:00
* Return value should be the ( modified ) results .
2008-05-30 11:40:08 +00:00
*
* @ param mixed $results The results of the find operation
* @ param boolean $primary Whether this model is being queried directly ( vs . being queried as an association )
* @ return mixed Result of the find operation
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #afterFind-1050
2008-05-30 11:40:08 +00:00
*/
function afterFind ( $results , $primary = false ) {
return $results ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called before each save operation , after validation . Return a non - true result
* to halt the save .
2008-05-30 11:40:08 +00:00
*
* @ return boolean True if the operation should continue , false if it should abort
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #beforeSave-1052
2008-05-30 11:40:08 +00:00
*/
2008-09-23 18:50:27 +00:00
function beforeSave ( $options = array ()) {
2008-05-30 11:40:08 +00:00
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-11-08 02:54:07 +00:00
* Called after each successful save operation .
2008-05-30 11:40:08 +00:00
*
* @ param boolean $created True if this save created a new record
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #afterSave-1053
2008-05-30 11:40:08 +00:00
*/
function afterSave ( $created ) {
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2009-05-22 01:20:47 +00:00
* Called before every deletion operation .
2008-05-30 11:40:08 +00:00
*
* @ param boolean $cascade If true records that depend on this record will also be deleted
* @ return boolean True if the operation should continue , false if it should abort
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #beforeDelete-1054
2008-05-30 11:40:08 +00:00
*/
function beforeDelete ( $cascade = true ) {
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called after every deletion operation .
2008-05-30 11:40:08 +00:00
*
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #afterDelete-1055
2008-05-30 11:40:08 +00:00
*/
function afterDelete () {
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2009-11-04 12:36:17 -05:00
* Called during validation operations , before validation . Please note that custom
2008-10-31 19:05:30 +00:00
* validation rules can be defined in $validate .
2008-05-30 11:40:08 +00:00
*
* @ return boolean True if validate operation should continue , false to abort
2008-09-30 00:13:15 +00:00
* @ param $options array Options passed from model :: save (), see $options of model :: save () .
2008-05-30 11:40:08 +00:00
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #beforeValidate-1051
2008-05-30 11:40:08 +00:00
*/
2008-09-23 18:50:27 +00:00
function beforeValidate ( $options = array ()) {
2008-05-30 11:40:08 +00:00
return true ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called when a DataSource - level error occurs .
2008-05-30 11:40:08 +00:00
*
* @ access public
2010-03-05 22:07:39 -05:00
* @ link http :// book . cakephp . org / view / 1048 / Callback - Methods #onError-1056
2008-05-30 11:40:08 +00:00
*/
function onError () {
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Private method . Clears cache for this model .
2008-05-30 11:40:08 +00:00
*
* @ param string $type If null this deletes cached views if Cache . check is true
2010-03-05 22:07:39 -05:00
* Will be used to allow deleting query cache also
2008-05-30 11:40:08 +00:00
* @ return boolean true on delete
* @ access protected
* @ todo
*/
function _clearCache ( $type = null ) {
if ( $type === null ) {
if ( Configure :: read ( 'Cache.check' ) === true ) {
$assoc [] = strtolower ( Inflector :: pluralize ( $this -> alias ));
2008-10-26 18:51:38 +00:00
$assoc [] = strtolower ( Inflector :: underscore ( Inflector :: pluralize ( $this -> alias )));
2008-05-30 11:40:08 +00:00
foreach ( $this -> __associations as $key => $association ) {
foreach ( $this -> $association as $key => $className ) {
$check = strtolower ( Inflector :: pluralize ( $className [ 'className' ]));
if ( ! in_array ( $check , $assoc )) {
$assoc [] = strtolower ( Inflector :: pluralize ( $className [ 'className' ]));
2008-10-26 18:51:38 +00:00
$assoc [] = strtolower ( Inflector :: underscore ( Inflector :: pluralize ( $className [ 'className' ])));
2008-05-30 11:40:08 +00:00
}
}
}
clearCache ( $assoc );
return true ;
}
} else {
//Will use for query cache deleting
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called when serializing a model .
2008-05-30 11:40:08 +00:00
*
* @ return array Set of object variable names this model has
* @ access private
*/
function __sleep () {
$return = array_keys ( get_object_vars ( $this ));
return $return ;
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 19:05:30 +00:00
* Called when de - serializing a model .
2008-05-30 11:40:08 +00:00
*
* @ access private
2008-09-25 16:49:56 +00:00
* @ todo
2008-05-30 11:40:08 +00:00
*/
function __wakeup () {
}
}
2010-07-15 23:11:30 -04:30