1
0
mirror of https://github.com/e107inc/e107.git synced 2025-01-18 13:14:55 +01:00
php-e107/e107_handlers/model_class.php

3780 lines
87 KiB
PHP
Raw Normal View History

2009-08-20 13:06:34 +00:00
<?php
/*
* e107 website system
*
2010-12-29 13:39:15 +00:00
* Copyright (C) 2008-2010 e107 Inc (e107.org)
2009-08-20 13:06:34 +00:00
* Released under the terms and conditions of the
* GNU General Public License (http://www.gnu.org/licenses/gpl.txt)
*
* e107 Base Model
*
2010-12-29 13:39:15 +00:00
* $Id$
2010-02-10 18:18:01 +00:00
* $Author$
2009-08-20 13:06:34 +00:00
*/
if (!defined('e107_INIT')) { exit; }
/**
* Base e107 Object class
*
2009-08-20 13:06:34 +00:00
* @package e107
* @category e107_handlers
* @version 1.0
* @author SecretR
* @copyright Copyright (C) 2010, e107 Inc.
2009-08-20 13:06:34 +00:00
*/
class e_object
2009-08-20 13:06:34 +00:00
{
/**
* Object data
*
* @var array
*/
protected $_data = array();
/**
* Model parameters passed mostly from external sources
*
* @var array
*/
protected $_params = array();
2014-02-25 11:43:28 +02:00
/**
* Name of object id field
* Required for {@link getId()()} method
*
* @var string
*/
protected $_field_id;
/**
* Constructor - set data on initialization
*
* @param array $data
*/
function __construct($data = array())
{
if(is_array($data)) $this->setData($data);
}
/**
* Set name of object's field id
*
* @see getId()
*
* @param string $name
* @return e_object
*/
public function setFieldIdName($name)
{
$this->_field_id = $name;
return $this;
}
/**
* Retrieve name of object's field id
*
* @see getId()
*
* @param string $name
* @return string
*/
public function getFieldIdName()
{
return $this->_field_id;
}
2014-02-25 11:43:28 +02:00
/**
* Retrieve object primary id field value
*
* @return mixed
*/
public function getId()
{
if ($this->getFieldIdName())
{
return $this->get($this->getFieldIdName(), 0); // default of NULL will break MySQL strict in most cases.
}
return $this->get('id', 0);
}
/**
* Set object primary id field value
*
* @return e_object
*/
public function setId($id)
{
if ($this->getFieldIdName())
{
return $this->set($this->getFieldIdName(), $id);
}
return $this;
}
/**
* Retrieves data from the object ($_data) without
* key parsing (performance wise, prefered when possible)
*
* @param string $key
* @param mixed $default
* @return mixed
*/
public function get($key, $default = null)
{
return (isset($this->_data[$key]) ? $this->_data[$key] : $default);
}
2014-02-25 11:43:28 +02:00
/**
* Get object data
* @return array
*/
public function getData()
{
return $this->_data;
}
2014-02-25 11:43:28 +02:00
/**
* Overwrite data in the object for a single field.
*
* @param string $key
* @param mixed $value
* @return e_object
*/
public function set($key, $value)
{
$this->_data[$key] = $value;
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Set object data
* @return e_object
*/
public function setData($data)
{
$this->_data = $data;
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Update object data
* @return e_object
*/
public function addData($data)
{
foreach($data as $key => $val)
{
$this->set($key, $val);
}
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Remove object data key
*
* @param string $key
* @return e_object
*/
2010-12-13 16:00:18 +00:00
public function remove($key)
{
unset($this->_data[$key]);
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Reset object data key
*
* @return e_object
*/
public function removeData()
{
$this->_data = array();
return $this;
}
/**
* Check if key is set
* @param string $key
* @return boolean
*/
public function is($key)
{
return (isset($this->_data[$key]));
}
2014-02-25 11:43:28 +02:00
/**
* Check if key is set and not empty
* @param string $key
* @return boolean
*/
public function has($key)
{
return (isset($this->_data[$key]) && !empty($this->_data[$key]));
}
2014-02-25 11:43:28 +02:00
/**
* Check if object has data
* @return boolean
*/
public function hasData()
{
return !empty($this->_data);
}
2014-02-25 11:43:28 +02:00
/**
* Set parameter array
* @param array $params
* @return e_object
*/
public function setParams(array $params)
{
$this->_params = $params;
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Update parameter array
* @param array $params
* @return e_object
*/
public function updateParams(array $params)
{
2014-02-25 11:43:28 +02:00
foreach ($params as $k => $v)
{
$this->setParam($k, $v);
}
2014-02-25 11:43:28 +02:00
return $this;
}
/**
* Get parameter array
*
* @return array parameters
*/
public function getParams()
{
return $this->_params;
}
/**
* Set parameter
*
* @param string $key
* @param mixed $value
* @return e_object
*/
public function setParam($key, $value)
{
2010-12-29 13:39:15 +00:00
if(null === $value)
{
unset($this->_params[$key]);
}
else $this->_params[$key] = $value;
2014-02-25 11:43:28 +02:00
return $this;
}
/**
* Get parameter
*
* @param string $key
* @param mixed $default
*/
public function getParam($key, $default = null)
{
return (isset($this->_params[$key]) ? $this->_params[$key] : $default);
}
2014-02-25 11:43:28 +02:00
/**
* Convert object data to simple shortcodes (e_vars object)
* @return string
*/
public function toSc()
{
return new e_vars($this->_data);
2014-02-25 11:43:28 +02:00
}
/**
* Convert object data to array
* @return string
*/
public function toJson()
{
return json_encode($this->_data);
}
/**
* Convert object to array
* @return array object data
*/
public function toArray()
{
return $this->_data;
}
2014-02-25 11:43:28 +02:00
/**
* Magic method - convert object data to an array
*
* @return array
*/
public function __toArray()
{
return $this->toArray();
}
/**
* Convert object data to a string
*
* @param boolean $AddSlashes
* @return string
*/
public function toString($AddSlashes = false)
{
return (string) e107::getArrayStorage()->WriteArray($this->toArray(), $AddSlashes);
}
/**
* Magic method - convert object data to a string
* NOTE: before PHP 5.2.0 the __toString method was only
* called when it was directly combined with echo() or print()
*
* NOTE: PHP 5.3+ is throwing parse error if __toString has optional arguments.
*
* @return string
*/
public function __toString()
{
return $this->toString(false);
}
/**
* Magic setter
* Triggered on e.g. <code><?php $e_object->myKey = 'someValue'; </code>
2014-02-25 11:43:28 +02:00
*
* @param string $key
* @param mixed $value
*/
public function __set($key, $value)
{
// Unset workaround - PHP < 5.1.0
if(null === $value) $this->remove($key);
else $this->set($key, $value);
}
/**
* Magic getter
* Triggered on e.g. <code><?php print($e_object->myKey); </code>
* @param string $key
* @return mixed value or null if key not found
*/
public function __get($key)
{
if($this->is($key))
{
return $this->get($key);
}
return null;
}
/**
* Magic method to check if given data key is set.
* Triggered on <code><?php isset($e_object->myKey); </code>
* NOTE: works on PHP 5.1.0+
*
* @param string $key
* @return boolean
*/
public function __isset($key)
{
return $this->is($key);
}
/**
* Magic method to unset given data key.
* Triggered on <code><?php unset($e_object->myKey); </code>
* NOTE: works on PHP 5.1.0+
*
* @param string $key
*/
public function __unset($key)
{
$this->remove($key);
}
}
/**
* Data object for e_parse::simpleParse()
* NEW - not inherits core e_object
* Moved from e_parse_class.php
* Could go in separate file in the future, together with e_object class
*/
class e_vars extends e_object
{
/**
* Get data array
*
* @return array
*/
public function getVars()
{
return $this->getData();
}
/**
* Set array data
*
* @param array $array
* @return e_vars
*/
public function setVars(array $array)
{
$this->setData($array);
return $this;
}
/**
* Add array data to the object (merge with existing)
*
* @param array $array
* @return e_vars
*/
public function addVars(array $array)
{
$this->addData($array);
}
/**
* Reset object data
*
* @return e_vars
*/
public function emptyVars()
{
$this->removeData();
return $this;
}
/**
* Check if there is data available
*
* @return boolean
*/
public function isEmpty()
{
2010-12-23 10:26:40 +00:00
return (!$this->hasData());
}
2014-02-25 11:43:28 +02:00
/**
* Check if given data key is set
* @param string $key
* @return boolean
*/
public function isVar($key)
{
return $this->is($key);
}
2014-02-25 11:43:28 +02:00
/**
* No need of object conversion, optional cloning
* @param boolean $clone return current object clone
* @return e_vars
*/
public function toSc($clone = false)
{
if($clone) return clone $this;
return $this;
}
}
/**
* Base e107 Model class
*
* @package e107
* @category e107_handlers
* @version 1.0
* @author SecretR
* @copyright Copyright (C) 2010, e107 Inc.
*/
class e_model extends e_object
{
2009-10-22 14:18:18 +00:00
/**
* Data structure (types) array, required for {@link e_front_model::sanitize()} method,
2009-10-22 14:18:18 +00:00
* it also serves as a map (find data) for building DB queries,
* copy/sanitize posted data to object data, etc.
*
2009-10-22 14:18:18 +00:00
* This can/should be overwritten by extending the class
*
* @var array
*/
protected $_data_fields = array();
/**
* Current model field types eg. text, bbarea, dropdown etc.
*
*
* @var string
*/
protected $_field_input_types = array();
2009-10-30 17:59:32 +00:00
/**
* Current model DB table, used in all db calls
*
2009-10-30 17:59:32 +00:00
* This can/should be overwritten/set by extending the class
*
2009-10-30 17:59:32 +00:00
* @var string
*/
protected $_db_table;
2014-02-25 11:43:28 +02:00
/**
* Current url Profile data
* Example: array('route'=>'page/view/index', 'vars' => array('id' => 'page_id', 'sef' => 'page_sef'), 'name' => 'page_title', 'description' => '');
* @var string
*/
2013-02-28 13:40:21 +02:00
protected $_url = array();
2014-02-25 11:43:28 +02:00
/**
* Current Featurebox Profile data
* Example: array('title' => 'page_title', 'text' => '');
* @var string
*/
protected $_featurebox = array();
2009-08-20 13:06:34 +00:00
/**
* Runtime cache of parsed from {@link _getData()} keys
*
* @var array
*/
protected $_parsed_keys = array();
2009-08-20 13:06:34 +00:00
/**
* Avoid DB calls if data is not changed
*
* @see _setData()
2009-08-20 13:06:34 +00:00
* @var boolean
*/
protected $data_has_changed = false;
/**
* Namespace to be used for model related system messages in {@link eMessage} handler
*
* @var string
*/
protected $_message_stack = 'default';
/**
* Cache string to be used from _get/set/clearCacheData() methods
*
* @var string
*/
protected $_cache_string = null;
/**
* Force Cache even if system cahche is disabled
* Default is false
*
* @var boolean
*/
protected $_cache_force = false;
2014-02-25 11:43:28 +02:00
2009-10-30 17:59:32 +00:00
/**
* Optional DB table - used for auto-load data from the DB
* @param string $table
* @return e_model
*/
2009-10-30 17:59:32 +00:00
public function getModelTable()
{
return $this->_db_table;
}
/**
* Set model DB table
* @param string $table
* @return e_model
*/
2009-10-30 17:59:32 +00:00
public function setModelTable($table)
{
$this->_db_table = $table;
return $this;
}
/**
* Set model Url Profile
* @param string $table
* @return e_model
*/
public function setUrl($url)
{
2013-02-28 13:40:21 +02:00
if(!is_array($url)) $url = array('route' => $url);
$this->_url = $url;
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Get url profile
* @return array
*/
public function getUrl()
{
return $this->_url;
}
2014-02-25 11:43:28 +02:00
/**
* Set model Featurebox Profile
* @param string $table
* @return e_model
*/
public function setFeaturebox($fb)
{
// if(!is_array($url)) $url = array('route' => $url);
$this->_featurebox = $fb;
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Get Featurebox profile
* @return array
*/
public function getFeaturebox()
{
return $this->_featurebox;
2014-02-25 11:43:28 +02:00
}
/**
* Generic URL assembling method
* @param array $options [optional] see eRouter::assemble() for $options structure
* @param boolean $extended [optional] if true, method will return an array containing url, title and description of the url
* @return mixed URL string or extended array data
*/
2016-06-09 16:43:36 -07:00
public function url($ids, $options = array(), $extended = false)
{
2014-02-25 11:43:28 +02:00
$urldata = $this->getUrl();
if(empty($urldata) || !vartrue($urldata['route'])) return ($extended ? array() : null);
2014-02-25 11:43:28 +02:00
$eurl = e107::getUrl();
2014-02-25 11:43:28 +02:00
if(empty($options)) $options = array();
elseif(!is_array($options)) parse_str($options, $options);
2014-02-25 11:43:28 +02:00
$vars = $this->toArray();
if(!isset($options['allow']) || empty($options['allow']))
{
if(vartrue($urldata['vars']) && is_array($urldata['vars']))
{
$vars = array();
2014-02-25 11:43:28 +02:00
foreach ($urldata['vars'] as $var => $field)
{
if($field === true) $field = $var;
$vars[$var] = $this->get($field);
}
}
}
2014-02-25 11:43:28 +02:00
$method = isset($options['sc']) ? 'sc' : 'create';
2014-02-25 11:43:28 +02:00
$url = e107::getUrl()->$method($urldata['route'], $vars, $options);
2014-02-25 11:43:28 +02:00
if(!$extended)
{
return $url;
}
2014-02-25 11:43:28 +02:00
return array(
2014-02-25 11:43:28 +02:00
'url' => $url,
'name' => vartrue($urldata['name']) ? $this->get($urldata['name']) : '',
'description' => vartrue($urldata['description']) ? $this->get($urldata['description']) : '',
);
}
2014-02-25 11:43:28 +02:00
/**
* Generic Featurebox assembling method
* @return mixed URL string or extended array data
*/
public function featurebox($options = array(), $extended = false)
{
2014-02-25 11:43:28 +02:00
}
2009-10-22 14:18:18 +00:00
/**
2009-10-28 17:05:35 +00:00
* Get data fields array
2009-10-22 14:18:18 +00:00
* @return array
*/
public function getDataFields()
{
return $this->_data_fields;
}
/**
* @param $key
* @return bool
*/
public function getFieldInputType($key)
{
if(isset($this->_field_input_types[$key]))
{
return $this->_field_input_types[$key];
}
return false;
}
2009-10-28 17:05:35 +00:00
/**
* Set Predefined data fields in format key => type
* @return e_model
*/
public function setDataFields($data_fields)
{
$this->_data_fields = $data_fields;
return $this;
}
/**
* Set Predefined data fields in format key => type
* @return e_model
*/
public function setFieldInputTypes($fields)
{
$this->_field_input_types = $fields;
return $this;
}
/**
* Set Predefined data field
* @return e_model
*/
public function setDataField($field, $type)
{
$this->_data_fields[$field] = $type;
return $this;
}
2009-08-20 13:06:34 +00:00
/**
* Retrieves data from the object ($_data) without
* key parsing (performance wise, prefered when possible)
*
* @see _getDataSimple()
* @param string $key
* @param mixed $default
* @return mixed
*/
2009-09-03 14:15:36 +00:00
public function get($key, $default = null)
2009-08-20 13:06:34 +00:00
{
2009-09-03 14:15:36 +00:00
return $this->_getDataSimple((string) $key, $default);
2009-08-20 13:06:34 +00:00
}
2009-08-20 13:06:34 +00:00
/**
* Retrieves data from the object ($_data)
* If $key is empty, return all object data
*
* @see _getData()
* @param string $key
* @param mixed $default
* @param integer $index
* @return mixed
*/
public function getData($key = '', $default = null, $index = null)
{
return $this->_getData($key, $default, $index);
}
2009-08-20 13:06:34 +00:00
/**
* Overwrite data in the object for a single field. Key is not parsed.
* Public proxy of {@link _setDataSimple()}
* Data isn't sanitized so use this method only when data comes from trustable sources (e.g. DB)
*
2009-08-20 13:06:34 +00:00
*
* @see _setData()
* @param string $key
* @param mixed $value
* @param boolean $strict update only
* @return e_model
*/
public function set($key, $value = null, $strict = false)
{
return $this->_setDataSimple($key, $value, $strict);
}
2009-08-20 13:06:34 +00:00
/**
* Overwrite data in the object. Public proxy of {@link _setData()}
* Data isn't sanitized so use this method only when data comes from trustable sources (e.g. DB)
*
* @see _setData()
* @param string|array $key
* @param mixed $value
* @param boolean $strict update only
* @return e_model
*/
public function setData($key, $value = null, $strict = false)
{
return $this->_setData($key, $value, $strict);
}
2009-08-20 13:06:34 +00:00
/**
* Add data to the object.
* Retains existing data in the object.
* Public proxy of {@link _addData()}
*
2009-08-20 13:06:34 +00:00
* If $override is false, data will be updated only (check against existing data)
*
2009-08-20 13:06:34 +00:00
* @param string|array $key
* @param mixed $value
* @param boolean $override override existing data
* @return e_model
*/
public function addData($key, $value = null, $override = true)
{
return $this->_addData($key, $value, $override);
}
2009-08-20 13:06:34 +00:00
/**
* Unset single field from the object.
* Public proxy of {@link _unsetDataSimple()}
*
* @param string $key
* @return e_model
*/
public function remove($key)
{
return $this->_unsetDataSimple($key);
}
2009-08-20 13:06:34 +00:00
/**
* Unset data from the object.
* $key can be a string only. Array will be ignored.
* '/' inside the key will be treated as array path
* if $key is null entire object will be reset
*
2009-08-20 13:06:34 +00:00
* Public proxy of {@link _unsetData()}
*
* @param string|null $key
* @return e_model
*/
public function removeData($key = null)
{
return $this->_unsetData($key);
}
2009-08-20 13:06:34 +00:00
/**
* @param string $key
* @return boolean
*/
public function has($key)
{
return $this->_hasData($key);
}
2009-08-20 13:06:34 +00:00
/**
* @param string $key
* @return boolean
*/
public function hasData($key = '')
{
return $this->_hasData($key);
}
2009-08-20 13:06:34 +00:00
/**
* @param string $key
* @return boolean
*/
public function isData($key)
{
return $this->_isData($key);
}
/**
* @param boolean $new_state new object state if set
* @return boolean
*/
public function isModified($new_state = null)
{
if(is_bool($new_state))
{
$this->data_has_changed = $new_state;
}
return $this->data_has_changed;
}
2009-08-20 13:06:34 +00:00
/**
* Retrieves data from the object
2009-08-20 13:06:34 +00:00
*
* If $key is empty will return all the data as an array
* Otherwise it will return value of the attribute specified by $key
* '/' inside the key will be treated as array path (x/y/z equals to [x][y][z]
2009-08-20 13:06:34 +00:00
*
* If $index is specified it will assume that attribute data is an array
* and retrieve corresponding member.
2014-02-25 11:43:28 +02:00
*
* NEW: '/' supported in keys now, just use double slashes '//' as key separator
* Examples:
* - 'key//some/key/with/slashes//more' -> [key][some/key/with/slashes][more]
* - '//some/key' -> [some/key] - starting with // means - don't parse!
* - '///some/key/' -> [/some/key/]
* - '//key//some/key/with/slashes//more' WRONG -> single key [key//some/key/with/slashes//more]
2014-02-25 11:43:28 +02:00
*
* @param string $key
* @param mixed $default
* @param integer $index
* @param boolean $posted data source
* @return mixed
2009-08-20 13:06:34 +00:00
*/
protected function _getData($key = '', $default = null, $index = null, $data_src = '_data')
{
if ('' === $key)
2009-08-20 13:06:34 +00:00
{
return $this->$data_src;
}
2014-02-25 11:43:28 +02:00
$simple = false;
if(strpos($key, '//') === 0)
{
$key = substr($key, 2);
$simple = true;
}
/*elseif($key[0] == '/')
{
// just use it!
2014-02-25 11:43:28 +02:00
$simple = true;
}*/
else
{
2014-02-25 11:43:28 +02:00
$simple = strpos($key, '/') === false;
}
2009-08-20 13:06:34 +00:00
// Fix - check if _data[path/to/value] really doesn't exist
if (!$simple)
2009-08-20 13:06:34 +00:00
{
//$key = trim($key, '/');
2009-08-20 13:06:34 +00:00
if(isset($this->_parsed_keys[$data_src.'/'.$key]))
{
return $this->_parsed_keys[$data_src.'/'.$key];
}
2014-02-25 11:43:28 +02:00
// new feature (double slash) - when searched key string is key//some/key/with/slashes//more
// -> search for 'key' => array('some/key/with/slashes', array('more' => value));
$keyArr = explode(strpos($key, '//') ? '//' : '/', $key);
2009-08-20 13:06:34 +00:00
$data = $this->$data_src;
foreach ($keyArr as $i => $k)
2009-08-20 13:06:34 +00:00
{
if ('' === $k)
2009-08-20 13:06:34 +00:00
{
return $default;
}
if (is_array($data))
2009-08-20 13:06:34 +00:00
{
if (!isset($data[$k]))
2009-08-20 13:06:34 +00:00
{
return $default;
}
$data = $data[$k];
}
else
2009-08-20 13:06:34 +00:00
{
return $default;
}
}
$this->_parsed_keys[$data_src.'/'.$key] = $data;
return $data;
}
//get $index
if (isset($this->{$data_src}[$key]))
2009-08-20 13:06:34 +00:00
{
if (null === $index)
2009-08-20 13:06:34 +00:00
{
return $this->{$data_src}[$key];
}
$value = $this->{$data_src}[$key];
if (is_array($value))
2009-08-20 13:06:34 +00:00
{
if (isset($value[$index]))
2009-08-20 13:06:34 +00:00
{
return $value[$index];
}
return $default;
}
elseif (is_string($value))
2009-08-20 13:06:34 +00:00
{
$arr = explode("\n", $value);
return (isset($arr[$index]) ? $arr[$index] : $default);
}
return $default;
}
return $default;
}
2009-08-20 13:06:34 +00:00
/**
* Get value from _data array without parsing the key
*
* @param string $key
* @param mixed $default
* @param string $posted data source
* @return mixed
*/
protected function _getDataSimple($key, $default = null, $data_src = '_data')
{
return isset($this->{$data_src}[$key]) ? $this->{$data_src}[$key] : $default;
}
2009-08-20 13:06:34 +00:00
/**
* Overwrite data in the object.
*
* $key can be string or array.
* If $key is string, the attribute value will be overwritten by $value
* '/' inside the key will be treated as array path
*
* If $key is an array and $strict is false, it will overwrite all the data in the object.
*
2009-08-20 13:06:34 +00:00
* If $strict is true and $data_src is '_data', data will be updated only (no new data will be added)
*
* NEW: '/' supported in keys now, just use double slashes '//' as key separator
* Examples:
* - 'key//some/key/with/slashes//more' -> [key][some/key/with/slashes][more]
* - '//some/key' -> [some/key] - starting with // means - don't parse!
* - '///some/key/' -> [/some/key/]
* - '//key//some/key/with/slashes//more' WRONG -> single key [key//some/key/with/slashes//more]
2014-02-25 11:43:28 +02:00
*
*
2009-08-20 13:06:34 +00:00
* @param string|array $key
* @param mixed $value
* @param boolean $strict
* @param string $data_src
* @return e_model
*/
protected function _setData($key, $value = null, $strict = false, $data_src = '_data')
{
if(is_array($key))
2009-08-20 13:06:34 +00:00
{
if($strict)
2009-08-20 13:06:34 +00:00
{
foreach($key as $k => $v)
2009-08-20 13:06:34 +00:00
{
$this->_setData($k, $v, true, $data_src);
2009-08-20 13:06:34 +00:00
}
return $this;
}
$this->$data_src = $key;
2009-08-20 13:06:34 +00:00
return $this;
}
2009-08-20 13:06:34 +00:00
//multidimensional array support - strict _setData for values of type array
if($strict && !empty($value) && is_array($value))
{
foreach($value as $k => $v)
{
2014-02-25 11:43:28 +02:00
// new - $k couldn't be a path - e.g. 'key' 'value/value1'
// will result in 'key' => 'value/value1' and NOT 'key' => array('value' => value1)
$this->_setData($key.'//'.$k, $v, true, $data_src);
2009-08-20 13:06:34 +00:00
}
return $this;
}
2014-02-25 11:43:28 +02:00
$simple = false;
if(strpos($key, '//') === 0)
{
// NEW - leading '//' means set 'some/key' without parsing it
// Example: '//some/key'; NOTE: '//some/key//more/depth' is NOT parsed
// if you wish to have array('some/key' => array('more/depth' => value))
// right syntax is 'some/key//more/depth'
$key = substr($key, 2);
$simple = true;
}
/*elseif($key[0] == '/')
{
2014-02-25 11:43:28 +02:00
$simple = true;
}*/
else
{
2014-02-25 11:43:28 +02:00
$simple = strpos($key, '/') === false;
}
2014-02-25 11:43:28 +02:00
//multidimensional array support - parse key
if(!$simple)
2009-08-20 13:06:34 +00:00
{
//$key = trim($key, '/');
2009-08-20 13:06:34 +00:00
//if strict - update only
if($strict && !$this->isData($key))
{
return $this;
}
2014-02-25 11:43:28 +02:00
// new feature (double slash) - when parsing key: key//some/key/with/slashes//more
// -> result is 'key' => array('some/key/with/slashes', array('more' => value));
$keyArr = explode(strpos($key, '//') ? '//' : '/', $key);
//$keyArr = explode('/', $key);
$data = &$this->{$data_src};
for ($i = 0, $l = count($keyArr); $i < $l; $i++)
2009-08-20 13:06:34 +00:00
{
2017-01-13 10:34:03 -08:00
2009-08-20 13:06:34 +00:00
$k = $keyArr[$i];
2017-01-13 10:34:03 -08:00
if (!isset($data[$k]) || empty($data[$k])) // PHP7.1 fix. Reset to empty array() if $data[$k] is an empty string. Reason for empty string still unknown.
2009-08-20 13:06:34 +00:00
{
$data[$k] = array();
}
2017-01-13 10:34:03 -08:00
2009-08-20 13:06:34 +00:00
$data = &$data[$k];
}
2009-08-20 13:06:34 +00:00
//data has changed - optimized
if('_data' === $data_src && !$this->data_has_changed)
{
$this->data_has_changed = (!isset($this->_data[$key]) || $this->_data[$key] != $value);
2009-08-20 13:06:34 +00:00
}
$this->_parsed_keys[$data_src.'/'.$key] = $value;
$data = $value;
}
else
2009-08-20 13:06:34 +00:00
{
//if strict - update only
if($strict && !isset($this->_data[$key]))
{
return $this;
}
if('_data' === $data_src && !$this->data_has_changed)
{
$this->data_has_changed = (!isset($this->_data[$key]) || $this->{$data_src}[$key] != $value);
2009-08-20 13:06:34 +00:00
}
$this->{$data_src}[$key] = $value;
}
return $this;
}
2009-08-20 13:06:34 +00:00
/**
* Set data for the given source. More simple (and performance wise) version
* of {@link _setData()}
*
* @param string $key
* @param mixed $value
* @param boolean $strict
* @param string $data_src
* @return e_model
*/
2009-09-03 14:15:36 +00:00
protected function _setDataSimple($key, $value = null, $strict = false, $data_src = '_data')
2009-08-20 13:06:34 +00:00
{
2009-09-03 14:15:36 +00:00
$key = $key.'';//smart toString
2009-08-20 13:06:34 +00:00
if(!$strict)
{
//data has changed
if('_data' === $data_src && !$this->data_has_changed)
{
$this->data_has_changed = (!isset($this->_data[$key]) || $this->_data[$key] != $value);
2009-08-20 13:06:34 +00:00
}
$this->{$data_src}[$key] = $value;
return $this;
}
2009-08-20 13:06:34 +00:00
if($this->isData($key))
{
if('_data' === $data_src && !$this->data_has_changed)
{
$this->data_has_changed = (!isset($this->_data[$key]) || $this->_data[$key] != $value);
2009-08-20 13:06:34 +00:00
}
$this->{$data_src}[$key] = $value;
}
return $this;
}
2009-08-20 13:06:34 +00:00
/**
* Add data to the object.
* Retains existing data in the object.
*
2009-08-20 13:06:34 +00:00
* If $override is false, only new (non-existent) data will be added
*
2009-08-20 13:06:34 +00:00
* @param string|array $key
* @param mixed $value
* @param boolean $override allow override of existing data
* @param string $data_src data source
* @return e_model
*/
protected function _addData($key, $value = null, $override = true, $data_src = '_data')
{
if(is_array($key))
{
foreach($key as $k => $v)
{
$this->_addData($k, $v, $override, $data_src);
}
return $this;
}
2009-08-20 13:06:34 +00:00
if($override || !$this->_isData($key, $data_src))
{
if(is_array($value))
{
if(is_array($key))
2009-08-20 13:06:34 +00:00
{
foreach($key as $k => $v)
{
$this->_addData($key.'/'.$k, $v, $override, $data_src);
}
2009-08-20 13:06:34 +00:00
}
return $this;
}
$this->_setData($key, $value, false, $data_src);
}
return $this;
}
2009-08-20 13:06:34 +00:00
/**
* Unset data from the object from the given source.
* $key can be a string only. Array will be ignored.
* '/' inside the key will be treated as array path
* if $key is null entire object will be reset
*
* @param string|null $key
* @param string $data_src data source
* @return e_model
*/
protected function _unsetData($key = null, $data_src = '_data')
{
if (null === $key)
2009-08-20 13:06:34 +00:00
{
if('_data' === $data_src && !empty($this->_data))
{
$this->data_has_changed = true;
}
2009-10-07 10:53:33 +00:00
$this->{$data_src} = array();
2009-08-20 13:06:34 +00:00
return $this;
}
2009-08-20 13:06:34 +00:00
$key = trim($key, '/');
if(strpos($key,'/'))
2009-08-20 13:06:34 +00:00
{
$keyArr = explode('/', $key);
2009-10-07 10:53:33 +00:00
$data = &$this->{$data_src};
2009-10-07 10:53:33 +00:00
$unskey = array_pop($keyArr);
for ($i = 0, $l = count($keyArr); $i < $l; $i++)
2009-08-20 13:06:34 +00:00
{
$k = $keyArr[$i];
if (!isset($data[$k]))
2009-08-20 13:06:34 +00:00
{
return $this; //not found
}
$data = &$data[$k];
}
if(is_array($data))
{
if('_data' === $data_src && isset($data[$unskey]))
{
$this->data_has_changed = true;
}
unset($data[$unskey], $this->_parsed_keys[$data_src.'/'.$key]);
}
}
else
2009-08-20 13:06:34 +00:00
{
if('_data' === $data_src && isset($this->{$data_src}[$key]))
{
$this->data_has_changed = true;
}
unset($this->{$data_src}[$key]);
}
return $this;
}
2009-08-20 13:06:34 +00:00
/**
* Unset single field from the object from the given source. Key is not parsed
*
* @param string $key
* @param string $data_src data source
* @return e_model
*/
protected function _unsetDataSimple($key, $data_src = '_data')
{
if('_data' === $data_src && isset($this->{$data_src}[$key]))
{
$this->data_has_changed = true;
}
unset($this->{$data_src}[$key]);
return $this;
}
/**
* If $key is empty, checks whether there's any data in the object
* Otherwise checks if the specified key is empty/set.
*
* @param string $key
* @param string $data_src data source
* @return boolean
*/
protected function _hasData($key = '', $data_src = '_data')
{
if (empty($key))
2009-08-20 13:06:34 +00:00
{
return !empty($this->$data_src);
}
$value = $this->_getData($key, null, null, $data_src);
return !empty($value);
}
2009-08-20 13:06:34 +00:00
/**
* Checks if the specified key is set
*
* @param string $key
* @param string $data_src data source
* @return boolean
*/
protected function _isData($key, $data_src = '_data')
{
return (null !== $this->_getData($key, null, null, $data_src));
}
/**
* Add system message of type Information
*
* @param string $message
* @param boolean $session [optional]
2009-11-01 19:05:26 +00:00
* @return e_model
*/
public function addMessageInfo($message, $session = false)
{
e107::getMessage()->addStack($message, $this->_message_stack, E_MESSAGE_INFO, $session);
return $this;
}
2009-11-01 19:05:26 +00:00
/**
* Add system message of type Success
*
2009-11-01 19:05:26 +00:00
* @param string $message
* @param boolean $session [optional]
* @return e_model
*/
public function addMessageSuccess($message, $session = false)
{
e107::getMessage()->addStack($message, $this->_message_stack, E_MESSAGE_SUCCESS, $session);
return $this;
}
/**
* Add system message of type Warning
*
* @param string $message
* @param boolean $session [optional]
2009-11-01 19:05:26 +00:00
* @return e_model
*/
public function addMessageWarning($message, $session = false)
{
e107::getMessage()->addStack($message, $this->_message_stack, E_MESSAGE_WARNING, $session);
return $this;
}
/**
* Add system message of type Error
*
* @param string $message
* @param boolean $session [optional]
2009-11-01 19:05:26 +00:00
* @return e_model
*/
public function addMessageError($message, $session = false)
{
e107::getMessage()->addStack($message, $this->_message_stack, E_MESSAGE_ERROR, $session);
2014-01-17 16:46:24 -08:00
e107::getAdminLog()->addError($message,false)->save('ADMINUI_04');
return $this;
}
/**
* Add system message of type Information
*
* @param string $message
* @param boolean $session [optional]
2009-11-01 19:05:26 +00:00
* @return e_model
*/
public function addMessageDebug($message, $session = false)
{
e107::getMessage()->addStack($message, $this->_message_stack, E_MESSAGE_DEBUG, $session);
return $this;
}
2009-08-20 13:06:34 +00:00
/**
2009-10-21 09:12:12 +00:00
* Render System messages (if any)
*
* @param boolean $session store messages to session
* @param boolean $reset reset errors
* @return string
2009-08-20 13:06:34 +00:00
*/
2009-10-21 09:12:12 +00:00
public function renderMessages($session = false, $reset = true)
2009-08-20 13:06:34 +00:00
{
2009-10-21 09:12:12 +00:00
return e107::getMessage()->render($this->_message_stack, $session, $reset);
2009-08-20 13:06:34 +00:00
}
2009-10-22 14:18:18 +00:00
/**
* Move model System messages (if any) to the default eMessage stack
*
2009-10-22 14:18:18 +00:00
* @param boolean $session store messages to session
2009-11-01 19:05:26 +00:00
* @return e_model
2009-10-22 14:18:18 +00:00
*/
public function setMessages($session = false)
{
e107::getMessage()->moveStack($this->_message_stack, 'default', false, $session);
return $this;
}
/**
* Reset model System messages
*
* @param boolean|string $type E_MESSAGE_INFO | E_MESSAGE_SUCCESS | E_MESSAGE_WARNING | E_MESSAGE_WARNING | E_MESSAGE_DEBUG | false (all)
* @param boolean $session reset also session messages
* @return e_model
*/
public function resetMessages($type = false, $session = false)
{
e107::getMessage()->reset($type, $this->_message_stack, $session);
return $this;
}
/**
* Set model message stack
* @param string $stack_name
* @return e_model
*/
public function setMessageStackName($stack_name)
{
$this->_message_stack = $stack_name;
return $this;
}
/**
* Get model message stack name
* @return string
*/
public function getMessageStackName()
{
return $this->_message_stack;
}
2009-08-20 13:06:34 +00:00
/**
* User defined model validation
* Awaiting for child class implementation
*
2009-08-20 13:06:34 +00:00
*/
public function verify()
2009-08-20 13:06:34 +00:00
{
}
2009-08-20 13:06:34 +00:00
/**
* Model validation
* @see e_model_admin
2009-08-20 13:06:34 +00:00
*/
2009-10-21 09:12:12 +00:00
public function validate()
2009-08-20 13:06:34 +00:00
{
}
2009-08-20 13:06:34 +00:00
/**
2009-10-30 17:59:32 +00:00
* Generic load data from DB
2013-10-16 18:13:21 +03:00
* @param mixed $id
2009-10-30 17:59:32 +00:00
* @param boolean $force
* @return e_model
2009-08-20 13:06:34 +00:00
*/
2013-10-16 18:13:21 +03:00
public function load($id = null, $force = false)
2009-10-30 17:59:32 +00:00
{
if(!$force && $this->getId())
2009-10-30 17:59:32 +00:00
{
return $this;
}
2009-11-17 15:34:54 +00:00
if($force)
{
$this->setData(array())
->_clearCacheData();
2009-11-17 15:34:54 +00:00
}
2013-10-16 18:13:21 +03:00
if($id) $id = e107::getParser()->toDB($id);
if(!$id && !$this->getParam('db_query'))
{
return $this;
}
$cached = $this->_getCacheData();
if($cached !== false)
{
$this->setData($cached);
return $this;
}
$sql = e107::getDb();
2009-10-30 17:59:32 +00:00
$qry = str_replace('{ID}', $id, $this->getParam('db_query'));
if($qry)
2009-10-30 17:59:32 +00:00
{
2016-02-14 19:00:12 -08:00
$res = $sql->gen($qry, $this->getParam('db_debug') ? true : false);
}
else
{
if(!is_numeric($id)) $id = "'{$id}'";
2014-02-25 11:43:28 +02:00
2016-02-14 19:00:12 -08:00
$res = $sql->select(
$this->getModelTable(),
$this->getParam('db_fields', '*'),
$this->getFieldIdName().'='.$id.' '.trim($this->getParam('db_where', '')),
'default',
($this->getParam('db_debug') ? true : false)
);
2009-10-30 17:59:32 +00:00
}
if($res)
2009-10-30 17:59:32 +00:00
{
2016-02-14 19:00:12 -08:00
$this->setData($sql->fetch());
2009-10-30 17:59:32 +00:00
}
if($sql->getLastErrorNumber())
{
$this->addMessageDebug('SQL error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
$this->addMessageDebug($sql->getLastQuery());
}
else
{
$this->_setCacheData();
}
2009-10-30 17:59:32 +00:00
return $this;
}
/**
* Retrieve system cache (if any)
* @return array|false
*/
protected function _getCacheData()
{
if(!$this->isCacheEnabled())
{
return false;
}
$cached = e107::getCache()->retrieve_sys($this->getCacheString(true), false, $this->_cache_force);
if(false !== $cached)
{
return e107::unserialize($cached);
}
return false;
}
/**
* Set system cache if enabled for the model
* @return e_model
*/
protected function _setCacheData()
{
if(!$this->isCacheEnabled())
{
return $this;
}
e107::getCache()->set_sys($this->getCacheString(true), $this->toString(false), $this->_cache_force, false);
return $this;
}
/**
* Clrear system cache if enabled for the model
* @return e_model
*/
protected function _clearCacheData()
{
if(!$this->isCacheEnabled(false))
{
return $this;
}
e107::getCache()->clear_sys($this->getCacheString(true), false);
return $this;
}
2014-02-25 11:43:28 +02:00
/**
* Clrear system cache (public proxy) if enabled for the model
* @return e_model
*/
public function clearCache()
{
return $this->_clearCacheData();
}
/**
* Check if cache is enabled for the current model
* @param boolean $checkId check if there is model ID
* @return boolean
*/
public function isCacheEnabled($checkId = true)
{
return (null !== $this->getCacheString() && (!$checkId || $this->getId()));
}
/**
* Get model cache string
* @param boolean $replace try to add current model ID (replace destination is {ID})
* @return string
*/
public function getCacheString($replace = false)
{
return ($replace ? str_replace('{ID}', $this->getId(), $this->_cache_string) : $this->_cache_string);
}
/**
* Set model cache string
* @param string $str
* @return e_model
*/
public function setCacheString($str)
{
$this->_cache_string = $str;
return $this;
}
/**
* Save data to DB
* Awaiting for child class implementation
* @see e_model_admin
*/
public function save()
{
}
/**
* Delete DB record
* Awaiting for child class implementation
* @see e_model_admin
*/
2016-02-14 19:00:12 -08:00
public function delete($ids, $destroy = true, $session_messages = false)
{
}
2009-08-20 13:06:34 +00:00
/**
* Create new DB recorrd
2009-08-20 13:06:34 +00:00
* Awaiting for child class implementation
* @see e_model_admin
2009-08-20 13:06:34 +00:00
*/
public function create()
2009-08-20 13:06:34 +00:00
{
}
2009-08-20 13:06:34 +00:00
/**
* Insert data to DB
* Awaiting for child class implementation
* @see e_model_admin
2009-08-20 13:06:34 +00:00
*/
protected function dbInsert()
2009-08-20 13:06:34 +00:00
{
}
2009-08-20 13:06:34 +00:00
/**
* Update DB data
* Awaiting for child class implementation
* @see e_model_admin
2009-08-20 13:06:34 +00:00
*/
2016-02-14 19:00:12 -08:00
protected function dbUpdate($force = false, $session_messages = false)
2009-08-20 13:06:34 +00:00
{
}
2009-10-30 17:59:32 +00:00
/**
* Replace DB record
* Awaiting for child class implementation
* @see e_model_admin
*/
protected function dbReplace()
2009-10-30 17:59:32 +00:00
{
}
2009-10-30 17:59:32 +00:00
/**
* Delete DB data
* Awaiting for child class implementation
* @see e_model_admin
*/
protected function dbDelete()
2009-10-30 17:59:32 +00:00
{
}
2009-10-30 17:59:32 +00:00
/**
* Set parameter array
* Core implemented:
* - db_query: string db query to be passed to load() ($sql->gen())
* - db_query
* - db_fields
* - db_where
* - db_debug
* - model_class: e_tree_model class/subclasses - string class name for creating nodes inside default load() method
* - clearModelCache: e_tree_model class/subclasses - clear cache per node after successful DB operation
* - noCacheStringModify: e_tree_model class/subclasses - do not add additional md5 sum to tree cache string
2009-10-30 17:59:32 +00:00
* @param array $params
* @return e_model
*/
public function setParams(array $params)
{
parent::setParams($params);
2009-10-30 17:59:32 +00:00
return $this;
}
2009-11-26 17:14:07 +00:00
/**
* Render model data, all 'sc_*' methods will be recongnized
* as shortcodes.
*
2009-11-26 17:14:07 +00:00
* @param string $template
* @param boolean $parsesc parse external shortcodes, default is true
* @param e_vars $eVars simple parser data
2009-11-26 17:14:07 +00:00
* @return string parsed template
*/
public function toHTML($template, $parsesc = true, $eVars = null)
2009-11-26 17:14:07 +00:00
{
return e107::getParser()->parseTemplate($template, $parsesc, $this, $eVars);
2009-11-26 17:14:07 +00:00
}
/**
* Export a Model configuration
* @return string
*/
2009-11-26 17:14:07 +00:00
public function toXML()
{
$ret = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>\n";
$ret .= "<e107Export type=\"model\" version=\"1.0\" timestamp=\"".time()."\" >\n";
2009-11-26 17:14:07 +00:00
$ret .= "\t<data>\n";
// TODO - handle multi dimensional arrays (already possible - field1/field2?), method toXMLValue($value, $type)
foreach ($this->getDataFields() as $field => $type)
{
$ret .= "\t\t<field name=\"{$field}\" type=\"{$type}\">";
$ret .= $type == 'str' || $type == 'string' ? "<![CDATA[".$this->getData($field)."]]>" : $this->getData($field);
$ret .= "</field>\n";
}
$ret .= "\t</data>\n";
2009-11-26 17:14:07 +00:00
$ret .= "</e107Export>";
return $ret;
}
2009-08-20 13:06:34 +00:00
/**
* Try to convert string to a number
* Shoud fix locale related troubles
*
* @param string $value
2009-11-26 17:14:07 +00:00
* @return integer|float
2009-08-20 13:06:34 +00:00
*/
public function toNumber($value)
2009-08-20 13:06:34 +00:00
{
$larr = localeconv();
$search = array(
2014-02-25 11:43:28 +02:00
$larr['decimal_point'],
$larr['mon_decimal_point'],
$larr['thousands_sep'],
$larr['mon_thousands_sep'],
$larr['currency_symbol'],
$larr['int_curr_symbol']
);
$replace = array('.', '.', '', '', '', '');
2014-02-25 11:43:28 +02:00
return str_replace($search, $replace, $value);
}
2009-08-20 13:06:34 +00:00
/**
* Convert object data to a string
*
* @param boolean $AddSlashes
* @param string $key optional, if set method will return corresponding value as a string
2009-08-20 13:06:34 +00:00
* @return string
*/
public function toString($AddSlashes = true, $key = null)
2009-08-20 13:06:34 +00:00
{
if (null !== $key)
{
$value = $this->getData($key);
if(is_array($value))
{
return e107::getArrayStorage()->WriteArray($value, $AddSlashes);
}
return (string) $value;
}
return (string) e107::getArrayStorage()->WriteArray($this->toArray(), $AddSlashes);
2009-08-20 13:06:34 +00:00
}
2009-10-30 17:59:32 +00:00
public function destroy()
{
$this->_data = array();
$this->_params = array();
$this->_data_fields = array();
$this->_parsed_keys = array();
$this->_db_table = $this->_field_id = '';
$this->data_has_changed = false;
}
/**
* Disable Magic setter
*/
public function __set($key, $value)
{
}
/**
* Disable Magic getter
*/
public function __get($key)
{
}
/**
* Disable
*/
public function __isset($key)
{
}
/**
* Disable
*/
public function __unset($key)
{
}
}
/**
* Base e107 Fron Model class interface
*
* Some important points:
* - model data should be always in toDB() format:
* - retrieved direct from DB
* - set & sanitized via setPostedData()->mergePostedData()
* - manually sanitized before passed to model setter (set(), setData(), add(), addData(), etc.) methods
2009-10-22 14:18:18 +00:00
* - $_data_fields property is important, it tells to sanitize() method how to sanitize posted data
* - if $_data_fields is missing, sanitize() will call internally e107::getParser()->toDB() on the data
* - sanitize() is triggered by default on mergePostedData() and mergeData() methods
* - mergePostedData() and mergeData() methods will filter posted/passed data against (in this order):
* - getValidator()->getValidData() if true is passed as validate parameter (currently disabled, gather feedback)
2009-10-22 14:18:18 +00:00
* - $_data_fields if true is passed as sanitize parameter
* - toSqlQuery() needs $_data_fields and $_field_id to work proper, $_FIELD_TYPES is optional but recommended (faster SQL queries)
* - result array from toSqlQuery() call will be filtered against $_data_fields
* - in almost every case $_FIELD_TYPES shouldn't contain 'escape' and 'todb' - dont't forget you are going to pass already sanitized data (see above)
2009-10-22 14:18:18 +00:00
* - most probably $_FIELD_TYPES will go in the future, $_data_fields alone could do the job
* - default db related methods (save(), dbUpdate(), etc.) need $_db_table
*
* @package e107
* @category e107_handlers
* @version $Id$
* @author SecretR
* @copyright Copyright (C) 2008-2010 e107 Inc.
*/
class e_front_model extends e_model
{
/**
* Posted data
* Back-end related
*
* @var array
*/
protected $_posted_data = array();
/**
* DB format array - see db::_getTypes() and db::_getFieldValue() (mysql_class.php)
* for example
*
* This can/should be overwritten by extending the class
*
* @var array
*/
protected $_FIELD_TYPES = array();
/**
* Validation structure - see {@link e_validator::$_required_rules} for
* more information about the array format.
* Used in {@link validate()} method.
2009-10-30 17:59:32 +00:00
* TODO - check_rules (see e_validator::$_optional_rules)
* This can/should be overwritten by extending the class.
*
* @var array
*/
protected $_validation_rules = array();
protected $_optional_rules = array();
/**
* @var integer Last SQL error number
*/
protected $_db_errno = 0;
2010-03-01 12:43:56 +00:00
/**
* @var string Last SQL error message
*/
protected $_db_errmsg = '';
/**
* @var string Last SQL query
*/
protected $_db_qry = '';
/**
* Validator object
*
* @var e_validator
*/
protected $_validator = null;
/**
* @return array
*/
public function getValidationRules()
{
return $this->_validation_rules;
}
/**
* Set object validation rules if $_validation_rules array is empty
*
* @param array $vrules
* @return e_front_model
*/
public function setValidationRules(array $vrules, $force = false)
{
if($force || empty($this->_validation_rules))
{
$this->_validation_rules = $vrules;
}
return $this;
}
/**
* @return array
*/
public function getOptionalRules()
{
return $this->_optional_rules;
}
/**
* @param array $rules
* @return e_front_model
*/
public function setOptionalRules(array $rules)
{
$this->_optional_rules = $rules;
return $this;
}
/**
* Set object validation rules if $_validation_rules array is empty
*
* @param string $field
* @param array $rule
* @param boolean $required
* @return e_front_model
*/
public function setValidationRule($field, $rule, $required = true)
{
$pname = $required ? '_validation_rules' : '_optional_rules';
$rules = &$this->$pname;
$rules[$field] = $rule;
return $this;
}
/**
* Predefined data fields types, passed to DB handler
* @return array
*/
public function getDbTypes()
{
return ($this->_FIELD_TYPES ? $this->_FIELD_TYPES : $this->getDataFields());
}
/**
* Predefined data fields types, passed to DB handler
*
* @param array $field_types
* @return e_front_model
*/
public function setDbTypes($field_types)
{
$this->_FIELD_TYPES = $field_types;
return $this;
}
/**
* Auto field type definitions
* Disabled for now, it should auto-create _data_types
* @param boolean $force
* @return boolean
*/
// public function setFieldTypeDefs($force = false)
// {
// if($force || !$this->getFieldTypes())
// {
// $ret = e107::getDb()->getFieldDefs($this->getModelTable());
// if($ret)
// {
// foreach ($ret as $k => $v)
// {
// if('todb' == $v)
// {
// $ret[$k] = 'string';
// }
// }
// $this->setFieldTypes($ret);
// return true;
// }
// }
// return false;
// }
/**
* Retrieves data from the object ($_posted_data) without
* key parsing (performance wise, prefered when possible)
*
* @see _getDataSimple()
* @param string $key
* @param mixed $default
* @return mixed
*/
public function getPosted($key, $default = null)
{
return $this->_getDataSimple((string) $key, $default, '_posted_data');
}
/**
* Retrieves data from the object ($_posted_data)
* If $key is empty, return all object posted data
* @see _getData()
* @param string $key
* @param mixed $default
* @param integer $index
* @return mixed
*/
public function getPostedData($key = '', $default = null, $index = null)
{
return $this->_getData($key, $default, $index, '_posted_data');
}
/**
* Search for requested data from available sources in this order:
* - posted data
* - default object data
* - passed default value
*
* Use this method inside forms
*
* @param string $key
* @param string $default
* @param integer $index
* @return string
*/
public function getIfPosted($key, $default = '', $index = null)
{
$d = $this->getDataFields();
2016-02-13 12:57:34 -08:00
if(!empty($d[$key]) && ($d[$key] == 'array'))
{
return e107::unserialize($this->getData((string) $key, $default, $index));
}
$posted = $this->getPostedData((string) $key, null, $index);
if(null !== $posted)
{
// FIXED - double post_toFom() and toDB(post_toForm()) problems
// setPosted|setPostedData|addPostedData methods are storing RAW data now
return e107::getParser()->post_toForm($posted);
}
return e107::getParser()->toForm($this->getData((string) $key, $default, $index));
}
/**
* Overwrite posted data in the object for a single field. Key is not parsed.
* Public proxy of {@link _setDataSimple()}
* Use this method to store data from non-trustable sources (e.g. _POST) - it doesn't overwrite
* the original object data
*
* @param string $key
* @param mixed $value
* @param boolean $strict update only
* @return e_front_model
*/
public function setPosted($key, $value, $strict = false)
{
return $this->_setDataSimple($key, $value, $strict, '_posted_data');
}
/**
* Overwrite posted data in the object. Key is parsed (multidmensional array support).
* Public proxy of {@link _setData()}
* Use this method to store data from non-trustable sources (e.g. _POST) - it doesn't overwrite
* the original object data
*
* @param string|array $key
* @param mixed $value
* @param boolean $strict update only
* @return e_front_model
*/
public function setPostedData($key, $value = null, $strict = false)
{
return $this->_setData($key, $value, $strict, '_posted_data');
}
/**
* Add data to the object.
* Retains existing data in the object.
* Public proxy of {@link _addData()}
*
* If $override is false, data will be updated only (check against existing data)
*
* @param string|array $key
* @param mixed $value
* @param boolean $override override existing data
* @return e_front_model
*/
public function addPostedData($key, $value = null, $override = true)
{
return $this->_addData($key, $value, $override, '_posted_data');
}
/**
* Unset single posted data field from the object.
* Public proxy of {@link _unsetDataSimple()}
*
* @param string $key
* @return e_front_model
*/
public function removePosted($key)
{
return $this->_unsetDataSimple($key, '_posted_data');
}
/**
* Unset posted data from the object.
* $key can be a string only. Array will be ignored.
* '/' inside the key will be treated as array path
* if $key is null entire object will be reset
*
* Public proxy of {@link _unsetData()}
*
* @param string|null $key
* @return e_front_model
*/
public function removePostedData($key = null)
{
return $this->_unsetData($key, '_posted_data');
}
/**
* Check if given key exists and non-empty in the posted data array
* @param string $key
* @return boolean
*/
public function hasPosted($key)
{
return $this->_hasData($key, '_posted_data');
}
/**
* Check if posted data is empty
* @return boolean
*/
public function hasPostedData()
{
return $this->_hasData('', '_posted_data');
}
/**
* Check if given key exists in the posted data array
*
* @param string $key
* @return boolean
*/
public function isPosted($key)
{
return (isset($this->_posted_data[$key]));
}
/**
* Check if given key exists in the posted data array ($key us parsed)
*
* @param string $key
* @return boolean
*/
public function isPostedData($key)
{
return $this->_isData($key, '_posted_data');
}
/**
* Compares posted data vs object data
*
* @param string $field
* @param boolean $strict compare variable type as well
* @return boolean
*/
public function dataHasChangedFor($field, $strict = false)
{
$newData = $this->getData($field);
$postedData = $this->getPostedData($field);
return ($strict ? $newData !== $postedData : $newData != $postedData);
}
/**
* @return boolean
*/
public function dataHasChanged()
{
return $this->data_has_changed;
}
/**
* Merge posted data with the object data
* Should be used on edit/update/create record (back-end)
* Retrieved for copy Posted data will be removed (no matter if copy is successfull or not)
*
* If $strict is true, only existing object data will be copied (update)
* If $validate is true, data will be copied only after successful validation
*
* @param boolean $strict
* @param boolean $sanitize sanitize posted data before move it to the object data
* @param boolean $validate perform validation check
* @return e_front_model
*/
public function mergePostedData($strict = true, $sanitize = true, $validate = true)
{
if(!$this->hasPostedData() || ($validate && !$this->validate()))
{
return $this;
}
2014-01-17 16:46:24 -08:00
$oldData = $this->getData();
// $this->addMessageDebug("OLDD".print_a($oldData,true));
2014-02-25 11:43:28 +02:00
2014-01-17 16:46:24 -08:00
$data = $this->getPostedData();
2014-01-17 16:46:24 -08:00
$valid_data = $validate ? $this->getValidator()->getValidData() : array();
if($sanitize)
{
// search for db_field types
2009-10-22 14:18:18 +00:00
if($this->getDataFields())
{
$data = $this->sanitize($data);
}
else //no db field types, use toDB()
{
$data = e107::getParser()->toDB($data);
}
}
2014-01-17 16:46:24 -08:00
// $newData = $this->getPostedData();
e107::getAdminLog()->addArray($data,$oldData);
// $this->addMessageDebug("NEWD".print_a($data,true));
$tp = e107::getParser();
foreach ($data as $field => $dt)
{
// get values form validated array when possible
// we need it because of advanced validation methods e.g. 'compare'
// FIX - security issue, toDb required
if(isset($valid_data[$field])) $dt = $tp->toDb($valid_data[$field]);
$this->setData($field, $dt, $strict)
->removePostedData($field);
}
2014-02-25 11:43:28 +02:00
return $this;
}
/**
* Merge passed data array with the object data
* Should be used on edit/update/create record (back-end)
*
* If $strict is true, only existing object data will be copied (update)
* If $validate is true, data will be copied only after successful validation
*
* @param array $src_data
* @param boolean $sanitize
* @param boolean $validate perform validation check
* @return e_front_model
*/
public function mergeData(array $src_data, $strict = true, $sanitize = true, $validate = true)
{
//FIXME
if(!$src_data || ($validate && !$this->validate($src_data)))
{
return $this;
}
/* Wrong?
// retrieve only valid data
if($validate)
{
$src_data = $this->getValidator()->getValidData();
}*/
if($sanitize)
{
// search for db_field types
2009-10-22 14:18:18 +00:00
if($this->getDataFields())
{
$src_data = $this->sanitize($src_data);
}
else //no db field types, use toDB()
{
$src_data = e107::getParser()->toDB($src_data);
}
}
2014-01-17 16:46:24 -08:00
foreach ($src_data as $key => $value)
{
$this->setData($key, $value, $strict);
}
2014-02-25 11:43:28 +02:00
return $this;
}
/**
* Validate posted data:
* 1. validate posted data against object validation rules
* 2. add validation errors to the object if any
* 3. return true for valid and false for non-valid data
*
* @param array $data optional - data for validation, defaults to posted data
* @return boolean
*/
public function validate($data = null)
{
if(!$this->getValidationRules())
{
return true;
}
if(null === $data)
{
$data = $this->getPostedData();
}
2014-02-25 11:43:28 +02:00
// New param to control validate process - useful when part of the data is going to be updated
// Use it with cautious!!!
$availableOnly = false;
2014-02-25 11:43:28 +02:00
if($this->getParam('validateAvailable'))
{
$availableOnly = true;
$this->setParam('validateAvailable', null); // reset it
}
2014-02-25 11:43:28 +02:00
return $this->getValidator()->validate($data, $availableOnly);
}
2009-10-21 09:12:12 +00:00
/**
* User defined model validation
* Awaiting for child class implementation
*
*/
public function verify()
{
}
/**
* @return e_validator
*/
public function getValidator()
{
if(null === $this->_validator)
{
2009-10-21 10:24:32 +00:00
$this->_validator = e107::getObject('e_validator');
$this->_validator->setRules($this->getValidationRules())
->setOptionalRules($this->getOptionalRules())
->setMessageStack($this->_message_stack.'_validator');
//TODO - optional check rules
}
return $this->_validator;
}
/**
* Add custom validation message.
* $field_type and $error_code will be inserted via $tp->lanVars()
* in the $message string
* Example:
* <code>
* $model->addValidationError('Custom error message [#%d] for %s', 'My Field', 1000);
* //produces 'Custom error message [#1000] for My Field'
* </code>
*
* @param string $message
* @param string $field_title [optional]
* @param integer $error_code [optional]
2009-12-13 21:52:32 +00:00
* @return object
*/
2012-04-03 14:37:48 +00:00
public function addValidationError($message, $field_title = '', $error_code = 0)
{
$this->getValidator()->addValidateMessage($field_title, $error_code, $message)->setIsValidData(false);
return $this;
}
/**
* Render validation errors (if any)
*
* @param boolean $session store messages to session
* @param boolean $reset reset errors
* @return string
*/
public function renderValidationErrors($session = false, $reset = true)
{
return $this->getValidator()->renderValidateMessages($session, $reset);
}
2009-10-21 09:12:12 +00:00
/**
* Render System messages (if any)
*
2009-10-21 09:12:12 +00:00
* @param boolean $validation render validation messages as well
* @param boolean $session store messages to session
* @param boolean $reset reset errors
* @return string
*/
public function renderMessages($validation = true, $session = false, $reset = true)
{
if($validation)
{
e107::getMessage()->moveStack($this->_message_stack.'_validator', $this->_message_stack, false, $session);
}
return parent::renderMessages($session, $reset);
}
2009-10-22 14:18:18 +00:00
/**
* Move model System messages (if any) to the default eMessage stack
*
2009-10-22 14:18:18 +00:00
* @param boolean $session store messages to session
* @param boolean $validation move validation messages as well
* @return e_front_model
2009-10-22 14:18:18 +00:00
*/
public function setMessages($session = false, $validation = true)
2009-10-22 14:18:18 +00:00
{
if($validation)
{
e107::getMessage()->moveStack($this->_message_stack.'_validator', 'default', false, $session);
}
parent::setMessages($session);
return $this;
}
/**
* Reset model System messages
*
* @param boolean|string $type E_MESSAGE_INFO | E_MESSAGE_SUCCESS | E_MESSAGE_WARNING | E_MESSAGE_WARNING | E_MESSAGE_DEBUG | false (all)
* @param boolean $session reset session messages
* @param boolean $validation reset validation messages as well
* @return e_front_model
*/
public function resetMessages($type = false, $session = false, $validation = false)
{
if($validation)
{
e107::getMessage()->reset($type, $this->_message_stack.'_validator', $session);
}
parent::resetMessages($type, $session);
return $this;
}
/**
* @return boolean
*/
public function hasValidationError()
{
return $this->getValidator()->isValid();
}
/**
* @return boolean
*/
public function hasSqlError()
{
return !empty($this->_db_errno);
}
/**
* @return integer last mysql error number
*/
public function getSqlErrorNumber()
{
return $this->_db_errno;
}
/**
* @return string last mysql error message
*/
public function getSqlError()
{
2010-03-01 12:43:56 +00:00
return $this->_db_errmsg;
}
/**
* @return string last mysql error message
*/
public function getSqlQuery()
{
return $this->_db_qry;
}
/**
* @return boolean
*/
public function hasError()
{
2009-10-21 11:57:15 +00:00
return ($this->hasValidationError() || $this->hasSqlError());
}
2009-10-30 17:59:32 +00:00
/**
* Generic load data from DB
* @param boolean $force
* @return e_front_model
2009-10-30 17:59:32 +00:00
*/
2016-02-14 19:00:12 -08:00
public function load($id=null, $force = false)
2009-10-30 17:59:32 +00:00
{
parent::load($id, $force);
$sql = e107::getDb();
$this->_db_errno = $sql->getLastErrorNumber();
2010-03-01 12:43:56 +00:00
$this->_db_errmsg = $sql->getLastErrorText();
$this->_db_qry = $sql->getLastQuery();
2009-10-30 17:59:32 +00:00
if($this->_db_errno)
{
$this->addMessageError('SQL Select Error', $session_messages); //TODO - Lan
// already done by the parent
//$this->addMessageDebug('SQL Error #'.$this->_db_errno.': '.$sql->getLastErrorText());
2009-10-30 17:59:32 +00:00
}
return $this;
}
/**
* Build query array to be used with db methods (db_Update, db_Insert, db_Replace)
*
* @param string $force [optional] force action - possible values are create|update|replace
* @return array db query
*/
public function toSqlQuery($force = '')
{
$qry = array();
if($force)
{
$action = $force;
}
else
{
$action = $this->getId() ? 'update' : 'create';
}
$qry['_FIELD_TYPES'] = $this->_FIELD_TYPES; //DB field types are optional
2014-02-25 11:43:28 +02:00
// support for tables with no auto-increment PK
$id = $this->getId();
$qry['data'][$this->getFieldIdName()] = $id;
2014-02-25 11:43:28 +02:00
//XXX This check is done in _setModel() of admin-ui. NULL below will break MySQL strict.
// Allow admin config to specify the best data type.
/*
if($action == 'create' && !$id) $qry['_FIELD_TYPES'][$this->getFieldIdName()] = 'NULL';
elseif(is_numeric($id)) $qry['_FIELD_TYPES'][$this->getFieldIdName()] = 'integer';
else $qry['_FIELD_TYPES'][$this->getFieldIdName()] = 'string';
*/
2014-02-25 11:43:28 +02:00
foreach ($this->_data_fields as $key => $type)
{
2014-02-25 11:43:28 +02:00
if(!isset($qry['_FIELD_TYPES'][$key]))
{
$qry['_FIELD_TYPES'][$key] = $type; //_FIELD_TYPES much more optional now...
}
2014-02-25 11:43:28 +02:00
if($qry['_FIELD_TYPES'][$key] == 'set') //new 'set' type, could be moved in mysql handler now
{
2014-02-25 11:43:28 +02:00
$qry['_FIELD_TYPES'][$key] = 'str';
if(is_array($this->getData($key))) $this->setData($key, implode(',', $this->getData($key)));
}
$qry['data'][$key] = $this->getData($key);
2014-02-25 11:43:28 +02:00
}
switch($action)
{
case 'create':
//$qry['data'][$this->getFieldIdName()] = NULL;
break;
case 'replace':
$qry['_REPLACE'] = true;
break;
case 'update':
unset($qry['data'][$this->getFieldIdName()]);
if(is_numeric($id)) $id = intval($id);
else $id = "'".e107::getParser()->toDB($id)."'";
2014-02-25 11:43:28 +02:00
$qry['WHERE'] = $this->getFieldIdName().'='.$id;
break;
}
2014-02-25 11:43:28 +02:00
if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
{
$this->addMessageDebug('SQL Qry: '.print_a($qry,true), null);
}
return $qry;
}
/**
* Sanitize value based on its db field type ($_data_fields),
* method will return null only if db field rule is not found.
* If $value is null, it'll be retrieved from object posted data
* If $key is an array, $value is omitted.
*
* NOTE: If $key is not found in object's _data_fields array, null is returned
*
* @param mixed $key string key name or array data to be sanitized
* @param mixed $value
* @return mixed sanitized $value or null on failure
*/
public function sanitize($key, $value = null)
2009-10-30 17:59:32 +00:00
{
$tp = e107::getParser();
if(is_array($key))
2009-10-30 17:59:32 +00:00
{
$ret = array();
foreach ($key as $k=>$v)
2009-10-30 17:59:32 +00:00
{
if(isset($this->_data_fields[$k]))
{
$ret[$k] = $this->sanitize($k, $v);
}
2009-10-30 17:59:32 +00:00
}
return $ret;
2009-10-30 17:59:32 +00:00
}
2014-02-25 11:43:28 +02:00
if(!isset($this->_data_fields[$key]))
{
return null;
}
$type = $this->_data_fields[$key];
if(null === $value)
{
$value = $this->getPostedData($key);
}
switch ($type)
{
case 'int':
case 'integer':
return intval($this->toNumber($value));
break;
case 'safestr':
return $tp->filter($value);
break;
case 'str':
case 'string':
case 'array':
$type = $this->getFieldInputType($key);
return $tp->toDB($value, false, false, 'model', array('type'=>$type, 'field'=>$key));
break;
case 'json':
if(empty($value))
{
return null;
}
return e107::serialize($value,'json');
break;
case 'code':
return $tp->toDB($value, false, false, 'pReFs');
break;
case 'float':
return $this->toNumber($value);
break;
case 'bool':
case 'boolean':
return ($value ? true : false);
break;
case 'model':
return $value->mergePostedData(false, true, true);
break;
case 'null':
return ($value ? $tp->toDB($value) : null);
break;
}
return null;
}
public function destroy()
{
parent::destroy();
$this->_validator = null;
$this->_validation_rules = array();
$this->_db_errno = null;
$this->_posted_data = array();
$this->data_has_changed = array();
$this->_FIELD_TYPES = array();
}
/**
* Update DB data
*
* @param boolean $force force query even if $data_has_changed is false
* @param boolean $session_messages to use or not session to store system messages
*/
protected function dbUpdate($force = false, $session_messages = false)
{
$this->_db_errno = 0;
$this->_db_errmsg = '';
$this->_db_qry = '';
2014-02-25 11:43:28 +02:00
2014-01-17 16:46:24 -08:00
// $this->getData();
// $this->getPostedData();
2014-02-25 11:43:28 +02:00
if($this->hasError()) return false;
if(!$this->data_has_changed && !$force)
{
$this->addMessageInfo(LAN_NO_CHANGE);
return 0;
}
2010-03-01 12:43:56 +00:00
$sql = e107::getDb();
$qry = $this->toSqlQuery('update');
$table = $this->getModelTable();
2014-02-25 11:43:28 +02:00
$res = $sql->db_Update($table, $qry, $this->getParam('db_debug', false));
$this->_db_qry = $sql->getLastQuery();
if(!$res)
{
2010-03-01 12:43:56 +00:00
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
if($this->_db_errno)
{
$this->addMessageError('SQL Update Error', $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$this->_db_errno.': '.$sql->getLastErrorText());
return false;
}
$this->addMessageInfo(LAN_NO_CHANGE);
return 0;
}
$this->clearCache()->addMessageSuccess(LAN_UPDATED);
2014-02-25 11:43:28 +02:00
e107::getAdminLog()->addSuccess('TABLE: '.$table, false);
e107::getAdminLog()->addSuccess('WHERE: '.$qry['WHERE'], false);
2014-01-17 16:46:24 -08:00
e107::getAdminLog()->save('ADMINUI_02');
2014-02-25 11:43:28 +02:00
return $res;
}
/**
* Save data to DB
*
* @param boolen $from_post
* @return boolean|integer
*/
public function save($from_post = true, $force = false, $session_messages = false)
{
if(!$this->getFieldIdName())
{
return false;
}
if($from_post)
{
2014-02-25 11:43:28 +02:00
//no strict copy, validate & sanitize
$this->mergePostedData(false, true, true);
}
if($this->getId())
{
return $this->dbUpdate($force, $session_messages);
}
return false;
}
2014-02-25 11:43:28 +02:00
/**
* Update record
*
* @param boolen $from_post
* @return boolean|integer
*/
public function update($from_post = true, $force = false, $session_messages = false)
{
if(!$this->getFieldIdName())
{
return false;
}
if($from_post)
{
//no strict copy, validate & sanitize
$this->mergePostedData(false, true, true);
}
return $this->dbUpdate($force, $session_messages);
}
/**
* Exactly what it says - your debug helper
* @param boolean $retrun
* @param boolean $undo
* @return void
*/
public function saveDebug($return = false, $undo = true)
{
$ret = array();
$ret['validation_rules'] = $this->getValidationRules();
$ret['optional_validation_rules'] = $this->getOptionalRules();
$ret['model_base_ismodfied'] = $this->isModified();
$ret['model_base_data'] = $this->getData();
$ret['posted_data'] = $this->getPostedData();
$this->mergePostedData(false, true, true);
$ret['model_modified_data'] = $this->getData();
$ret['model_modified_ismodfied'] = $this->isModified();
$ret['validator_valid_data'] = $this->getValidator()->getValidData();
// undo
if($undo)
{
$this->setData($ret['model_base_data'])
->isModified($ret['model_base_ismodfied'])
->setPostedData($ret['posted_data']);
}
if($return) return $ret;
print_a($ret);
}
}
//FIXME - move e_model_admin to e_model_admin.php
/**
* Base e107 Admin Model class
*
* @package e107
* @category e107_handlers
* @version $Id$
* @author SecretR
* @copyright Copyright (C) 2008-2010 e107 Inc.
*/
class e_admin_model extends e_front_model
{
/**
* Save data to DB
*
* @param boolen $from_post
*/
public function save($from_post = true, $force = false, $session_messages = false)
{
if(!$this->getFieldIdName())
{
return false;
}
if($from_post)
{
//no strict copy, validate & sanitize
$this->mergePostedData(false, true, true);
}
if($this->getId() && $this->getPostedData('etrigger_submit') !='create') // Additional Check to allow primary ID to be manually set when auto-increment PID is not used. @see userclass2.php
{
return $this->dbUpdate($force, $session_messages);
}
return $this->dbInsert($session_messages);
}
2014-02-25 11:43:28 +02:00
/**
* Insert record
*
* @param boolen $from_post
* @param boolean $session_messages
* @return integer inserted ID or false on error
*/
public function insert($from_post = true, $session_messages = false)
{
if($from_post)
{
//no strict copy, validate & sanitize
$this->mergePostedData(false, true, true);
}
return $this->dbInsert($session_messages);
}
2016-06-09 16:43:36 -07:00
public function delete($ids, $destroy = true, $session_messages = false)
{
$ret = $this->dbDelete();
if($ret)
{
if($destroy)
{
$this->setMessages($session_messages)->destroy();
}
}
return $ret;
}
/**
* Insert data to DB
* @param boolean $session_messages to use or not session to store system messages
* @return integer
*/
protected function dbInsert($session_messages = false)
{
$this->_db_errno = 0;
$this->_db_errmsg = '';
$this->_db_qry = '';
if($this->hasError()/* || (!$this->data_has_changed && !$force)*/) // not appropriate here!
{
return false;
}
$sql = e107::getDb();
2014-01-17 16:46:24 -08:00
$sqlQry = $this->toSqlQuery('create');
$table = $this->getModelTable();
2014-02-25 11:43:28 +02:00
$res = $sql->insert($table, $sqlQry, $this->getParam('db_debug', false));
$this->_db_qry = $sql->getLastQuery();
if(!$res)
{
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
$this->addMessageError('SQL Insert Error', $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$this->_db_errno.': '.$this->_db_errmsg);
$this->addMessageDebug('SQL QRY Error '.print_a($sqlQry,true));
2014-02-25 11:43:28 +02:00
return false;
}
e107::getAdminLog()->addSuccess('TABLE: '.$table, false);
2014-01-17 16:46:24 -08:00
e107::getAdminLog()->save('ADMINUI_01');
// e107::getAdminLog()->clear()->addSuccess($table,false)->addArray($sqlQry)->save('ADMINUI_01');
2014-02-25 11:43:28 +02:00
// Set the reutrned ID
$this->setId($res);
$this->clearCache()->addMessageSuccess(LAN_CREATED);
return $res;
}
/**
* Replace data in DB
*
* @param boolean $force force query even if $data_has_changed is false
* @param boolean $session_messages to use or not session to store system messages
*/
protected function dbReplace($force = false, $session_messages = false)
{
$this->_db_errno = 0;
$this->_db_errmsg = '';
$this->_db_qry = '';
2014-02-25 11:43:28 +02:00
if($this->hasError()) return false;
if(!$this->data_has_changed && !$force)
2009-10-30 17:59:32 +00:00
{
return 0;
}
2010-03-01 12:43:56 +00:00
$sql = e107::getDb();
$res = $sql->db_Insert($this->getModelTable(), $this->toSqlQuery('replace'));
$this->_db_qry = $sql->getLastQuery();
if(!$res)
{
2010-03-01 12:43:56 +00:00
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
if($this->_db_errno)
{
$this->addMessageError('SQL Replace Error', $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$this->_db_errno.': '.$sql->getLastErrorText());
}
}
else
{
$this->clearCache();
}
2009-10-28 17:05:35 +00:00
return $res;
}
2009-10-28 17:05:35 +00:00
/**
* Delete DB data
*
2009-10-28 17:05:35 +00:00
* @param boolean $force force query even if $data_has_changed is false
* @param boolean $session_messages to use or not session to store system messages
*/
protected function dbDelete($session_messages = false)
2009-10-28 17:05:35 +00:00
{
$this->_db_errno = 0;
2010-03-01 12:43:56 +00:00
$this->_db_errmsg = '';
$this->_db_qry = '';
2014-02-25 11:43:28 +02:00
2009-10-30 17:59:32 +00:00
if($this->hasError())
{
return false;
2009-10-30 17:59:32 +00:00
}
2009-10-28 17:05:35 +00:00
if(!$this->getId())
{
$this->addMessageError('Record not found', $session_messages); //TODO - Lan
return 0;
}
2010-03-01 12:43:56 +00:00
$sql = e107::getDb();
$id = $this->getId();
if(is_numeric($id)) $id = intval($id);
else $id = "'".e107::getParser()->toDB($id)."'";
2014-01-17 16:46:24 -08:00
$table = $this->getModelTable();
$where = $this->getFieldIdName().'='.$id;
$res = $sql->delete($table, $where);
$this->_db_qry = $sql->getLastQuery();
2009-10-28 17:05:35 +00:00
if(!$res)
{
2010-03-01 12:43:56 +00:00
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
2009-10-28 17:05:35 +00:00
if($this->_db_errno)
{
$this->addMessageError('SQL Delete Error', $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$this->_db_errno.': '.$sql->getLastErrorText());
2009-10-28 17:05:35 +00:00
}
}
else
{
if($table != 'admin_log')
{
$logData = array('TABLE'=>$table, 'WHERE'=>$where);
e107::getAdminLog()->addSuccess($table,false);
e107::getAdminLog()->addArray($logData)->save('ADMINUI_03');
}
$this->clearCache();
}
2009-10-28 17:05:35 +00:00
return $res;
}
}
2009-10-28 17:05:35 +00:00
/**
* Model collection handler
*/
class e_tree_model extends e_front_model
2009-10-28 17:05:35 +00:00
{
/**
* Current model DB table, used in all db calls
* This can/should be overwritten by extending the class
*
2009-10-28 17:05:35 +00:00
* @var string
*/
protected $_db_table;
2009-10-28 17:05:35 +00:00
/**
* All records (no limit) cache
*
2009-10-28 17:05:35 +00:00
* @var string
*/
2009-11-17 15:34:54 +00:00
protected $_total = false;
2009-10-30 17:59:32 +00:00
/**
* Constructor
*
*/
function __construct($tree_data = array())
{
if($tree_data)
{
$this->setTree($tree_data);
}
}
2009-10-28 17:05:35 +00:00
public function getTotal()
{
return $this->_total;
2009-10-28 17:05:35 +00:00
}
2009-10-28 17:05:35 +00:00
public function setTotal($num)
{
$this->_total = $num;
return $this;
}
2009-10-28 17:05:35 +00:00
/**
* Set table name
* @param object $table
* @return e_admin_tree_model
*/
public function setModelTable($table)
{
$this->_db_table = $table;
return $this;
}
2009-10-28 17:05:35 +00:00
/**
* Get table name
* @return string
*/
public function getModelTable()
{
return $this->_db_table;
}
2009-10-28 17:05:35 +00:00
/**
* Get array of models
2009-10-28 17:05:35 +00:00
* @return array
*/
function getTree()
2009-10-28 17:05:35 +00:00
{
return $this->get('__tree', array());
2009-10-28 17:05:35 +00:00
}
2009-10-28 17:05:35 +00:00
/**
* Set array of models
* @return e_tree_model
*/
function setTree($tree_data, $force = false)
{
if($force || !$this->isTree())
2009-10-28 17:05:35 +00:00
{
$this->set('__tree', $tree_data);
}
return $this;
}
/**
* Unset all current data
* @return e_tree_model
*/
function unsetTree()
{
$this->remove('__tree');
return $this;
}
2016-02-14 19:00:12 -08:00
public function isCacheEnabled($checkId = true)
{
return (null !== $this->getCacheString());
}
2016-02-14 19:00:12 -08:00
public function getCacheString($replace = false)
{
return $this->_cache_string;
}
protected function _setCacheData()
{
if(!$this->isCacheEnabled())
{
return $this;
}
2014-02-25 11:43:28 +02:00
e107::getCache()->set_sys(
$this->getCacheString(true),
$this->toString(false, null, $this->getParam('nocount') ? false : true),
$this->_cache_force,
false
);
return $this;
}
protected function _loadFromArray($array)
{
if(isset($array['total']))
{
$this->setTotal((integer) $array['total']);
unset($array['total']);
}
$class_name = $this->getParam('model_class', 'e_model');
$tree = array();
foreach ($array as $id => $data)
{
$tree[$id] = new $class_name($data);
$this->_onLoad($tree[$id]);
}
$this->setTree($tree, true);
}
/**
* Additional on load logic to be set from subclasses
*
* @param e_model $node
* @return e_tree_model
*/
protected function _onLoad($node)
{
return $this;
}
2009-10-28 17:05:35 +00:00
/**
* Default load method
*
2009-10-28 17:05:35 +00:00
* @return e_tree_model
*/
2009-10-30 17:59:32 +00:00
public function load($force = false)
2009-10-28 17:05:35 +00:00
{
// XXX What would break if changed to the most proper isTree()?
if(!$force && $this->isTree()) //!$this->isEmpty()
2009-10-30 17:59:32 +00:00
{
return $this;
}
2009-11-17 15:34:54 +00:00
if ($force)
{
$this->unsetTree()
->_clearCacheData();
2009-11-17 15:34:54 +00:00
$this->_total = false;
}
2014-02-25 11:43:28 +02:00
if($this->isCacheEnabled() && !$this->getParam('noCacheStringModify'))
{
2014-02-25 11:43:28 +02:00
$str = !$this->getParam('db_query')
?
$this->getModelTable()
.$this->getParam('nocount')
.$this->getParam('db_where')
.$this->getParam('db_order')
.$this->getParam('db_limit')
:
$this->getParam('db_query');
2014-02-25 11:43:28 +02:00
$this->setCacheString($this->getCacheString().'_'.md5($str));
}
$cached = $this->_getCacheData();
if($cached !== false)
{
$this->_loadFromArray($cached);
return $this;
}
2014-02-25 11:43:28 +02:00
$class_name = $this->getParam('model_class', 'e_model');
// auto-load all
if(!$this->getParam('db_query') && $this->getModelTable())
{
$this->setParam('db_query', 'SELECT'.(!$this->getParam('nocount') ? ' SQL_CALC_FOUND_ROWS' : '')
.($this->getParam('db_cols') ? ' '.$this->getParam('db_cols') : ' *').' FROM #'.$this->getModelTable()
.($this->getParam('db_joins') ? ' '.$this->getParam('db_joins') : '')
.($this->getParam('db_where') ? ' WHERE '.$this->getParam('db_where') : '')
.($this->getParam('db_order') ? ' ORDER BY '.$this->getParam('db_order') : '')
.($this->getParam('db_limit') ? ' LIMIT '.$this->getParam('db_limit') : '')
);
}
2010-05-02 14:04:41 +00:00
if($this->getParam('db_query') && $class_name && class_exists($class_name))
2009-10-28 17:05:35 +00:00
{
$sql = e107::getDb($this->getParam('model_class', 'e_model'));
$this->_total = $sql->total_results = false;
if($sql->gen($this->getParam('db_query'), $this->getParam('db_debug') ? true : false))
2009-10-28 17:05:35 +00:00
{
2009-11-17 15:34:54 +00:00
$this->_total = is_integer($sql->total_results) ? $sql->total_results : false; //requires SQL_CALC_FOUND_ROWS in query - see db handler
2009-10-28 17:05:35 +00:00
while($tmp = $sql->db_Fetch())
{
$tmp = new $class_name($tmp);
if($this->getParam('model_message_stack'))
{
$tmp->setMessageStackName($this->getParam('model_message_stack'));
}
$this->_onLoad($tmp)->setNode($tmp->get($this->getFieldIdName()), $tmp);
2009-10-28 17:05:35 +00:00
}
if(false === $this->_total && $this->getModelTable() && !$this->getParam('nocount'))
2009-10-30 17:59:32 +00:00
{
//SQL_CALC_FOUND_ROWS not found in the query, do one more query
// $this->_total = e107::getDb()->db_Count($this->getModelTable()); // fails with specific listQry
2014-02-25 11:43:28 +02:00
// Calculates correct total when using filters and search. //XXX Optimize.
$countQry = preg_replace('/(LIMIT ([\d,\s])*)$/', "", $this->getParam('db_query'));
2014-02-25 11:43:28 +02:00
$this->_total = e107::getDb()->gen($countQry);
2014-02-25 11:43:28 +02:00
2009-10-30 17:59:32 +00:00
}
2009-10-28 17:05:35 +00:00
unset($tmp);
}
if($sql->getLastErrorNumber())
{
// TODO - admin log?
$this->addMessageError('Application Error - DB query failed.') // TODO LAN
->addMessageDebug('SQL Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText())
->addMessageDebug($sql->getLastQuery());
}
else
{
$this->_setCacheData();
}
2009-10-28 17:05:35 +00:00
}
return $this;
}
/**
* Get single model instance from the collection
* @param integer $node_id
* @return e_model
*/
function getNode($node_id)
{
return $this->getData('__tree/'.$node_id);
}
2009-10-28 17:05:35 +00:00
/**
* Add or remove (when $node is null) model to the collection
*
2009-10-28 17:05:35 +00:00
* @param integer $node_id
* @param e_model $node
* @return e_tree_model
*/
function setNode($node_id, $node)
{
if(null === $node)
{
$this->removeData('__tree/'.$node_id);
return $this;
}
2009-10-28 17:05:35 +00:00
$this->setData('__tree/'.$node_id, $node);
return $this;
}
2009-10-28 17:05:35 +00:00
/**
* Check if model with passed id exists in the collection
*
2009-10-28 17:05:35 +00:00
* @param integer $node_id
* @return boolean
*/
public function isNode($node_id)
{
return $this->isData('__tree/'.$node_id);
}
2009-10-28 17:05:35 +00:00
/**
* Check if model with passed id exists in the collection and is not empty
*
2009-10-28 17:05:35 +00:00
* @param integer $node_id
* @return boolean
*/
public function hasNode($node_id)
{
return $this->hasData('__tree/'.$node_id);
}
2009-10-28 17:05:35 +00:00
/**
* Check if collection is empty
*
* @return boolean
*/
function isEmpty()
{
2009-10-30 17:59:32 +00:00
return (!$this->has('__tree'));
2009-10-28 17:05:35 +00:00
}
/**
* Check if collection is loaded (not null)
*
* @return boolean
*/
function isTree()
{
return $this->is('__tree');
}
/**
* Same as isEmpty(), but with opposite boolean logic
*
* @return boolean
*/
function hasTree()
{
return $this->has('__tree');
}
/**
* Render model data, all 'sc_*' methods will be recongnized
* as shortcodes.
*
* @param string $template
* @param boolean $parsesc parse external shortcodes, default is true
* @param e_vars $eVars simple parser data
* @return string parsed template
*/
public function toHTML($template, $parsesc = true, $eVars = null)
{
$ret = '';
2015-02-15 02:37:36 -08:00
$i = 1;
foreach ($this->getTree() as $model)
{
if($eVars) $eVars->treeCounter = $i;
$ret .= $model->toHTML($template, $parsesc, $eVars);
$i++;
}
return $ret;
}
public function toXML()
{
return '';
// UNDER CONSTRUCTION
}
/**
* Convert model object to array
* @param boolean $total include total results property
* @return array object data
*/
public function toArray($total = false)
{
$ret = array();
foreach ($this->getTree() as $id => $model)
{
$ret[$id] = $model->toArray();
}
if($total) $ret['total'] = $this->getTotal();
return $ret;
}
/**
* Convert object data to a string
*
* @param boolean $AddSlashes
* @param string $node_id optional, if set method will return corresponding value as a string
* @param boolean $total include total results property
* @return string
*/
public function toString($AddSlashes = true, $node_id = null, $total = false)
{
if (null !== $node_id && $this->isNode($node_id))
{
return $this->getNode($node_id)->toString($AddSlashes);
}
return (string) e107::getArrayStorage()->WriteArray($this->toArray($total), $AddSlashes);
}
2016-02-14 19:00:12 -08:00
public function update($from_post = true, $force = false, $session_messages = false)
{
}
2016-02-14 19:00:12 -08:00
public function delete($ids, $destroy = true, $session_messages = false)
{
}
2009-10-28 17:05:35 +00:00
}
class e_front_tree_model extends e_tree_model
2009-10-28 17:05:35 +00:00
{
2010-03-01 12:43:56 +00:00
/**
* @var integer Last SQL error number
*/
protected $_db_errno = 0;
/**
* @var string Last SQL error message
*/
protected $_db_errmsg = '';
/**
* @var string Last SQL query
*/
protected $_db_qry = '';
2010-03-01 12:43:56 +00:00
/**
* @return boolean
*/
public function hasSqlError()
{
return !empty($this->_db_errno);
}
/**
* @return integer last mysql error number
*/
public function getSqlErrorNumber()
{
return $this->_db_errno;
}
/**
* @return string last mysql error message
*/
public function getSqlError()
{
return $this->_db_errmsg;
}
/**
* @return string last mysql error message
*/
public function getSqlQuery()
{
return $this->_db_qry;
}
2010-03-01 12:43:56 +00:00
/**
* @return boolean
*/
public function hasError()
{
return $this->hasSqlError();
}
/**
* Batch update tree records/nodes
* @param string $field field name
* @param string $value
* @param string|array $ids numerical array or string comma separated ids
* @param mixed $syncvalue value to be used for model data synchronization (db value could be something like '1-field_name'), null - no sync
* @param boolean $sanitize [optional] default true
* @param boolean $session_messages [optional] default false
* @return integer updated count or false on error
*/
public function update($field, $value, $ids, $syncvalue = null, $sanitize = true, $session_messages = false)
{
$tp = e107::getParser();
$sql = e107::getDb();
if(empty($ids))
{
return 0;
}
if(!is_array($ids))
{
$ids = explode(',', $ids);
}
if(true === $syncvalue)
{
$syncvalue = $value;
}
2014-02-25 11:43:28 +02:00
if($sanitize)
{
$ids = array_map(array($tp, 'toDB'), $ids);
$field = $tp->toDb($field);
$value = "'".$tp->toDB($value)."'";
}
$idstr = implode(', ', $ids);
2013-10-16 18:13:21 +03:00
$res = $sql->db_Update($this->getModelTable(), "{$field}={$value} WHERE ".$this->getFieldIdName().' IN ('.$idstr.')', $this->getParam('db_debug', false));
2010-03-01 12:43:56 +00:00
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
$this->_db_qry = $sql->getLastQuery();
2014-02-25 11:43:28 +02:00
if(!$res)
{
if($sql->getLastErrorNumber())
{
$this->addMessageError(LAN_UPDATED_FAILED, $session_messages);
$this->addMessageDebug('SQL Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
}
else
{
$this->addMessageInfo(LAN_NO_CHANGE, $session_messages);
}
}
else
{
$this->clearCache();
}
$modelCacheCheck = $this->getParam('clearModelCache');
if(null === $syncvalue && !$modelCacheCheck) return $res;
foreach ($ids as $id)
{
$node = $this->getNode($id);
if(!$node) continue;
2014-02-25 11:43:28 +02:00
if(null !== $syncvalue)
{
$node->set($field, $syncvalue)
->setMessages($session_messages);
}
if($modelCacheCheck) $this->clearCache();
}
return $res;
}
}
class e_admin_tree_model extends e_front_tree_model
{
/**
* Batch Delete records
* @param mixed $ids
* @param boolean $destroy [optional] destroy object instance after db delete
* @param boolean $session_messages [optional]
* @return integer deleted records number or false on DB error
*/
public function delete($ids, $destroy = true, $session_messages = false)
{
if(!$ids) return 0;
if(!is_array($ids))
{
$ids = explode(',', $ids);
}
$tp = e107::getParser();
$ids = array_map(array($tp, 'toDB'), $ids);
$idstr = implode(', ', $ids);
$sql = e107::getDb();
2014-01-17 16:46:24 -08:00
$table = $this->getModelTable();
$sqlQry = $this->getFieldIdName().' IN (\''.$idstr.'\')';
2014-02-25 11:43:28 +02:00
$res = $sql->delete($table, $sqlQry);
2014-02-25 11:43:28 +02:00
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
$this->_db_qry = $sql->getLastQuery();
2014-02-25 11:43:28 +02:00
$modelCacheCheck = $this->getParam('clearModelCache');
2014-02-25 11:43:28 +02:00
if(!$res)
{
if($sql->getLastErrorNumber())
{
$this->addMessageError('SQL Delete Error: ' . $sql->getLastQuery(), $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
}
}
elseif($destroy || $modelCacheCheck)
{
foreach ($ids as $id)
{
if($this->hasNode($id))
{
$this->getNode($id)->clearCache()->setMessages($session_messages);
if($destroy)
{
call_user_func(array($this->getNode(trim($id)), 'destroy')); // first call model destroy method if any
$this->setNode($id, null);
}
}
}
}
if($table != 'admin_log')
{
$logData = array('TABLE'=>$table, 'WHERE'=>$sqlQry);
e107::getAdminLog()->addArray($logData)->save('ADMINUI_03');
}
return $res;
}
/**
2014-02-25 11:43:28 +02:00
* Batch Copy Table Rows.
*/
2014-02-26 16:56:17 +02:00
public function copy($ids, $session_messages = false)
{
$tp = e107::getParser();
$ids = array_map(array($tp, 'toDB'), $ids);
$idstr = implode(', ', $ids);
$sql = e107::getDb();
$res = $sql->db_CopyRow($this->getModelTable(), "*", $this->getFieldIdName().' IN ('.$idstr.')');
if(false !== $res)
{
$this->addMessageSuccess('Copied #'.$idstr);
}
else
{
if($sql->getLastErrorNumber())
{
$this->addMessageError('SQL Copy Error', $session_messages); //TODO - Lan
$this->addMessageDebug('SQL Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
2014-02-25 11:43:28 +02:00
}
}
$this->_db_errno = $sql->getLastErrorNumber();
$this->_db_errmsg = $sql->getLastErrorText();
$this->_db_qry = $sql->getLastQuery();
2014-02-25 11:43:28 +02:00
return $res;
}
2014-02-25 11:43:28 +02:00
/**
* Get urls/url data for given nodes
*/
public function url($ids, $options = array(), $extended = false)
{
$ret = array();
2014-02-25 11:43:28 +02:00
foreach ($ids as $id)
{
if(!$this->hasNode($id)) continue;
2014-02-25 11:43:28 +02:00
$model = $this->getNode($id);
if($this->getUrl()) $model->setUrl($this->getUrl()); // copy url config data if available
2016-06-09 16:43:36 -07:00
$ret[$id] = $model->url(null, $options, $extended);
}
return $ret;
}
/**
* Export Selected Data
* @param $ids
* @return null
*/
public function export($ids)
{
$ids = e107::getParser()->filter($ids,'int');
if(empty($ids))
{
return false;
}
$idstr = implode(', ', $ids);
$table = array($this->getModelTable());
$filename = "e107Export_" .$this->getModelTable()."_". date("YmdHi").".xml";
$query = $this->getFieldIdName().' IN ('.$idstr.') '; // ORDER BY '.$this->getParam('db_order') ;
e107::getXML()->e107Export(null,$table,null,array('file'=>$filename,'query'=>$query));
return null;
}
}