.* protected $_disable_wild_select = FALSE; // Suppress ORMs inclusion of . to column joins protected $_disable_join_table_name = FALSE; // Suppress ORMs use of limit protected $_disable_limit = FALSE; /** * Model configuration, reload on wakeup? * @var bool */ protected $_reload_on_wakeup = TRUE; /** * Database Object * @var Database */ protected $_db = NULL; /** * Database config group * @var String */ protected $_db_group = NULL; /** * Database methods applied * @var array */ protected $_db_applied = array(); /** * Database methods pending * @var array */ protected $_db_pending = array(); /** * Reset builder * @var bool */ protected $_db_reset = TRUE; /** * Database query builder * @var Database_Query_Builder_Select */ protected $_db_builder; /** * With calls already applied * @var array */ protected $_with_applied = array(); /** * Data to be loaded into the model from a database call cast * @var array */ protected $_cast_data = array(); /** * The message filename used for validation errors. * Defaults to ORM::$_object_name * @var string */ protected $_errors_filename = NULL; /** * Constructs a new model and loads a record if given * * @param mixed $id Parameter for find or object to load */ public function __construct($id = NULL) { $this->_initialize(); if ($id !== NULL) { if (is_array($id)) { foreach ($id as $column => $value) { // Passing an array of column => values $this->where($column, '=', $value); } $this->find(); } else { // Passing the primary key $this->where(($this->_disable_join_table_name ? '' : $this->_object_name.'.').$this->_primary_key, '=', $id)->find(); } } elseif ( ! empty($this->_cast_data)) { // Load preloaded data from a database call cast $this->_load_values($this->_cast_data); $this->_cast_data = array(); } } /** * Prepares the model database connection, determines the table name, * and loads column information. * * @return void */ protected function _initialize() { // Set the object name if none predefined if (empty($this->_object_name)) { $this->_object_name = strtolower(substr(get_class($this), 6)); } // Check if this model has already been initialized if ( ! $init = Arr::get(ORM::$_init_cache, $this->_object_name, FALSE)) { $init = array( '_belongs_to' => array(), '_has_one' => array(), '_has_many' => array(), ); // Set the object plural name if none predefined if ( ! isset($this->_object_plural)) { $init['_object_plural'] = Inflector::plural($this->_object_name); } if ( ! $this->_errors_filename) { $init['_errors_filename'] = $this->_object_name; } if ( ! is_object($this->_db)) { // Get database instance $init['_db'] = Database::instance($this->_db_group); } if (empty($this->_table_name)) { // Table name is the same as the object name $init['_table_name'] = $this->_object_name; if ($this->_table_names_plural === TRUE) { // Make the table name plural $init['_table_name'] = Arr::get($init, '_object_plural', $this->_object_plural); } } $defaults = array(); $defaults['cached'] = FALSE; foreach ($this->_belongs_to as $alias => $details) { if ( ! isset($details['model'])) { $defaults['model'] = str_replace(' ', '_', ucwords(str_replace('_', ' ', $alias))); } $defaults['foreign_key'] = $alias.$this->_foreign_key_suffix; $init['_belongs_to'][$alias] = array_merge($defaults, $details); } foreach ($this->_has_one as $alias => $details) { if ( ! isset($details['model'])) { $defaults['model'] = str_replace(' ', '_', ucwords(str_replace('_', ' ', $alias))); } $defaults['foreign_key'] = $this->_object_name.$this->_foreign_key_suffix; $defaults['far_key'] = Inflector::singular($alias).$this->_foreign_key_suffix; $init['_has_one'][$alias] = array_merge($defaults, $details); } foreach ($this->_has_many as $alias => $details) { if ( ! isset($details['model'])) { $defaults['model'] = str_replace(' ', '_', ucwords(str_replace('_', ' ', ($this->_model_names_plural ? Inflector::singular($alias) : $alias)))); } $defaults['foreign_key'] = $this->_object_name.$this->_foreign_key_suffix; $defaults['through'] = NULL; if ( ! isset($details['far_key'])) { $defaults['far_key'] = Inflector::singular($alias).$this->_foreign_key_suffix; } $init['_has_many'][$alias] = array_merge($defaults, $details); } ORM::$_init_cache[$this->_object_name] = $init; } // Assign initialized properties to the current object foreach ($init as $property => $value) { $this->{$property} = $value; } // Load column information $this->reload_columns(); // Clear initial model state $this->clear(); } /** * Initializes validation rules, and labels * * @return void */ protected function _validation() { // Build the validation object with its rules $this->_validation = Validation::factory($this->_object) ->bind(':model', $this) ->bind(':original_values', $this->_original_values) ->bind(':changed', $this->_changed); foreach ($this->rules() as $field => $rules) { $this->_validation->rules($field, $rules); } // Use column names by default for labels $columns = array_keys($this->_table_columns); // Merge user-defined labels $labels = array_merge(array_combine($columns, $columns), $this->labels()); foreach ($labels as $field => $label) { $this->_validation->label($field, $label); } } /** * Reload column definitions. * * @chainable * @param boolean $force Force reloading * @return ORM */ public function reload_columns($force = FALSE) { if ($force === TRUE OR empty($this->_table_columns)) { if (isset(ORM::$_column_cache[$this->_object_name])) { // Use cached column information $this->_table_columns = ORM::$_column_cache[$this->_object_name]; } else { // Grab column information from database $this->_table_columns = $this->list_columns(); // Load column cache ORM::$_column_cache[$this->_object_name] = $this->_table_columns; } } return $this; } /** * Unloads the current object and clears the status. * * @chainable * @return ORM */ public function clear() { // Create an array with all the columns set to NULL $values = array_combine(array_keys($this->_table_columns), array_fill(0, count($this->_table_columns), NULL)); // Replace the object and reset the object status $this->_object = $this->_changed = $this->_related = $this->_original_values = array(); // Replace the current object with an empty one $this->_load_values($values); // Reset primary key $this->_primary_key_value = NULL; // Reset the loaded state $this->_loaded = FALSE; $this->reset(); return $this; } /** * Reloads the current object from the database. * * @chainable * @return ORM */ public function reload() { $primary_key = $this->pk(); // Replace the object and reset the object status $this->_object = $this->_changed = $this->_related = $this->_original_values = array(); // Only reload the object if we have one to reload if ($this->_loaded) return $this->clear() ->where($this->_object_name.'.'.$this->_primary_key, '=', $primary_key) ->find(); else return $this->clear(); } /** * Checks if object data is set. * * @param string $column Column name * @return boolean */ public function __isset($column) { return (isset($this->_object[$column]) OR isset($this->_related[$column]) OR isset($this->_has_one[$column]) OR isset($this->_belongs_to[$column]) OR isset($this->_has_many[$column])); } /** * Unsets object data. * * @param string $column Column name * @return void */ public function __unset($column) { unset($this->_object[$column], $this->_changed[$column], $this->_related[$column]); } /** * Displays the primary key of a model when it is converted to a string. * * @return string */ public function __toString() { return (string) $this->pk(); } /** * Allows serialization of only the object data and state, to prevent * "stale" objects being unserialized, which also requires less memory. * * @return string */ public function serialize() { // Store only information about the object foreach (array('_primary_key_value', '_object', '_changed', '_loaded', '_saved', '_sorting', '_original_values') as $var) { $data[$var] = $this->{$var}; } return serialize($data); } /** * Check whether the model data has been modified. * If $field is specified, checks whether that field was modified. * * @param string $field field to check for changes * @return bool Whether or not the field has changed */ public function changed($field = NULL) { return ($field === NULL) ? $this->_changed : Arr::get($this->_changed, $field); } /** * Prepares the database connection and reloads the object. * * @param string $data String for unserialization * @return void */ public function unserialize($data) { // Initialize model $this->_initialize(); foreach (unserialize($data) as $name => $var) { $this->{$name} = $var; } if ($this->_reload_on_wakeup === TRUE) { // Reload the object $this->reload(); } } /** * Handles retrieval of all model values, relationships, and metadata. * [!!] This should not be overridden. * * @param string $column Column name * @return mixed */ public function __get($column) { return $this->get($column); } /** * Handles getting of column * Override this method to add custom get behavior * * @param string $column Column name * @throws Kohana_Exception * @return mixed */ public function get($column) { if (array_key_exists($column, $this->_object)) { return (in_array($column, $this->_serialize_columns)) ? $this->_unserialize_value($this->_object[$column]) : $this->_object[$column]; } elseif (isset($this->_related[$column])) { // Return related model that has already been fetched return $this->_related[$column]; } elseif (isset($this->_belongs_to[$column])) { $model = $this->_related($column); // Use this model's column and foreign model's primary key $col = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$model->_primary_key; $val = $this->_object[$this->_belongs_to[$column]['foreign_key']]; // Make sure we don't run WHERE "AUTO_INCREMENT column" = NULL queries. This would // return the last inserted record instead of an empty result. // See: http://mysql.localhost.net.ar/doc/refman/5.1/en/server-session-variables.html#sysvar_sql_auto_is_null if ($val !== NULL) { // Enable caching query if ($this->_belongs_to[$column]['cached']) $model->cached(); $model->where($col, '=', $val)->find(); } return $this->_related[$column] = $model; } elseif (isset($this->_has_one[$column])) { $model = $this->_related($column); if (! is_array($this->_has_one[$column]['foreign_key'])) { // Use this model's primary key value and foreign model's column $col = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$this->_has_one[$column]['foreign_key']; $val = $this->_object[$this->_has_one[$column]['far_key']]; } else { foreach ($this->_has_one[$column]['foreign_key'] as $fk) { // Simple has_many relationship, search where target model's foreign key is this model's primary key $col = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$fk; $val = $this->_object[$fk]; $model = $model->where($col, '=', $val); } } // Enable caching query if ($this->_has_one[$column]['cached']) $model->cached(); $model->where($col, '=', $val)->find(); return $this->_related[$column] = $model; } elseif (isset($this->_has_many[$column])) { $model = ORM::factory($this->_has_many[$column]['model']); if (! is_array($this->_has_many[$column]['foreign_key'])) { if (isset($this->_has_many[$column]['through'])) { // Grab has_many "through" relationship table $through = $this->_has_many[$column]['through']; // Join on through model's target foreign key (far_key) and target model's primary key $join_col1 = ($this->_disable_join_table_name ? '' : $through.'.').$this->_has_many[$column]['far_key']; $join_col2 = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$model->_primary_key; $model->join($through)->on($join_col1, '=', $join_col2); // Through table's source foreign key (foreign_key) should be this model's primary key $col = ($this->_disable_join_table_name ? '' : $through.'.').$this->_has_many[$column]['foreign_key']; $val = $this->pk(); } else { // Simple has_many relationship, search where target model's foreign key is this model's primary key $col = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$this->_has_many[$column]['foreign_key']; $val = $this->_object[$this->_has_many[$column]['far_key']]; } // Enable caching query if ($this->_has_many[$column]['cached']) $model->cached(); return $model->where($col, '=', $val); } else { foreach ($this->_has_many[$column]['foreign_key'] as $mk => $fk) { if (isset($this->_has_many[$column]['through'])) { throw new Kohana_Exception('This code hasnt been written yet!'); } else { // Simple has_many relationship, search where target model's foreign key is this model's primary key $col = ($this->_disable_join_table_name ? '' : $model->_object_name.'.').$fk; $val = $this->_object[$mk]; } $model = $model->where($col, '=', $val); } // Enable caching query if ($this->_has_many[$column]['cached']) $model->cached(); return $model; } } else { throw new Kohana_Exception('The :property property does not exist in the :class class', array(':property' => $column, ':class' => get_class($this))); } } /** * Base set method. * [!!] This should not be overridden. * * @param string $column Column name * @param mixed $value Column value * @return void */ public function __set($column, $value) { $this->set($column, $value); } /** * Handles setting of columns * Override this method to add custom set behavior * * @param string $column Column name * @param mixed $value Column value * @throws Kohana_Exception * @return ORM */ public function set($column, $value) { if ( ! isset($this->_object_name)) { // Object not yet constructed, so we're loading data from a database call cast $this->_cast_data[$column] = $value; return $this; } if (in_array($column, $this->_serialize_columns)) { $value = $this->_serialize_value($value); } if (array_key_exists($column, $this->_object)) { // Filter the data $value = $this->run_filter($column, $value); // See if the data really changed if ($value !== $this->_object[$column]) { $this->_object[$column] = $value; // Data has changed $this->_changed[$column] = $column; // Object is no longer saved or valid $this->_saved = $this->_valid = FALSE; } // If the replacement value is the same as the original value, we dont need to update. // @note This assumes that the value returned from the DB is always a string. if (array_key_exists($column, $this->_original_values) AND (string)$value === $this->_original_values[$column] AND array_key_exists($column, $this->_changed)) { unset($this->_changed[$column]); // Reset back our saved status. if (! count($this->_changed)) { $this->_saved = $this->_valid = TRUE; } } } elseif (isset($this->_belongs_to[$column])) { // Update related object itself $this->_related[$column] = $value; // Update the foreign key of this model $this->_object[$this->_belongs_to[$column]['foreign_key']] = ($value instanceof ORM) ? $value->pk() : NULL; $this->_changed[$column] = $this->_belongs_to[$column]['foreign_key']; } else { throw new Kohana_Exception('The :property: property does not exist in the :class: class', array(':property:' => $column, ':class:' => get_class($this))); } return $this; } /** * Set values from an array with support for one-one relationships. This method should be used * for loading in post data, etc. * * @param array $values Array of column => val * @param array $expected Array of keys to take from $values * @return ORM */ public function values(array $values, array $expected = NULL) { // Default to expecting everything except the primary key if ($expected === NULL) { $expected = array_keys($this->_table_columns); // Don't set the primary key by default unset($values[$this->_primary_key]); } foreach ($expected as $key => $column) { if (is_string($key)) { // isset() fails when the value is NULL (we want it to pass) if ( ! array_key_exists($key, $values)) continue; // Try to set values to a related model $this->{$key}->values($values[$key], $column); } else { // isset() fails when the value is NULL (we want it to pass) if ( ! array_key_exists($column, $values)) continue; // Update the column, respects __set() $this->$column = $values[$column]; } } return $this; } /** * Returns the values of this object as an array, including any related one-one * models that have already been loaded using with() * * @return array */ public function as_array() { $object = array(); foreach ($this->_object as $column => $value) { // Call __get for any user processing $object[$column] = $this->__get($column); } foreach ($this->_related as $column => $model) { // Include any related objects that are already loaded $object[$column] = $model->as_array(); } return $object; } /** * Binds another one-to-one object to this model. One-to-one objects * can be nested using 'object1:object2' syntax * * @param string $target_path Target model to bind to * @return ORM */ public function with($target_path) { if (isset($this->_with_applied[$target_path])) { // Don't join anything already joined return $this; } // Split object parts $aliases = explode(':', $target_path); $target = $this; foreach ($aliases as $alias) { // Go down the line of objects to find the given target $parent = $target; $target = $parent->_related($alias); if ( ! $target) { // Can't find related object return $this; } } // Target alias is at the end $target_alias = $alias; // Pop-off top alias to get the parent path (user:photo:tag becomes user:photo - the parent table prefix) array_pop($aliases); $parent_path = implode(':', $aliases); if (empty($parent_path)) { // Use this table name itself for the parent path $parent_path = $this->_object_name; } else { if ( ! isset($this->_with_applied[$parent_path])) { // If the parent path hasn't been joined yet, do it first (otherwise LEFT JOINs fail) $this->with($parent_path); } } // Add to with_applied to prevent duplicate joins $this->_with_applied[$target_path] = TRUE; // Use the keys of the empty object to determine the columns foreach (array_keys($target->_object) as $column) { $name = $target_path.'.'.$column; $alias = $target_path.':'.$column; // Add the prefix so that load_result can determine the relationship $this->select(array($name, $alias)); } if (isset($parent->_belongs_to[$target_alias])) { // Parent belongs_to target, use target's primary key and parent's foreign key $join_col1 = $target_path.'.'.$target->_primary_key; $join_col2 = $parent_path.'.'.$parent->_belongs_to[$target_alias]['foreign_key']; } else { // Parent has_one target, use parent's primary key as target's foreign key $join_col1 = $parent_path.'.'.$parent->_primary_key; $join_col2 = $target_path.'.'.$parent->_has_one[$target_alias]['foreign_key']; } // Join the related object into the result $this->join(array($target->_table_name, $target_path), 'LEFT')->on($join_col1, '=', $join_col2); return $this; } /** * Initializes the Database Builder to given query type * * @param integer $type Type of Database query * @return ORM */ protected function _build($type) { // Construct new builder object based on query type switch ($type) { case Database::SELECT: $this->_db_builder = DB::select(); break; case Database::UPDATE: $this->_db_builder = DB::update(array($this->_table_name, $this->_object_name)); break; case Database::DELETE: // Cannot use an alias for DELETE queries $this->_db_builder = DB::delete($this->_table_name); } // Process pending database method calls foreach ($this->_db_pending as $method) { $name = $method['name']; $args = $method['args']; $this->_db_applied[$name] = $name; call_user_func_array(array($this->_db_builder, $name), $args); } return $this; } /** * Finds and loads a single database row into the object. * * @chainable * @throws Kohana_Exception * @return ORM */ public function find() { if ($this->_loaded) throw new Kohana_Exception('Method find() cannot be called on loaded objects'); if ( ! empty($this->_load_with)) { foreach ($this->_load_with as $alias) { // Bind auto relationships $this->with($alias); } } $this->_build(Database::SELECT); return $this->_load_result(FALSE); } /** * Finds multiple database rows and returns an iterator of the rows found. * * @throws Kohana_Exception * @return Database_Result */ public function find_all() { if ($this->_loaded) throw new Kohana_Exception('Method find_all() cannot be called on loaded objects'); if ( ! empty($this->_load_with)) { foreach ($this->_load_with as $alias) { // Bind auto relationships $this->with($alias); } } $this->_build(Database::SELECT); return $this->_load_result(TRUE); } /** * Returns an array of columns to include in the select query. This method * can be overridden to change the default select behavior. * * @return array Columns to select */ protected function _build_select() { $columns = array(); foreach ($this->_table_columns as $column => $_) { $columns[] = $this->_disable_column_alias ? $column : array($this->_object_name.'.'.$column, $column); } return $columns; } /** * Loads a database result, either as a new record for this model, or as * an iterator for multiple rows. * * @chainable * @param bool $multiple Return an iterator or load a single row * @return ORM|Database_Result */ protected function _load_result($multiple = FALSE) { $this->_db_builder->from(array($this->_table_name, $this->_object_name)); if ($multiple === FALSE AND ! $this->_disable_limit) { // Only fetch 1 record $this->_db_builder->limit(1); } // Select all columns by default if (! $this->_disable_wild_select) $this->_db_builder->select_array($this->_build_select()); if ( ! isset($this->_db_applied['order_by']) AND ! empty($this->_sorting)) { foreach ($this->_sorting as $column => $direction) { if (strpos($column, '.') === FALSE) { // Sorting column for use in JOINs $column = ($this->_disable_join_table_name ? '' : $this->_object_name.'.').$column; } $this->_db_builder->order_by($column, $direction); } } if ($multiple === TRUE) { // Return database iterator casting to this object type $result = $this->_db_builder->as_object(get_class($this))->execute($this->_db); $this->reset(); return $result; } else { // Load the result as an associative array $result = $this->_db_builder->as_assoc()->execute($this->_db); $this->reset(); if ($result->count() === 1) { // Load object values $this->_load_values($result->current()); } else { // Clear the object, nothing was found $this->clear(); } return $this; } } /** * Loads an array of values into into the current object. * * @chainable * @param array $values Values to load * @return ORM */ protected function _load_values(array $values) { if (array_key_exists($this->_primary_key, $values)) { if ($values[$this->_primary_key] !== NULL) { // Flag as loaded and valid $this->_loaded = $this->_valid = TRUE; // Store primary key $this->_primary_key_value = $values[$this->_primary_key]; } else { // Not loaded or valid $this->_loaded = $this->_valid = FALSE; } } // Related objects $related = array(); foreach ($values as $column => $value) { if (strpos($column, ':') === FALSE) { // Load the value to this model $this->_object[$column] = $value; } else { // Column belongs to a related model list ($prefix, $column) = explode(':', $column, 2); $related[$prefix][$column] = $value; } } if ( ! empty($related)) { foreach ($related as $object => $values) { // Load the related objects with the values in the result $this->_related($object)->_load_values($values); } } if ($this->_loaded) { // Store the object in its original state $this->_original_values = $this->_object; } return $this; } /** * Rule definitions for validation * * @return array */ public function rules() { return array(); } /** * Filters a value for a specific column * * @param string $field The column name * @param string $value The value to filter * @return string */ protected function run_filter($field, $value, $filters=NULL) { if (is_null($filters)) $filters = $this->filters(); // Get the filters for this column $wildcards = empty($filters[TRUE]) ? array() : $filters[TRUE]; // Merge in the wildcards $filters = empty($filters[$field]) ? $wildcards : array_merge($wildcards, $filters[$field]); // Bind the field name and model so they can be used in the filter method $_bound = array ( ':field' => $field, ':model' => $this, ); foreach ($filters as $array) { // Value needs to be bound inside the loop so we are always using the // version that was modified by the filters that already ran $_bound[':value'] = $value; // Filters are defined as array($filter, $params) $filter = $array[0]; $params = Arr::get($array, 1, array(':value')); foreach ($params as $key => $param) { if (is_string($param) AND array_key_exists($param, $_bound)) { // Replace with bound value $params[$key] = $_bound[$param]; } } if (is_array($filter) OR ! is_string($filter)) { // This is either a callback as an array or a lambda $value = call_user_func_array($filter, $params); } elseif (strpos($filter, '::') === FALSE) { // Use a function call $function = new ReflectionFunction($filter); // Call $function($this[$field], $param, ...) with Reflection $value = $function->invokeArgs($params); } else { // Split the class and method of the rule list($class, $method) = explode('::', $filter, 2); // Use a static method call $method = new ReflectionMethod($class, $method); // Call $Class::$method($this[$field], $param, ...) with Reflection $value = $method->invokeArgs(NULL, $params); } } return $value; } /** * Filter definitions for validation * * @return array */ public function filters() { return array(); } /** * Label definitions for validation * * @return array */ public function labels() { return array(); } /** * Validates the current model's data * * @param Validation $extra_validation Validation object * @throws ORM_Validation_Exception * @return ORM */ public function check(Validation $extra_validation = NULL) { // Determine if any external validation failed $extra_errors = ($extra_validation AND ! $extra_validation->check()); // Always build a new validation object $this->_validation(); $array = $this->_validation; if (($this->_valid = $array->check()) === FALSE OR $extra_errors) { $exception = new ORM_Validation_Exception($this->errors_filename(), $array); if ($extra_errors) { // Merge any possible errors from the external object $exception->add_object('_external', $extra_validation); } throw $exception; } return $this; } /** * Insert a new object to the database * @param Validation $validation Validation object * @throws Kohana_Exception * @return ORM */ public function create(Validation $validation = NULL) { if ($this->_loaded) throw new Kohana_Exception('Cannot create :model model because it is already loaded.', array(':model' => $this->_object_name)); // Require model validation before saving if ( ! $this->_valid OR $validation) { $this->check($validation); } $data = array(); foreach ($this->_changed as $column) { // Generate list of column => values $data[$column] = $this->_object[$column]; } if (is_array($this->_created_column)) { // Fill the created column $column = $this->_created_column['column']; $format = $this->_created_column['format']; $data[$column] = $this->_object[$column] = ($format === TRUE) ? time() : date($format); } $result = DB::insert($this->_table_name) ->columns(array_keys($data)) ->values(array_values($data)) ->execute($this->_db); if ( ! array_key_exists($this->_primary_key, $data)) { // Load the insert id as the primary key if it was left out $this->_object[$this->_primary_key] = $this->_primary_key_value = $result[0]; } else { $this->_primary_key_value = $this->_object[$this->_primary_key]; } // Object is now loaded and saved $this->_loaded = $this->_saved = TRUE; // All changes have been saved $this->_changed = array(); $this->_original_values = $this->_object; return $this; } /** * Updates a single record or multiple records * * @chainable * @param Validation $validation Validation object * @throws Kohana_Exception * @return ORM */ public function update(Validation $validation = NULL) { if ( ! $this->_loaded) throw new Kohana_Exception('Cannot update :model model because it is not loaded.', array(':model' => $this->_object_name)); // Run validation if the model isn't valid or we have additional validation rules. if ( ! $this->_valid OR $validation) { $this->check($validation); } if (empty($this->_changed)) { // Nothing to update return $this; } $data = array(); foreach ($this->_changed as $column) { // Compile changed data $data[$column] = $this->_object[$column]; } if (is_array($this->_updated_column)) { // Fill the updated column $column = $this->_updated_column['column']; $format = $this->_updated_column['format']; $data[$column] = $this->_object[$column] = ($format === TRUE) ? time() : date($format); } // Use primary key value $id = $this->pk(); // Update a single record DB::update($this->_table_name) ->set($data) ->where($this->_primary_key, '=', $id) ->execute($this->_db); if (isset($data[$this->_primary_key])) { // Primary key was changed, reflect it $this->_primary_key_value = $data[$this->_primary_key]; } // Object has been saved $this->_saved = TRUE; // All changes have been saved $this->_changed = array(); $this->_original_values = $this->_object; return $this; } /** * Updates or Creates the record depending on loaded() * * @chainable * @param Validation $validation Validation object * @return ORM */ public function save(Validation $validation = NULL) { return $this->loaded() ? $this->update($validation) : $this->create($validation); } /** * Deletes a single record while ignoring relationships. * * @chainable * @throws Kohana_Exception * @return ORM */ public function delete() { if ( ! $this->_loaded) throw new Kohana_Exception('Cannot delete :model model because it is not loaded.', array(':model' => $this->_object_name)); // Use primary key value $id = $this->pk(); // Delete the object DB::delete($this->_table_name) ->where($this->_primary_key, '=', $id) ->execute($this->_db); return $this->clear(); } /** * Tests if this object has a relationship to a different model, * or an array of different models. When providing far keys, the number * of relations must equal the number of keys. * * * // Check if $model has the login role * $model->has('roles', ORM::factory('role', array('name' => 'login'))); * // Check for the login role if you know the roles.id is 5 * $model->has('roles', 5); * // Check for all of the following roles * $model->has('roles', array(1, 2, 3, 4)); * // Check if $model has any roles * $model->has('roles') * * @param string $alias Alias of the has_many "through" relationship * @param mixed $far_keys Related model, primary key, or an array of primary keys * @return boolean */ public function has($alias, $far_keys = NULL) { $count = $this->count_relations($alias, $far_keys); if ($far_keys === NULL) { return (bool) $count; } else { return $count === count($far_keys); } } /** * Tests if this object has a relationship to a different model, * or an array of different models. When providing far keys, this function * only checks that at least one of the relationships is satisfied. * * // Check if $model has the login role * $model->has('roles', ORM::factory('role', array('name' => 'login'))); * // Check for the login role if you know the roles.id is 5 * $model->has('roles', 5); * // Check for any of the following roles * $model->has('roles', array(1, 2, 3, 4)); * // Check if $model has any roles * $model->has('roles') * * @param string $alias Alias of the has_many "through" relationship * @param mixed $far_keys Related model, primary key, or an array of primary keys * @return boolean */ public function has_any($alias, $far_keys = NULL) { return (bool) $this->count_relations($alias, $far_keys); } /** * Returns the number of relationships * * // Counts the number of times the login role is attached to $model * $model->count_relations('roles', ORM::factory('role', array('name' => 'login'))); * // Counts the number of times role 5 is attached to $model * $model->count_relations('roles', 5); * // Counts the number of times any of roles 1, 2, 3, or 4 are attached to * // $model * $model->count_relations('roles', array(1, 2, 3, 4)); * // Counts the number roles attached to $model * $model->count_relations('roles') * * @param string $alias Alias of the has_many "through" relationship * @param mixed $far_keys Related model, primary key, or an array of primary keys * @return integer */ public function count_relations($alias, $far_keys = NULL) { if ($far_keys === NULL) { return (int) DB::select(array(DB::expr('COUNT(*)'), 'records_found')) ->from($this->_has_many[$alias]['through']) ->where($this->_has_many[$alias]['foreign_key'], '=', $this->pk()) ->execute($this->_db)->get('records_found'); } $far_keys = ($far_keys instanceof ORM) ? $far_keys->pk() : $far_keys; // We need an array to simplify the logic $far_keys = (array) $far_keys; // Nothing to check if the model isn't loaded or we don't have any far_keys if ( ! $far_keys OR ! $this->_loaded) return 0; $count = (int) DB::select(array(DB::expr('COUNT(*)'), 'records_found')) ->from($this->_has_many[$alias]['through']) ->where($this->_has_many[$alias]['foreign_key'], '=', $this->pk()) ->where($this->_has_many[$alias]['far_key'], 'IN', $far_keys) ->execute($this->_db)->get('records_found'); // Rows found need to match the rows searched return (int) $count; } /** * Adds a new relationship to between this model and another. * * // Add the login role using a model instance * $model->add('roles', ORM::factory('role', array('name' => 'login'))); * // Add the login role if you know the roles.id is 5 * $model->add('roles', 5); * // Add multiple roles (for example, from checkboxes on a form) * $model->add('roles', array(1, 2, 3, 4)); * * @param string $alias Alias of the has_many "through" relationship * @param mixed $far_keys Related model, primary key, or an array of primary keys * @return ORM */ public function add($alias, $far_keys) { $far_keys = ($far_keys instanceof ORM) ? $far_keys->pk() : $far_keys; $columns = array($this->_has_many[$alias]['foreign_key'], $this->_has_many[$alias]['far_key']); $foreign_key = $this->pk(); $query = DB::insert($this->_has_many[$alias]['through'], $columns); foreach ( (array) $far_keys as $key) { $query->values(array($foreign_key, $key)); } $query->execute($this->_db); return $this; } /** * Removes a relationship between this model and another. * * // Remove a role using a model instance * $model->remove('roles', ORM::factory('role', array('name' => 'login'))); * // Remove the role knowing the primary key * $model->remove('roles', 5); * // Remove multiple roles (for example, from checkboxes on a form) * $model->remove('roles', array(1, 2, 3, 4)); * // Remove all related roles * $model->remove('roles'); * * @param string $alias Alias of the has_many "through" relationship * @param mixed $far_keys Related model, primary key, or an array of primary keys * @return ORM */ public function remove($alias, $far_keys = NULL) { $far_keys = ($far_keys instanceof ORM) ? $far_keys->pk() : $far_keys; $query = DB::delete($this->_has_many[$alias]['through']) ->where($this->_has_many[$alias]['foreign_key'], '=', $this->pk()); if ($far_keys !== NULL) { // Remove all the relationships in the array $query->where($this->_has_many[$alias]['far_key'], 'IN', (array) $far_keys); } $query->execute($this->_db); return $this; } /** * Count the number of records in the table. * * @return integer */ public function count_all() { $selects = array(); foreach ($this->_db_pending as $key => $method) { if ($method['name'] == 'select') { // Ignore any selected columns for now $selects[] = $method; unset($this->_db_pending[$key]); } } if ( ! empty($this->_load_with)) { foreach ($this->_load_with as $alias) { // Bind relationship $this->with($alias); } } $this->_build(Database::SELECT); $records = $this->_db_builder->from(array($this->_table_name, $this->_object_name)) ->select(array(DB::expr('COUNT('.$this->_db->quote_column($this->_object_name.'.'.$this->_primary_key).')'), 'records_found')) ->execute($this->_db) ->get('records_found'); // Add back in selected columns $this->_db_pending += $selects; $this->reset(); // Return the total number of records in a table return (int) $records; } /** * Proxy method to Database list_columns. * * @return array */ public function list_columns() { // Proxy to database return $this->_db->list_columns($this->_table_name); } /** * Returns an ORM model for the given one-one related alias * * @param string $alias Alias name * @return ORM */ protected function _related($alias) { if (isset($this->_related[$alias])) { return $this->_related[$alias]; } elseif (isset($this->_has_one[$alias])) { return $this->_related[$alias] = ORM::factory($this->_has_one[$alias]['model']); } elseif (isset($this->_belongs_to[$alias])) { return $this->_related[$alias] = ORM::factory($this->_belongs_to[$alias]['model']); } else { return FALSE; } } /** * Returns the value of the primary key * * @return mixed Primary key */ public function pk() { return $this->_primary_key_value; } /** * Returns last executed query * * @return string */ public function last_query() { return $this->_db->last_query; } /** * Clears query builder. Passing FALSE is useful to keep the existing * query conditions for another query. * * @param bool $next Pass FALSE to avoid resetting on the next call * @return ORM */ public function reset($next = TRUE) { if ($next AND $this->_db_reset) { $this->_db_pending = array(); $this->_db_applied = array(); $this->_db_builder = NULL; $this->_with_applied = array(); } // Reset on the next call? $this->_db_reset = $next; return $this; } protected function _serialize_value($value) { return json_encode($value); } protected function _unserialize_value($value) { return json_decode($value, TRUE); } public function object_name() { return $this->_object_name; } public function object_plural() { return $this->_object_plural; } public function loaded() { return $this->_loaded; } public function saved() { return $this->_saved; } public function primary_key() { return $this->_primary_key; } public function table_name() { return $this->_table_name; } public function table_columns() { return $this->_table_columns; } public function has_one() { return $this->_has_one; } public function belongs_to() { return $this->_belongs_to; } public function has_many() { return $this->_has_many; } public function load_with() { return $this->_load_with; } public function original_values() { return $this->_original_values; } public function created_column() { return $this->_created_column; } public function updated_column() { return $this->_updated_column; } public function validation() { if ( ! isset($this->_validation)) { // Initialize the validation object $this->_validation(); } return $this->_validation; } public function object() { return $this->_object; } public function errors_filename() { return $this->_errors_filename; } /** * Alias of and_where() * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function where($column, $op, $value) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'where', 'args' => array($column, $op, $value), ); return $this; } /** * Creates a new "AND WHERE" condition for the query. * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function and_where($column, $op, $value) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_where', 'args' => array($column, $op, $value), ); return $this; } /** * Creates a new "OR WHERE" condition for the query. * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function or_where($column, $op, $value) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_where', 'args' => array($column, $op, $value), ); return $this; } /** * Alias of and_where_open() * * @return $this */ public function where_open() { return $this->and_where_open(); } /** * Opens a new "AND WHERE (...)" grouping. * * @return $this */ public function and_where_open() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_where_open', 'args' => array(), ); return $this; } /** * Opens a new "OR WHERE (...)" grouping. * * @return $this */ public function or_where_open() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_where_open', 'args' => array(), ); return $this; } /** * Closes an open "AND WHERE (...)" grouping. * * @return $this */ public function where_close() { return $this->and_where_close(); } /** * Closes an open "AND WHERE (...)" grouping. * * @return $this */ public function and_where_close() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_where_close', 'args' => array(), ); return $this; } /** * Closes an open "OR WHERE (...)" grouping. * * @return $this */ public function or_where_close() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_where_close', 'args' => array(), ); return $this; } /** * Applies sorting with "ORDER BY ..." * * @param mixed $column column name or array($column, $alias) or object * @param string $direction direction of sorting * @return $this */ public function order_by($column, $direction = NULL) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'order_by', 'args' => array($column, $direction), ); return $this; } /** * Return up to "LIMIT ..." results * * @param integer $number maximum results to return * @return $this */ public function limit($number) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'limit', 'args' => array($number), ); return $this; } /** * Enables or disables selecting only unique columns using "SELECT DISTINCT" * * @param boolean $value enable or disable distinct columns * @return $this */ public function distinct($value) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'distinct', 'args' => array($value), ); return $this; } /** * Choose the columns to select from. * * @param mixed $columns column name or array($column, $alias) or object * @param ... * @return $this */ public function select($columns = NULL) { $columns = func_get_args(); // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'select', 'args' => $columns, ); return $this; } /** * Choose the tables to select "FROM ..." * * @param mixed $tables table name or array($table, $alias) or object * @param ... * @return $this */ public function from($tables) { $tables = func_get_args(); // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'from', 'args' => $tables, ); return $this; } /** * Adds addition tables to "JOIN ...". * * @param mixed $table column name or array($column, $alias) or object * @param string $type join type (LEFT, RIGHT, INNER, etc) * @return $this */ public function join($table, $type = NULL) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'join', 'args' => array($table, $type), ); return $this; } /** * Adds "ON ..." conditions for the last created JOIN statement. * * @param mixed $c1 column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $c2 column name or array($column, $alias) or object * @return $this */ public function on($c1, $op, $c2) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'on', 'args' => array($c1, $op, $c2), ); return $this; } /** * Creates a "GROUP BY ..." filter. * * @param mixed $columns column name or array($column, $alias) or object * @param ... * @return $this */ public function group_by($columns) { $columns = func_get_args(); // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'group_by', 'args' => $columns, ); return $this; } /** * Alias of and_having() * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function having($column, $op, $value = NULL) { return $this->and_having($column, $op, $value); } /** * Creates a new "AND HAVING" condition for the query. * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function and_having($column, $op, $value = NULL) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_having', 'args' => array($column, $op, $value), ); return $this; } /** * Creates a new "OR HAVING" condition for the query. * * @param mixed $column column name or array($column, $alias) or object * @param string $op logic operator * @param mixed $value column value * @return $this */ public function or_having($column, $op, $value = NULL) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_having', 'args' => array($column, $op, $value), ); return $this; } /** * Alias of and_having_open() * * @return $this */ public function having_open() { return $this->and_having_open(); } /** * Opens a new "AND HAVING (...)" grouping. * * @return $this */ public function and_having_open() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_having_open', 'args' => array(), ); return $this; } /** * Opens a new "OR HAVING (...)" grouping. * * @return $this */ public function or_having_open() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_having_open', 'args' => array(), ); return $this; } /** * Closes an open "AND HAVING (...)" grouping. * * @return $this */ public function having_close() { return $this->and_having_close(); } /** * Closes an open "AND HAVING (...)" grouping. * * @return $this */ public function and_having_close() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'and_having_close', 'args' => array(), ); return $this; } /** * Closes an open "OR HAVING (...)" grouping. * * @return $this */ public function or_having_close() { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'or_having_close', 'args' => array(), ); return $this; } /** * Start returning results after "OFFSET ..." * * @param integer $number starting result number * @return $this */ public function offset($number) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'offset', 'args' => array($number), ); return $this; } /** * Enables the query to be cached for a specified amount of time. * * @param integer $lifetime number of seconds to cache * @return $this * @uses Kohana::$cache_life */ public function cached($lifetime = NULL) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'cached', 'args' => array($lifetime), ); return $this; } /** * Set the value of a parameter in the query. * * @param string $param parameter key to replace * @param mixed $value value to use * @return $this */ public function param($param, $value) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'param', 'args' => array($param, $value), ); return $this; } /** * Adds "USING ..." conditions for the last created JOIN statement. * * @param string $columns column name * @return $this */ public function using($columns) { // Add pending database call which is executed after query type is determined $this->_db_pending[] = array( 'name' => 'using', 'args' => array($columns), ); return $this; } /** * Checks whether a column value is unique. * Excludes itself if loaded. * * @param string $field the field to check for uniqueness * @param mixed $value the value to check for uniqueness * @return bool whteher the value is unique */ public function unique($field, $value) { $model = ORM::factory($this->object_name()) ->where($field, '=', $value) ->find(); if ($this->loaded()) { return ( ! ($model->loaded() AND $model->pk() != $this->pk())); } return ( ! $model->loaded()); } } // End ORM