This repository has been archived on 2024-10-25. You can view files and clone it, but cannot push or open issues or pull requests.
2013-10-02 11:14:53 -05:00

1030 lines
31 KiB
PHP
Executable File

<?php
namespace Rails\ActiveRecord;
use PDO;
use Rails;
use Rails\ActiveRecord\ActiveRecord;
use Rails\ActiveRecord\Base\Methods;
use Rails\ActiveRecord\Validation\Validator;
use Rails\ActiveModel;
abstract class Base
{
use Methods\CounterMethods, Methods\RelationMethods, Methods\ScopingMethods,
Methods\AttributeMethods, Methods\ModelSchemaMethods;
/**
* ActiveRecord_Registry instance.
*/
static private $registry;
/**
* Instance of Table for each model.
*/
static private $tables = [];
static private $cachedReflections = [];
/**
* Flag to prevent calling init() for some cases.
*/
static private $preventInit = false;
/**
* ActiveModel\Errors instance.
*/
private $errors;
/**
* @var bool
*/
private $isNewRecord = true;
/**
* An array where loaded associations will
* be stored.
*/
private $loadedAssociations = [];
static public function __callStatic($method, $params)
{
if ($rel = static::scope($method, $params))
return $rel;
throw new Exception\BadMethodCallException(
sprintf("Call to undefined static method %s::%s", get_called_class(), $method)
);
}
static public function create(array $attrs, array $options = [])
{
$new_model = new static();
$new_model->assignAttributes($attrs, $options);
$new_model->_create_do();
return $new_model;
}
/**
* Can receive parameters that will be directly passed to where();
*/
static public function destroyAll()
{
$models = call_user_func_array(get_called_class() . '::where', func_get_args())->take();
foreach ($models as $m) {
$m->destroy();
}
return $models;
}
# TODO
static public function deleteAll(array $conds)
{
}
static public function update($id, array $attrs)
{
$attrs_str = [];
foreach (array_keys($attrs) as $attr)
$attrs_str[] = '`'.$attr.'` = ?';
$sql = "UPDATE `" . static::tableName() . "` SET " . implode(', ', $attrs_str) . " WHERE id = ?";
array_unshift($attrs, $sql);
$attrs[] = $id;
static::connection()->executeSql($attrs);
}
/**
* Searches if record exists by id.
*
* How to use:
* Model::exists(1);
* Model::exists(1, 2, 3);
* Model::exists([1, 2, 3]);
*/
static public function exists($params)
{
$query = self::none();
if (ctype_digit((string)$params)) {
$query->where(['id' => $params]);
} else {
if (is_array($params))
$query->where('id IN (?)', $params);
else
$query->where('id IN (?)', func_get_args());
}
return $query->exists();
}
static public function countBySql()
{
$stmt = call_user_func_array([static::connection(), 'executeSql'], func_get_args());
if (ActiveRecord::lastError())
return false;
$rows = $stmt->fetchAll(PDO::FETCH_NUM);
if (isset($rows[0][0]))
return (int)$rows[0][0];
else
return false;
}
static public function maximum($attr)
{
return self::connection()->selectValue('SELECT MAX(' . $attr . ') FROM ' . static::tableName());
}
static public function I18n()
{
return Rails::application()->I18n();
}
static public function transaction($params = [], \Closure $block = null)
{
self::connection()->transaction($params, $block);
}
static public function connectionName()
{
return null;
}
/**
* If called class specifies a connection, such connection will be returned.
*/
static public function connection()
{
$name = static::connectionName();
return ActiveRecord::connection($name);
}
/**
* Public properties declared in a model class can be used to avoid defining
* setter and getter methods; the value will be set/get directly to/from it.
*/
static public function hasPublicProperty($propName)
{
$reflection = self::getReflection();
return $reflection->hasProperty($propName) && $reflection->getProperty($propName)->isPublic();
}
/**
* It is sometimes required an empty collection.
*/
static public function emptyCollection()
{
return new Collection();
}
static protected function cn()
{
return get_called_class();
}
static protected function _registry()
{
if (!self::$registry)
self::$registry = new Registry();
return self::$registry->model(get_called_class());
}
/**
* Creates and returns a non-empty model.
* This function is useful to centralize the creation of non-empty models,
* since isNewRecord must set to null by passing non_empty.
*/
static private function _create_model(array $data)
{
self::$preventInit = true;
$model = new static();
$model->attributes = $data;
# Check for primary_key and set init_attrs.
if (!static::table()->primaryKey()) {
$model->storedAttributes = $data;
}
$model->isNewRecord = false;
$model->_register();
$model->init();
return $model;
}
static private function _create_collection(array $data, $query = null)
{
$models = array();
foreach ($data as $d)
$models[] = self::_create_model($d);
$coll = new Collection($models, $query);
return $coll;
}
static private function getReflection($class = null)
{
if (!$class) {
$class = get_called_class();
}
if (!isset(self::$cachedReflections[$class])) {
self::$cachedReflections[$class] = new \ReflectionClass($class);
}
return self::$cachedReflections[$class];
}
public function __construct(array $attrs = [])
{
$this->assignAttributes($attrs);
if (!self::$preventInit) {
$this->init();
} else {
self::$preventInit = true;
}
}
/**
* Checks for the called property in this order
* 1. Checks if property is an attribute.
* 2. Checks if called property is an already loaded association.
* 3. Checks if called property is an association.
* If none if these validations is true, the property is actually called
* in order to generate an error.
*/
public function __get($prop)
{
# See Config/default_config.php for more info.
// if (!Rails::config()->ar2) {
if (static::isAttribute($prop)) {
return $this->getAttribute($prop);
} elseif ($this->getAssociation($prop) !== null) {
return $this->loadedAssociations[$prop];
}
throw new Exception\RuntimeException(
sprintf("Tried to get unknown property %s::$%s", get_called_class(), $prop)
);
// }
// # Force error/default behaviour
// return $this->$prop;
}
/**
* 1. Checks for setter method and calls it if available.
* 2. If the property is an attribute, the value is set to it (i.e. default setter for attributes).
* 3. The value is set as object property. < This shouldn't happen!
*/
public function __set($prop, $val)
{
// if (!Rails::config()->ar2) {
if (static::isAttribute($prop)) {
$this->setAttribute($prop, $val);
// } elseif ($this->setterExists($prop)) {
// $this->_run_setter($prop, $val);
} else {
throw new Exception\RuntimeException(
sprintf("Trying to set undefined property %s", $prop)
);
// # Default PHP behaviour.
// $this->$prop = $val;
}
// } else {
// if (static::isAttribute($prop)) {
// $this->setAttribute($prop, $val);
// } else {
// throw new Exception\RuntimeException(
// sprintf("Trying to set undefined property %s", $prop)
// );
// }
// }
}
public function __isset($prop)
{
return $this->issetAttribute($prop);
}
/**
* Some model's features can be overloaded:
* 1. Check if an attribute changed with attributeNameChanged();
* Calling this normally would be: attributeChanged('attribute_name');
* 2. Get the previous value of an attribute with attributeNameWas();
* Normally: attributeWas('attribute_name');
* 3. Overload attribute setters. Models *should* define setter
* methods for each attribute, but they can be called overloadingly. The correct
* or expected name of the setter method is camel-cased, like setAttributeName();
* 4. Overload scopes.
* 5. Overload associations.
* 6. Overload attribute getters. The expected method name is the camel-cased name of
* the attribute, like attributeName(), because the "get" prefix is omitted for getters.
* Models *should* define getter methods for each attribute.
*/
public function __call($method, $params)
{
# Overload attributeChange()
if (strlen($method) > 7 && substr($method, -7) == 'Changed') {
$attribute = static::properAttrName(substr($method, 0, -7));
return $this->attributeChanged($attribute);
}
# Overload attributeWas()
if (strlen($method) > 3 && substr($method, -3) == 'Was') {
$attribute = static::properAttrName(substr($method, 0, -3));
return $this->attributeWas($attribute);
}
# Overload attribute setter.
if (substr($method, 0, 3) == 'set') {
$attrName = substr($method, 3);
$underscored = Rails::services()->get('inflector')->underscore($method);
if (static::isAttribute($underscored)) {
$this->setAttribute($underscored, array_shift($params));
return;
}
}
# Overload scopes.
if ($rel = static::scope($method, $params)) {
return $rel;
}
// # Overload associations.
// if ($this->getAssociation($method) !== null) {
// return $this->loadedAssociations[$method];
// }
// # Overload attributes.
// $underscored = Rails::services()->get('inflector')->underscore($method);
// if (static::isAttribute($underscored)) {
// return $this->getAttribute($underscored);
// }
throw new Exception\BadMethodCallException(
sprintf("Call to undefined method %s::%s", get_called_class(), $method)
);
}
public function isNewRecord()
{
return $this->isNewRecord;
}
public function updateAttributes(array $attrs)
{
$this->assignAttributes($attrs);
$this->runCallbacks('before_update');
/**
* iTODO: Must let know save() we're updating, so it will
* validate data with action "update" and not "save".
* Should separate save() and make this and update_attribute call
* something like update()?
*/
if ($this->save(['action' => 'update'])) {
$this->runCallbacks('after_update');
return true;
}
return false;
}
/**
* Directly passes the new value to the attributes array and then
* saves the record.
*
* @param string $attrName The name of the attribute.
* @param mixed $value The value for the attribute,
*/
public function updateAttribute($attrName, $value)
{
$this->attributes[$attrName] = $value;
return $this->save(['skip_validation' => true, 'skip_callbacks' => true, 'action' => 'update']);
}
/**
* Save record
*
* Saves in the database the properties of the object that match
* the columns of the corresponding table in the database.
*
* Is the model is new, create() will be called instead.
*
* @array $values: If present, object will be updated according
* to this array, otherwise, according to its properties.
*/
public function save(array $opts = array())
{
// if (empty($opts['skip_validation'])) {
// if (!$this->_validate_data(!empty($opts['action']) ? $opts['action'] : 'save'))
// return false;
// }
if ($this->isNewRecord()) {
if (!$this->_create_do($opts))
return false;
} else {
if (!$this->_validate_data('save'))
return false;
if (empty($opts['skip_callbacks'])) {
if (!$this->runCallbacks('before_save'))
return false;
}
if (!$this->_save_do($opts))
return false;
if (empty($opts['skip_callbacks'])) {
$this->runCallbacks('after_save');
}
}
return true;
}
/**
* Delete
*
* Deletes row from database based on Primary keys.
*/
static public function delete()
{
// if ($this->_delete_from_db('delete')) {
// foreach (array_keys(get_object_vars($this)) as $p)
// unset($this->$p);
// return true;
// }
// return false;
}
# Deletes current model from database but keeps model's properties.
public function destroy()
{
return $this->_delete_from_db('destroy');
}
public function errors()
{
if (!$this->errors)
$this->errors = new ActiveModel\Errors();
return $this->errors;
}
public function reload()
{
try {
$data = $this->_get_stored_data();
} catch (Exception\ExceptionInterface $e) {
return false;
}
$cn = get_called_class();
$refl = new \ReflectionClass($cn);
$reflProps = $refl->getProperties(\ReflectionProperty::IS_PUBLIC | \ReflectionProperty::IS_PROTECTED);
$defProps = $refl->getDefaultProperties();
foreach ($reflProps as $reflProp) {
if ($reflProp->getDeclaringClass() == $cn && !$reflProp->isStatic()) {
$this->{$reflProp->name} = $defProps[$reflProp->name];
}
}
$this->isNewRecord = false;
$this->attributes = $data->attributes();
// $this->init();
return true;
}
public function asJson()
{
return $this->attributes();
}
public function toJson()
{
return json_encode($this->asJson());
}
# TODO:
public function toXml(array $params = [])
{
if (!isset($params['attributes'])) {
// $this->_merge_model_attributes();
$attrs = $this->attributes();
} else {
$attrs = $params['attributes'];
}
!isset($params['root']) && $params['root'] = strtolower(str_replace('_', '-', self::cn()));
if (!isset($params['builder'])) {
$xml = new \Rails\Xml\Xml($attrs, $params);
return $xml->output();
} else {
$builder = $params['builder'];
unset($params['builder']);
$builder->build($attrs, $params);
}
}
public function getAssociation($name)
{
if (isset($this->loadedAssociations[$name])) {
return $this->loadedAssociations[$name];
} elseif ($assoc = $this->get_association_data($name)) {
$model = $this->_load_association($name, $assoc[0], $assoc[1]);
$this->loadedAssociations[$name] = $model;
return $this->loadedAssociations[$name];
}
}
/**
* ***************************
* Default protected methods {
* ***************************
* attrAccessible and attrProtected can be found in Base\Methods\AttributeMethods.
*/
/**
* Code executed when initializing the object.
* Called by _create_model() and _create_do().
*/
protected function init()
{
}
protected function associations()
{
return [];
}
/**
* Example:
*
* return [
* // Validate attributes
* 'attribute_name' => [
* 'property' => rules...
* ],
*
* // Passing a value or an index that isn't recognized as
* // attribute will be considered as custom validation method.
* 'methodName',
* 'method2Name' => [ 'on' => [ actions... ] ],
* ];
*
* Custom validation methods must set the error manually. They aren't expected to
* return anything.
*/
protected function validations()
{
return [];
}
protected function callbacks()
{
return [];
}
/* } */
/**
* Returns model's current data in the database or using storedAttributes.
*/
protected function _get_stored_data()
{
if (!($prim_key = static::table()->primaryKey()) && !$this->storedAttributes) {
throw new Exception\RuntimeException(
"Can't find data without primary key nor storedAttributes"
);
}
$query = static::none();
if ($prim_key) {
$query->where('`'.$prim_key.'` = ?', $this->$prim_key);
} else {
$model = new static();
$cols_names = static::table()->columnNames();
foreach ($this->storedAttributes as $name => $value) {
if (in_array($name, $cols_names)) {
$model->attributes[$name] = $value;
}
}
if (!$model->attributes)
throw new Exception\RuntimeException(
"Model rebuilt from storedAttributes failed"
);
else
return $model;
}
$current = $query->first();
if (!$current)
throw new Exception\RuntimeException(
"Row not found in database (searched with storedAttributes)"
);
else
return $current;
}
protected function runCallbacks($callback_name)
{
$callbacks = array();
$tmp = $this->callbacks();
if (isset($tmp[$callback_name])) {
$callbacks = $tmp[$callback_name];
}
$callbacks = array_unique(array_filter(array_merge($callbacks, $this->_get_parents_callbacks($callback_name))));
if ($callbacks) {
foreach ($callbacks as $method) {
if (false === $this->$method()) {
return false;
}
}
}
return true;
}
/**
* @param array|Closure $params - Additional parameters to customize the query for the association
*/
private function _load_association($prop, $type, $params)
{
return $this->{'_find_' . $type}($prop, $params);
}
private function _get_parents_callbacks($callback_name)
{
$all_callbacks = array();
if (($class = get_parent_class($this)) != 'Rails\ActiveRecord\Base') {
$class = self::cn();
$obj = new $class();
while (($class = get_parent_class($obj)) != 'Rails\ActiveRecord\Base') {
$obj = new $class();
if ($callbacks = $obj->callbacks()) {
if (isset($callbacks[$callback_name]))
$all_callbacks = array_merge($callbacks, $callbacks[$callback_name]);
}
}
}
return $all_callbacks;
}
# Returns association property names.
private function _associations_names()
{
$associations = array();
foreach ($this->associations() as $assocs) {
foreach ($assocs as $k => $v)
$associations[] = is_int($k) ? $v : $k;
}
return $associations;
}
/**
* @return bool
* @see validations()
*/
private function _validate_data($action)
{
if (!$this->runCallbacks('before_validation'))
return false;
$validation_success = true;
$modelClass = get_called_class();
$classProps = get_class_vars($modelClass);
foreach ($this->validations() as $attrName => $validations) {
/**
* This should only happen when passing a custom validation method with
* no validation options.
*/
if (is_int($attrName)) {
$attrName = $validations;
$validations = [];
}
if (static::isAttribute($attrName) || array_key_exists($attrName, $classProps)) {
foreach ($validations as $type => $params) {
if (!is_array($params)) {
$params = [$params];
}
if ($modelClass::isAttribute($attrName)) {
$value = $this->getAttribute($attrName);
} else {
$value = $this->$attrName;
}
$validation = new Validator($type, $value, $params);
$validation->set_params($action, $this, $attrName);
if (!$validation->validate()->success()) {
$validation->set_error_message();
$validation_success = false;
}
}
} else {
/**
* The attrName passed isn't an attribute nor a property, so we assume it's a method.
*
* $attrName becomes the name of the method.
* $validations becomes validation options.
*/
if (!empty($validations['on']) && !in_array($action, $validations['on'])) {
continue;
}
// $this->getAttribute($attrName);
$this->$attrName();
if ($this->errors()->any()) {
$validation_success = false;
}
}
}
return $validation_success;
}
private function _create_do()
{
if (!$this->runCallbacks('before_validation_on_create')) {
return false;
} elseif (!$this->_validate_data('create')) {
return false;
}
$this->runCallbacks('after_validation_on_create');
if (!$this->runCallbacks('before_save'))
return false;
if (!$this->runCallbacks('before_create'))
return false;
$this->_check_time_column('created_at');
$this->_check_time_column('updated_at');
$this->_check_time_column('created_on');
$this->_check_time_column('updated_on');
$cols_values = $cols_names = array();
// $this->_merge_model_attributes();
foreach ($this->attributes() as $attr => $val) {
$proper = static::properAttrName($attr);
if (!static::table()->columnExists($proper)) {
continue;
}
$cols_names[] = '`'.$attr.'`';
$cols_values[] = $val;
$init_attrs[$attr] = $val;
}
if (!$cols_values)
return false;
$binding_marks = implode(', ', array_fill(0, (count($cols_names)), '?'));
$cols_names = implode(', ', $cols_names);
$sql = 'INSERT INTO `'.static::tableName().'` ('.$cols_names.') VALUES ('.$binding_marks.')';
array_unshift($cols_values, $sql);
static::connection()->executeSql($cols_values);
$id = static::connection()->lastInsertId();
$primary_key = static::table()->primaryKey();
if ($primary_key && count($primary_key) == 1) {
if (!$id) {
$this->errors()->addToBase('Couldn\'t retrieve new primary key.');
return false;
}
if ($pri_key = static::table()->primaryKey()) {
$this->setAttribute($pri_key, $id);
}
} else {
$this->storedAttributes = $init_attrs;
}
$this->isNewRecord = false;
// $this->init();
$this->runCallbacks('after_create');
$this->runCallbacks('after_save');
return true;
}
private function _save_do()
{
$w = $wd = $q = $d = array();
$dt = static::table()->columnNames();
try {
$current = $this->_get_stored_data();
} catch (Exception\ExceptionInterface $e) {
$this->errors()->addToBase($e->getMessage());
return;
}
$has_primary_keys = false;
foreach (static::table()->indexes() as $idx) {
$w[] = '`'.$idx.'` = ?';
$wd[] = $current->$idx;
}
if ($w) {
$has_primary_keys = true;
}
// $this->_merge_model_attributes();
foreach ($this->attributes() as $prop => $val) {
# Can't update properties that don't have a column in DB, or
# PRImary keys, or time columns.
if (!in_array($prop, $dt) || $prop == 'created_at' || $prop == 'updated_at' ||
$prop == 'created_on' || $prop == 'updated_on'
) {
continue;
} elseif (!$has_primary_keys) {
$w[] = '`'.$prop.'` = ?';
$wd[] = $current->$prop;
// foreach ($current->attributes() as $
}
// } elseif (!$has_primary_keys && $val === $current->$prop) {
// $w[] = '`'.$prop.'` = ?';
// $wd[] = $current->$prop;
// } else
if ($val != $current->$prop) {
$this->setChangedAttribute($prop, $current->$prop);
$q[] = '`'.$prop.'` = ?';
$d[] = $val;
}
}
# Update `updated_at|on` if exists.
if ($this->_check_time_column('updated_at')) {
$q[] = "`updated_at` = ?";
$d[] = $this->updated_at;
} elseif ($this->_check_time_column('updated_on')) {
$q[] = "`updated_on` = ?";
$d[] = $this->updated_on;
}
if ($q) {
$q = "UPDATE `" . static::tableName() . "` SET " . implode(', ', $q);
$w && $q .= ' WHERE '.implode(' AND ', $w);
$q .= ' LIMIT 1';
$d = array_merge($d, $wd);
array_unshift($d, $q);
static::connection()->executeSql($d);
if (ActiveRecord::lastError()) {
$this->errors()->addToBase(ActiveRecord::lastError());
return false;
}
}
$this->_update_init_attrs();
return true;
}
private function _delete_from_db($type)
{
if (!$this->runCallbacks('before_'.$type)) {
return false;
}
$w = $wd = [];
if ($keys = self::table()->indexes()) {
foreach ($keys as $k) {
$w[] = '`' . static::tableName() . '`.`'.$k.'` = ?';
$wd[] = $this->$k;
}
} elseif ($this->storedAttributes) {
foreach ($this->storedAttributes as $attr => $val) {
$w[] = '`'.$attr.'` = ?';
$wd[] = $val;
}
} else {
throw new Exception\LogicException(
"Can't delete model without attributes"
);
}
$w = implode(' AND ', $w);
$query = 'DELETE FROM `' . static::tableName() . '` WHERE '.$w;
array_unshift($wd, $query);
static::connection()->executeSql($wd);
$this->runCallbacks('after_'.$type);
return true;
}
private function get_association_data($prop)
{
if ($assocs = $this->associations()) {
foreach ($assocs as $type => $assoc) {
foreach ($assoc as $name => $params) {
if (is_int($name)) {
$name = $params;
$params = array();
}
if ($name == $prop) {
return array($type, $params);
}
}
}
}
return false;
}
private function _register()
{
self::_registry()->register($this);
}
/**
* Check time column
*
* Called by save() and create(), checks if $column
* exists and automatically sets a value to it.
*/
private function _check_time_column($column)
{
if (!static::table()->columnExists($column))
return false;
$type = static::table()->columnType($column);
if ($type == 'datetime')
$time = date('Y-m-d H:i:s');
elseif ($type == 'year')
$time = date('Y');
elseif ($type == 'date')
$time = date('Y-m-d');
elseif ($type == 'time')
$time = date('H:i:s');
elseif ($type == 'timestamp')
$time = time();
else
return false;
$this->attributes[$column] = $time;
return true;
}
private function _update_init_attrs()
{
foreach (array_keys($this->storedAttributes) as $name) {
if (isset($this->attributes[$name]))
$this->storedAttributes[$name] = $this->attributes[$name];
}
}
private function _getter_for($prop)
{
$camelized = Rails::services()->get('inflector')->camelize($prop);
$method = 'get' . ucfirst($camelized);
if (method_exists($this, $method)) {
return $this->$method();
} elseif (method_exists($this, $camelized)) {
return $this->$camelized();
}
return null;
}
private function setterExists($attrName)
{
$inflector = Rails::services()->get('inflector');
$reflection = self::getReflection();
$setter = 'set' . $inflector->camelize($attrName);
if ($reflection->hasMethod($setter) && $reflection->getMethod($setter)->isPublic()) {
return $setter;
} else {
false;
}
}
}