File: /www/wwwroot/oa.sanjiangapp.com/lib/base/dao/dao.class.php
<?php
/**
* ZenTaoPHP的dao和sql类。
* The dao and sql class file of ZenTaoPHP framework.
*
* The author disclaims copyright to this source code. In place of
* a legal notice, here is a blessing:
*
* May you do good and not evil.
* May you find forgiveness for yourself and forgive others.
* May you share freely, never taking more than you give.
*/
/**
* DAO类。
* DAO, data access object.
*
* @package framework
*/
class baseDAO
{
/* Use these strang strings to avoid conflicting with these keywords in the sql body. */
const WHERE = 'wHeRe';
const GROUPBY = 'gRoUp bY';
const HAVING = 'hAvInG';
const ORDERBY = 'oRdEr bY';
const LIMIT = 'lImiT';
/**
* 全局对象$app
* The global app object.
*
* @var object
* @access public
*/
public $app;
/**
* 全局对象$config
* The global config object.
*
* @var object
* @access public
*/
public $config;
/**
* 全局对象$lang
* The global lang object.
*
* @var object
* @access public
*/
public $lang;
/**
* 全局对象$dbh
* The global dbh(database handler) object.
*
* @var object
* @access public
*/
public $dbh;
/**
* 全局对象$slaveDBH。
* The global slaveDBH(database handler) object.
*
* @var object
* @access public
*/
public $slaveDBH;
/**
* sql对象,用于生成sql语句。
* The sql object, used to create the query sql.
*
* @var object
* @access public
*/
public $sqlobj;
/**
* 正在使用的表。
* The table of current query.
*
* @var string
* @access public
*/
public $table;
/**
* $this->table的别名。
* The alias of $this->table.
*
* @var string
* @access public
*/
public $alias;
/**
* 查询的字段。
* The fields will be returned.
*
* @var string
* @access public
*/
public $fields;
/**
* 查询模式,raw模式用于正常的select update等sql拼接操作,magic模式用于findByXXX等魔术方法。
* The query mode, raw or magic.
*
* This var is used to diff dao::from() with sql::from().
*
* @var string
* @access public
*/
public $mode;
/**
* 执行方式:insert, select, update, delete, replace。
* The query method: insert, select, update, delete, replace.
*
* @var string
* @access public
*/
public $method;
/**
* 是否自动增加lang条件。
* If auto add lang statement.
*
* @var bool
* @access public
*/
public $autoLang;
/**
* 需要修复表的错误代码
* The sql code of need repair table.
*
* @var string
* @access public
*/
public $repairCode = '|1034|1035|1194|1195|1459|';
/**
* 执行的请求,所有的查询都保存在该数组。
* The queries executed. Every query will be saved in this array.
*
* @var array
* @access public
*/
static public $querys = array();
/**
* 存放错误的数组。
* The errors.
*
* @var array
* @access public
*/
static public $errors = array();
/**
* 缓存。
* The cache.
*
* @var array
* @access public
*/
static public $cache = array();
/**
* 构造方法。
* The construct method.
*
* @access public
* @return void
*/
public function __construct()
{
global $app, $config, $lang, $dbh, $slaveDBH;
$this->app = $app;
$this->config = $config;
$this->lang = $lang;
$this->dbh = $dbh;
$this->slaveDBH = $slaveDBH ? $slaveDBH : false;
$this->reset();
}
/**
* 设置$table属性。
* Set the $table property.
*
* @param string $table
* @access public
* @return void
*/
public function setTable($table)
{
$this->table = $table;
}
/**
* 设置$alias属性。
* Set the $alias property.
*
* @param string $alias
* @access public
* @return void
*/
public function setAlias($alias)
{
$this->alias = $alias;
}
/**
* 设置$fields属性。
* Set the $fields property.
*
* @param string $fields
* @access public
* @return void
*/
public function setFields($fields)
{
$this->fields = $fields;
}
/**
* 设置autoLang项。
* Set autoLang item.
*
* @param bool $autoLang
* @access public
* @return void
*/
public function setAutoLang($autoLang)
{
$this->autoLang = $autoLang;
return $this;
}
/**
* 重置属性。
* Reset the vars.
*
* @access public
* @return void
*/
public function reset()
{
$this->setFields('');
$this->setTable('');
$this->setAlias('');
$this->setMode('');
$this->setMethod('');
$this->setAutoLang(isset($this->config->framework->autoLang) and $this->config->framework->autoLang);
}
//-----根据请求的方式,调用sql类相应的方法(Call according method of sql class by query method. -----//
/**
* 设置请求模式。像findByxxx之类的方法,使用的是magic模式;其他方法使用的是raw模式。
* Set the query mode. If the method if like findByxxx, the mode is magic. Else, the mode is raw.
*
* @param string $mode magic|raw
* @access public
* @return void
*/
public function setMode($mode = '')
{
$this->mode = $mode;
}
/**
* 设置请求方法:select|update|insert|delete|replace 。
* Set the query method: select|update|insert|delete|replace
*
* @param string $method
* @access public
* @return void
*/
public function setMethod($method = '')
{
$this->method = $method;
}
/**
* 开始事务。
* Begin Transaction
*
* @access public
* @return void
*/
public function begin()
{
$this->dbh->beginTransaction();
}
/**
* 事务回滚。
* Roll back
*
* @access public
* @return void
*/
public function rollBack()
{
$this->dbh->rollBack();
}
/**
* 提交事务。
* Commits a transaction.
*
* @access public
* @return void
*/
public function commit()
{
$this->dbh->commit();
}
/**
* select方法,调用sql::select()。
* The select method, call sql::select().
*
* @param string $fields
* @access public
* @return object the dao object self.
*/
public function select($fields = '*')
{
$this->setMode('raw');
$this->setMethod('select');
$this->sqlobj = sql::select($fields);
return $this;
}
/**
* 获取查询记录条数。
* The count method, call sql::select() and from().
* use as $this->dao->select()->from(TABLE_BUG)->where()->count();
*
* @param string $distinctField
* @access public
* @return void
*/
public function count($distinctField = '')
{
/* 获得SELECT,FROM的位置,使用count(*)替换其字段。 */
/* Get the SELECT, FROM position, thus get the fields, replace it by count(*). */
$sql = $this->get();
$selectPOS = strpos($sql, 'SELECT') + strlen('SELECT');
$fromPOS = strpos($sql, 'FROM');
$fields = substr($sql, $selectPOS, $fromPOS - $selectPOS);
$countField = $distinctField ? 'distinct ' . $distinctField : '*';
$sql = str_replace($fields, " COUNT($countField) AS recTotal ", substr($sql, 0, $fromPOS)) . substr($sql, $fromPOS);
/*
* 去掉SQL语句中order和limit之后的部分。
* Remove the part after order and limit.
**/
$subLength = strlen($sql);
$orderPOS = strripos($sql, 'order by');
$limitPOS = strripos($sql, 'limit');
if($limitPOS) $subLength = $limitPOS;
if($orderPOS) $subLength = $orderPOS;
$sql = substr($sql, 0, $subLength);
self::$querys[] = $sql;
/*
* 获取记录数。
* Get the records count.
**/
try
{
$row = $this->dbh->query($sql)->fetch(PDO::FETCH_OBJ);
}
catch (PDOException $e)
{
$this->sqlError($e);
}
return is_object($row) ? $row->recTotal : 0;
}
/**
* update方法,调用sql::update()。
* The update method, call sql::update().
*
* @param string $table
* @access public
* @return object the dao object self.
*/
public function update($table)
{
$this->setMode('raw');
$this->setMethod('update');
$this->sqlobj = sql::update($table);
$this->setTable($table);
return $this;
}
/**
* delete方法,调用sql::delete()。
* The delete method, call sql::delete().
*
* @access public
* @return object the dao object self.
*/
public function delete()
{
$this->setMode('raw');
$this->setMethod('delete');
$this->sqlobj = sql::delete();
return $this;
}
/**
* insert方法,调用sql::insert()。
* The insert method, call sql::insert().
*
* @param string $table
* @access public
* @return object the dao object self.
*/
public function insert($table)
{
$this->setMode('raw');
$this->setMethod('insert');
$this->sqlobj = sql::insert($table);
$this->setTable($table);
return $this;
}
/**
* replace方法,调用sql::replace()。
* The replace method, call sql::replace().
*
* @param string $table
* @access public
* @return object the dao object self.
*/
public function replace($table)
{
$this->setMode('raw');
$this->setMethod('replace');
$this->sqlobj = sql::replace($table);
$this->setTable($table);
return $this;
}
/**
* 设置要操作的表。
* Set the from table.
*
* @param string $table
* @access public
* @return object the dao object self.
*/
public function from($table)
{
$this->setTable($table);
if($this->mode == 'raw') $this->sqlobj->from($table);
return $this;
}
/**
* 设置字段。
* Set the fields.
*
* @param string $fields
* @access public
* @return object the dao object self.
*/
public function fields($fields)
{
$this->setFields($fields);
return $this;
}
/**
* 表别名,相当于sql里的AS。(as是php的关键词,使用alias代替)
* Alias a table, equal the AS keyword. (Don't use AS, because it's a php keyword.)
*
* @param string $alias
* @access public
* @return object the dao object self.
*/
public function alias($alias)
{
if(empty($this->alias)) $this->setAlias($alias);
$this->sqlobj->alias($alias);
return $this;
}
/**
* 设置需要更新或插入的数据。
* Set the data to update or insert.
*
* @param object $data the data object or array
* @access public
* @return object the dao object self.
*/
public function data($data, $skipFields = '')
{
if(!is_object($data)) $data = (object)$data;
if($this->autoLang and !isset($data->lang))
{
$data->lang = $this->app->getClientLang();
if(isset($this->app->config->cn2tw) and $this->app->config->cn2tw and $data->lang == 'zh-tw') $data->lang = 'zh-cn';
if(defined('RUN_MODE') and RUN_MODE == 'front' and !empty($this->app->config->cn2tw)) $data->lang = str_replace('zh-tw', 'zh-cn', $data->lang);
}
$this->sqlobj->data($data, $skipFields);
return $this;
}
//-------------------- sql相关的方法(The sql related method) --------------------//
/**
* 获取sql字符串。
* Get the sql string.
*
* @access public
* @return string the sql string after process.
*/
public function get()
{
return $this->processKeywords($this->processSQL());
}
/**
* 打印sql字符串。
* Print the sql string.
*
* @access public
* @return void
*/
public function printSQL()
{
echo $this->processSQL();
}
/**
* 查看SQL索引。
* Explain sql.
*
* @param string $sql
* @access public
* @return void
*/
public function explain($sql = '')
{
$sql = empty($sql) ? $this->processSQL() : $sql;
$result = $this->dbh->query('explain ' . $sql)->fetch();
a($result);
}
/**
* 处理sql语句,替换表和字段。
* Process the sql, replace the table, fields.
*
* @access public
* @return string the sql string after process.
*/
public function processSQL()
{
$sql = $this->sqlobj->get();
/**
* 如果是magic模式,处理表和字段。
* If the mode is magic, process the $fields and $table.
**/
if($this->mode == 'magic')
{
if($this->fields == '') $this->fields = '*';
if($this->table == '') $this->app->triggerError('Must set the table name', __FILE__, __LINE__, $exit = true);
$sql = sprintf($this->sqlobj->get(), $this->fields, $this->table);
}
/* If the method if select, update or delete, set the lang condition. */
if($this->autoLang and $this->table != '' and $this->method != 'insert' and $this->method != 'replace')
{
$lang = $this->app->getClientLang();
/* Get the position to insert lang = ?. */
$wherePOS = strrpos($sql, DAO::WHERE); // The position of WHERE keyword.
$groupPOS = strrpos($sql, DAO::GROUPBY); // The position of GROUP BY keyword.
$havingPOS = strrpos($sql, DAO::HAVING); // The position of HAVING keyword.
$orderPOS = strrpos($sql, DAO::ORDERBY); // The position of ORDERBY keyword.
$limitPOS = strrpos($sql, DAO::LIMIT); // The position of LIMIT keyword.
$splitPOS = $orderPOS ? $orderPOS : $limitPOS; // If $orderPOS, use it instead of $limitPOS.
$splitPOS = $havingPOS? $havingPOS: $splitPOS; // If $havingPOS, use it instead of $orderPOS.
$splitPOS = $groupPOS ? $groupPOS : $splitPOS; // If $groupPOS, use it instead of $havingPOS.
/* Set the conditon to be appened. */
$tableName = !empty($this->alias) ? $this->alias : $this->table;
if(!empty($this->app->config->cn2tw)) $lang = str_replace('zh-tw', 'zh-cn', $lang);
$langCondition = " $tableName.lang in('{$lang}', 'all') ";
/* If $spliPOS > 0, split the sql at $splitPOS. */
if($splitPOS)
{
$firstPart = substr($sql, 0, $splitPOS);
$lastPart = substr($sql, $splitPOS);
if($wherePOS)
{
$sql = $firstPart . " AND $langCondition " . $lastPart;
}
else
{
$sql = $firstPart . " WHERE $langCondition " . $lastPart;
}
}
else
{
$sql .= $wherePOS ? " AND $langCondition" : " WHERE $langCondition";
}
}
self::$querys[] = $this->processKeywords($sql);
return $sql;
}
/**
* 替换sql常量关键字。
* Process the sql keywords, replace the constants to normal.
*
* @param string $sql
* @access public
* @return string the sql string.
*/
public function processKeywords($sql)
{
return str_replace(array(DAO::WHERE, DAO::GROUPBY, DAO::HAVING, DAO::ORDERBY, DAO::LIMIT), array('WHERE', 'GROUP BY', 'HAVING', 'ORDER BY', 'LIMIT'), $sql);
}
//-------------------- 查询相关方法(Query related methods) --------------------//
/**
* 设置$dbh,数据库连接句柄。
* Set the dbh.
*
* You can use like this: $this->dao->dbh($dbh), thus you can handle two database.
*
* @param object $dbh
* @access public
* @return object the dao object self.
*/
public function dbh($dbh)
{
$this->dbh = $dbh;
return $this;
}
/**
* 执行SQL语句,返回PDOStatement结果集。
* Query the sql, return the statement object.
*
* @access public
* @return object the PDOStatement object.
*/
public function query($sql = '')
{
/* 如果有错误,返回一个空的PDOStatement对象,确保后续方法能够执行。*/
/* If any error, return an empty statement object to make sure the remain method to execute. */
if(!empty(dao::$errors)) return new PDOStatement();
if($sql)
{
$sql = trim($sql);
$sqlMethod = strtolower(substr($sql, 0, strpos($sql, ' ')));
$this->setMethod($sqlMethod);
$this->sqlobj = new sql();
$this->sqlobj->sql = $sql;
}
else
{
$sql = $this->processSQL();
}
try
{
$method = $this->method;
$this->reset();
if($this->slaveDBH and $method == 'select')
{
return $this->slaveDBH->query($sql);
}
else
{
return $this->dbh->query($sql);
}
}
catch (PDOException $e)
{
$this->sqlError($e);
}
}
/**
* 将记录进行分页,自动设置limit语句。
* Page the records, set the limit part auto.
*
* @param object $pager
* @access public
* @return object the dao object self.
*/
public function page($pager, $distinctField = '')
{
if(!is_object($pager)) return $this;
/*
* 如果$pager的总记录为0,需要计算总结果数。
* If the record total is 0, compute it.
**/
if($pager->recTotal == 0)
{
$recTotal = $this->count($distinctField);
$pager->setRecTotal($recTotal);
$pager->setPageTotal();
}
$this->sqlobj->limit($pager->limit());
return $this;
}
/**
* 执行SQL。query()会返回stmt对象,该方法只返回更改或删除的记录数。
* Execute the sql. It's different with query(), which return the stmt object. But this not.
*
* @param string $sql
* @access public
* @return int the modified or deleted records. 更改或删除的记录数。
*/
public function exec($sql = '')
{
if(!empty(dao::$errors)) return new PDOStatement(); // If any error, return an empty statement object to make sure the remain method to execute.
if($sql)
{
$this->sqlobj = new sql();
$this->sqlobj->sql = $sql;
}
else
{
$sql = $this->processSQL();
}
try
{
if($this->table) unset(dao::$cache[$this->table]);
$this->reset();
return $this->dbh->exec($sql);
}
catch (PDOException $e)
{
$this->sqlError($e);
}
}
//-------------------- Fetch相关方法(Fetch related methods) -------------------//
/**
* 获取一个记录。
* Fetch one record.
*
* @param string $field 如果已经设置获取的字段,则只返回这个字段的值,否则返回这个记录。
* if the field is set, only return the value of this field, else return this record
* @access public
* @return object|mixed
*/
public function fetch($field = '')
{
$sql = $this->processSQL();
$table = $this->table;
$key = 'fetch-' . md5($sql . $field);
if(isset(dao::$cache[$table][$key]))
{
if(empty($field)) return $this->getRow(dao::$cache[$table][$key]);
$result = dao::$cache[$table][$key];
return $result ? $result->$field : '';
}
if(empty($field))
{
$data = $this->query()->fetch();
dao::$cache[$table][$key] = $data;
return $this->getRow($data);
}
$this->setFields($field);
$result = $this->query()->fetch(PDO::FETCH_OBJ);
dao::$cache[$table][$key] = $this->getRow($result);
return $result ? $result->$field : '';
}
/**
* 获取所有记录。
* Fetch all records.
*
* @param string $keyField 返回以该字段做键的记录
* the key field, thus the return records is keyed by this field
* @access public
* @return array the records
*/
public function fetchAll($keyField = '')
{
$sql = $this->processSQL();
$table = $this->table;
$key = 'fetchAll-' . md5($sql . $keyField);
if(isset(dao::$cache[$table][$key]))
{
$rows = dao::$cache[$table][$key];
$result = array();
foreach($rows as $i => $row) $result[$i] = $this->getRow($row);
return $result;
}
$stmt = $this->query();
dao::$cache[$table][$key] = array();
if(empty($keyField))
{
$rows = $stmt->fetchAll();
$result = array();
dao::$cache[$table][$key] = $rows;
if($rows) foreach($rows as $i => $row) $result[$i] = $this->getRow($row);
return $result;
}
$rows = array();
while($row = $stmt->fetch())
{
dao::$cache[$table][$key][$row->$keyField] = $row;
$rows[$row->$keyField] = $this->getRow($row);
}
return $rows;
}
/**
* 获取所有记录并将按照字段分组。
* Fetch all records and group them by one field.
*
* @param string $groupField 分组的字段 the field to group by
* @param string $keyField 键字段 the field of key
* @access public
* @return array the records.
*/
public function fetchGroup($groupField, $keyField = '')
{
$sql = $this->processSQL();
$table = $this->table;
$key = 'fetchGroup-' . md5($sql . $groupField . $keyField);
if(isset(dao::$cache[$table][$key]))
{
$result = array();
$groupRows = dao::$cache[$table][$key];
foreach($groupRows as $groupField => $rows)
{
foreach($rows as $keyField => $row) $result[$groupField][$keyField] = $this->getRow($row);
}
return $result;
}
$stmt = $this->query();
$rows = array();
while($row = $stmt->fetch())
{
empty($keyField) ? $rows[$row->$groupField][] = $row : $rows[$row->$groupField][$row->$keyField] = $this->getRow($row);
}
dao::$cache[$table][$key] = $rows;
return $rows;
}
/**
* 获取的记录是以关联数组的形式
* Fetch array like key=>value.
*
* 如果没有设置参数,用首末两键作为参数。
* If the keyFiled and valueField not set, use the first and last in the record.
*
* @param string $keyField
* @param string $valueField
* @access public
* @return array
*/
public function fetchPairs($keyField = '', $valueField = '')
{
$keyField = trim($keyField, '`');
$valueField = trim($valueField, '`');
$sql = $this->processSQL();
$table = $this->table;
$key = 'fetchPairs-' . md5($sql . $keyField . $valueField);
if(isset(dao::$cache[$table][$key])) return dao::$cache[$table][$key];
$pairs = array();
$ready = false;
$stmt = $this->query();
while($row = $stmt->fetch(PDO::FETCH_ASSOC))
{
if(!$ready)
{
if(empty($keyField)) $keyField = key($row);
if(empty($valueField))
{
end($row);
$valueField = key($row);
}
$ready = true;
}
$pairs[$row[$keyField]] = $row[$valueField];
}
dao::$cache[$table][$key] = $pairs;
return $pairs;
}
/**
* 返回最后插入的ID。
* Return the last insert ID.
*
* @access public
* @return int
*/
public function lastInsertID()
{
return $this->dbh->lastInsertID();
}
/**
* 重新生成数据。
* Get row by data.
*
* @param array/object $data
* @access public
* @return array/object
*/
public function getRow($data)
{
if(!is_object($data)) return $data;
return json_decode(json_encode($data));
}
//-------------------- 魔术方法(Magic methods) --------------------//
/**
* 解析dao的方法名,处理魔术方法。
* Use it to do some convenient queries.
*
* @param string $funcName the function name to be called
* @param array $funcArgs the params
* @access public
* @return object the dao object self.
*/
public function __call($funcName, $funcArgs)
{
$funcName = strtolower($funcName);
/*
* 如果是findByxxx,转换为where条件语句。
* findByxxx, xxx as will be in the where.
**/
if(strpos($funcName, 'findby') !== false)
{
$this->setMode('magic');
$this->setFields('');
$field = str_replace('findby', '', $funcName);
if(count($funcArgs) == 1)
{
$operator = '=';
$value = $funcArgs[0];
}
else
{
$operator = $funcArgs[0];
$value = $funcArgs[1];
}
$this->sqlobj = sql::select('%s')->from('%s')->where($field, $operator, $value);
return $this;
}
/*
* 获取指定个数的记录:fetch10 获取10条记录。
* Fetch10.
**/
elseif(strpos($funcName, 'fetch') !== false)
{
$max = str_replace('fetch', '', $funcName);
$stmt = $this->query();
$rows = array();
$key = isset($funcArgs[0]) ? $funcArgs[0] : '';
$i = 0;
while($row = $stmt->fetch())
{
$key ? $rows[$row->$key] = $row : $rows[] = $row;
$i ++;
if($i == $max) break;
}
return $rows;
}
/*
* 其他的方法,转到sqlobj对象执行。
* Others, call the method in sql class.
**/
else
{
/*
* 使用$arg0, $arg1... 生成调用的参数。
* Create the max counts of sql class methods, and then create $arg0, $arg1...
**/
for($i = 0; $i < SQL::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($funcArgs[$i]) ? $funcArgs[$i] : null;
}
$this->sqlobj->$funcName($arg0, $arg1, $arg2);
return $this;
}
}
//-------------------- 条件检查( Data Checking)--------------------//
/**
* 检查字段是否满足条件。
* Check a filed is satisfied with the check rule.
*
* @param string $fieldName the field to check
* @param string $funcName the check rule
* @param string $condition the condition
* @access public
* @return object the dao object self.
*/
public function check($fieldName, $funcName, $condition = '')
{
/*
* 如果没数据中没有该字段,直接返回。
* If no this field in the data, return.
**/
if(!isset($this->sqlobj->data->$fieldName)) return $this;
/* 设置字段值。 */
/* Set the field label and value. */
global $lang, $config, $app;
if(isset($config->db->prefix))
{
$table = strtolower(str_replace(array($config->db->prefix, '`'), '', $this->table));
}
elseif(strpos($this->table, '_') !== false)
{
$table = strtolower(substr($this->table, strpos($this->table, '_') + 1));
$table = str_replace('`', '', $table);
}
else
{
$table = strtolower($this->table);
}
$fieldLabel = isset($lang->$table->$fieldName) ? $lang->$table->$fieldName : $fieldName;
$value = isset($this->sqlobj->data->$fieldName) ? $this->sqlobj->data->$fieldName : null;
/*
* 检查唯一性。
* Check unique.
**/
if($funcName == 'unique')
{
$args = func_get_args();
$sql = "SELECT COUNT(*) AS count FROM $this->table WHERE `$fieldName` = " . $this->sqlobj->quote($value);
if($condition) $sql .= ' AND ' . $condition;
try
{
$row = $this->dbh->query($sql)->fetch();
if($row->count != 0) $this->logError($funcName, $fieldName, $fieldLabel, array($value));
}
catch (PDOException $e)
{
$this->sqlError($e);
}
}
else
{
/*
* 创建参数。
* Create the params.
**/
$funcArgs = func_get_args();
unset($funcArgs[0]);
unset($funcArgs[1]);
for($i = 0; $i < VALIDATER::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($funcArgs[$i + 2]) ? $funcArgs[$i + 2] : null;
}
$checkFunc = 'check' . $funcName;
if(validater::$checkFunc($value, $arg0, $arg1, $arg2) === false)
{
$this->logError($funcName, $fieldName, $fieldLabel, $funcArgs);
}
}
return $this;
}
/**
* 检查一个字段是否满足条件。
* Check a field, if satisfied with the condition.
*
* @param string $condition
* @param string $fieldName
* @param string $funcName
* @access public
* @return object the dao object self.
*/
public function checkIF($condition, $fieldName, $funcName)
{
if(!$condition) return $this;
$funcArgs = func_get_args();
for($i = 0; $i < VALIDATER::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($funcArgs[$i + 3]) ? $funcArgs[$i + 3] : null;
}
$this->check($fieldName, $funcName, $arg0, $arg1, $arg2);
return $this;
}
/**
* 批量检查字段。
* Batch check some fileds.
*
* @param string $fields the fields to check, join with ,
* @param string $funcName
* @access public
* @return object the dao object self.
*/
public function batchCheck($fields, $funcName)
{
$fields = explode(',', str_replace(' ', '', $fields));
$funcArgs = func_get_args();
for($i = 0; $i < VALIDATER::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($funcArgs[$i + 2]) ? $funcArgs[$i + 2] : null;
}
foreach($fields as $fieldName) $this->check($fieldName, $funcName, $arg0, $arg1, $arg2);
return $this;
}
/**
* 批量检查字段是否满足条件。
* Batch check fields on the condition is true.
*
* @param string $condition
* @param string $fields
* @param string $funcName
* @access public
* @return object the dao object self.
*/
public function batchCheckIF($condition, $fields, $funcName)
{
if(!$condition) return $this;
$fields = explode(',', str_replace(' ', '', $fields));
$funcArgs = func_get_args();
for($i = 0; $i < VALIDATER::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($funcArgs[$i + 3]) ? $funcArgs[$i + 3] : null;
}
foreach($fields as $fieldName) $this->check($fieldName, $funcName, $arg0, $arg1, $arg2);
return $this;
}
/**
* 根据数据库结构检查字段。
* Check the fields according the the database schema.
*
* @param string $skipFields fields to skip checking
* @access public
* @return object the dao object self.
*/
public function autoCheck($skipFields = '')
{
$fields = $this->getFieldsType();
$skipFields = ",$skipFields,";
foreach($fields as $fieldName => $validater)
{
if(strpos($skipFields, $fieldName) !== false) continue; // skip it.
if(!isset($this->sqlobj->data->$fieldName)) continue;
if($validater['rule'] == 'skip') continue;
$options = array();
if(isset($validater['options'])) $options = array_values($validater['options']);
for($i = 0; $i < VALIDATER::MAX_ARGS; $i ++)
{
${"arg$i"} = isset($options[$i]) ? $options[$i] : null;
}
$this->check($fieldName, $validater['rule'], $arg0, $arg1, $arg2);
}
return $this;
}
/**
* 记录错误到日志。
* Log the error.
*
* module/common/lang中定义了错误提示信息。
* For the error notice, see module/common/lang.
*
* @param string $checkType the check rule
* @param string $fieldName the field name
* @param string $fieldLabel the field label
* @param array $funcArgs the args
* @access public
* @return void
*/
public function logError($checkType, $fieldName, $fieldLabel, $funcArgs = array())
{
global $lang;
$error = $lang->error->$checkType;
$replaces = array_merge(array($fieldLabel), $funcArgs); // the replace values.
/*
* 如果$error错误信息是一个字符串,进行替换。
* Just a string, cycle the $replaces.
**/
if(!is_array($error))
{
foreach($replaces as $replace)
{
$pos = strpos($error, '%s');
if($pos === false) break;
$error = substr($error, 0, $pos) . $replace . substr($error, $pos + 2);
}
}
/*
* 如果error错误信息是一个数组,选择一个%s满足替换个数的进行替换。
* If the error define is an array, select the one which %s counts match the $replaces.
**/
else
{
/*
* 去掉空值项。
* Remove the empty items.
**/
foreach($replaces as $key => $value) if(is_null($value)) unset($replaces[$key]);
$replacesCount = count($replaces);
foreach($error as $errorString)
{
if(substr_count($errorString, '%s') == $replacesCount)
{
$error = vsprintf($errorString, $replaces);
}
}
}
dao::$errors[$fieldName][] = $error;
}
/**
* 判断是否有错误。
* Judge any error or not.
*
* @access public
* @return bool
*/
public static function isError()
{
return !empty(dao::$errors);
}
/**
* 获取错误。
* Get the errors.
*
* @access public
* @return array
*/
public static function getError($join = false)
{
$errors = dao::$errors;
dao::$errors = array(); // 清除dao的错误信息(Must clear errors)
if(!$join) return $errors;
if(is_array($errors))
{
$message = '';
foreach($errors as $item)
{
is_array($item) ? $message .= join('\n', $item) . '\n' : $message .= $item . '\n';
}
return $message;
}
}
/**
* 获取表的字段类型。
* Get the defination of fields of the table.
*
* @access public
* @return array
*/
public function getFieldsType()
{
try
{
$this->dbh->setAttribute(PDO::ATTR_CASE, PDO::CASE_LOWER);
$sql = "DESC $this->table";
$rawFields = $this->dbh->query($sql)->fetchAll();
$this->dbh->setAttribute(PDO::ATTR_CASE, PDO::CASE_NATURAL);
}
catch (PDOException $e)
{
$this->sqlError($e);
}
foreach($rawFields as $rawField)
{
$firstPOS = strpos($rawField->type, '(');
$type = substr($rawField->type, 0, $firstPOS > 0 ? $firstPOS : strlen($rawField->type));
$type = str_replace(array('big', 'small', 'medium', 'tiny', 'var'), '', $type);
$field = array();
if($type == 'enum' or $type == 'set')
{
$rangeBegin = $firstPOS + 2; // 移除开始的引用符 Remove the first quote.
$rangeEnd = strrpos($rawField->type, ')') - 1; // 移除结束的引用符 Remove the last quote.
$range = substr($rawField->type, $rangeBegin, $rangeEnd - $rangeBegin);
$field['rule'] = 'reg';
$field['options']['reg'] = '/' . str_replace("','", '|', $range) . '/';
}
elseif($type == 'char')
{
$begin = $firstPOS + 1;
$end = strpos($rawField->type, ')', $begin);
$length = substr($rawField->type, $begin, $end - $begin);
$field['rule'] = 'length';
$field['options']['max'] = $length;
$field['options']['min'] = 0;
}
elseif($type == 'int')
{
$field['rule'] = 'int';
}
elseif($type == 'float' or $type == 'double' or $type == 'decimal')
{
$field['rule'] = 'float';
}
elseif($type == 'date')
{
$field['rule'] = 'date';
}
elseif($type == 'datetime')
{
$field['rule'] = 'datetime';
}
else
{
$field['rule'] = 'skip';
}
$fields[$rawField->field] = $field;
}
return $fields;
}
/**
* Process SQL error by code.
*
* @param object $exception
* @access public
* @return void
*/
public function sqlError($exception)
{
$errorInfo = $exception->errorInfo;
$errorCode = $errorInfo[1];
$errorMsg = $errorInfo[2];
$message = $exception->getMessage();
if(strpos($this->repairCode, "|$errorCode|") !== false or ($errorCode == '1016' and strpos($errorMsg, 'errno: 145') !== false) or strpos($message, 'repair') !== false)
{
global $config;
if(isset($config->framework->autoRepairTable) and $config->framework->autoRepairTable) die(js::locate($config->webRoot . 'checktable.php', 'top'));
$message .= ' ' . $this->lang->repairTable;
}
$sql = $this->sqlobj->get();
$this->app->triggerError($message . "<p>The sql is: $sql</p>", __FILE__, __LINE__, $exit = true);
}
}
/**
* SQL类。
* The SQL class.
*
* @package framework
*/
class baseSQL
{
/**
* 所有方法的最大参数个数。
* The max count of params of all methods.
*
*/
const MAX_ARGS = 3;
/**
* SQL字符串。
* The sql string.
*
* @var string
* @access public
*/
public $sql = '';
/**
* 全局变量$dbh。
* The global $dbh.
*
* @var object
* @access public
*/
public $dbh;
/**
* 更新或插入日期。
* The data to update or insert.
*
* @var mix
* @access public
*/
public $data;
/**
* 是否是第一次设置。
* Is the first time to call set.
*
* @var bool
* @access public;
*/
public $isFirstSet = true;
/**
* 是否是在条件语句中。
* If in the logic of judge condition or not.
*
* @var bool
* @access public;
*/
public $inCondition = false;
/**
* 条件是否为真。
* The condition is true or not.
*
* @var bool
* @access public;
*/
public $conditionIsTrue = false;
/**
* WHERE条件嵌套小括号标记。
* If in mark or not.
*
* @var bool
* @access public;
*/
public $inMark = false;
/**
* 是否开启特殊字符转义。
* Magic quote or not.
*
* @var bool
* @access public
*/
public $magicQuote;
/**
* 构造方法。
* The construct function.
*
* @access public
* @return void
*/
public function __construct($table = '')
{
global $dbh;
$this->dbh = $dbh;
$this->magicQuote = (version_compare(phpversion(), '5.4', '<') and function_exists('get_magic_quotes_gpc') and get_magic_quotes_gpc());
}
/**
* 工厂方法。
* The factory method.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public static function factory($table = '')
{
return new sql($table);
}
/**
* select语句。
* The sql is select.
*
* @param string $field
* @access public
* @return object the sql object.
*/
public static function select($field = '*')
{
$sqlobj = self::factory();
$sqlobj->sql = "SELECT $field ";
return $sqlobj;
}
/**
* update语句。
* The sql is update.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public static function update($table)
{
$sqlobj = self::factory();
$sqlobj->sql = "UPDATE $table SET ";
return $sqlobj;
}
/**
* insert语句。
* The sql is insert.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public static function insert($table)
{
$sqlobj = self::factory();
$sqlobj->sql = "INSERT INTO $table SET ";
return $sqlobj;
}
/**
* replace语句。
* The sql is replace.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public static function replace($table)
{
$sqlobj = self::factory();
$sqlobj->sql = "REPLACE $table SET ";
return $sqlobj;
}
/**
* delete语句。
* The sql is delete.
*
* @access public
* @return object the sql object.
*/
public static function delete()
{
$sqlobj = self::factory();
$sqlobj->sql = "DELETE ";
return $sqlobj;
}
/**
* 将关联数组转换为sql语句中 `key` = value 的形式。
* Join the data items by key = value.
*
* @param object $data
* @param string $skipFields the fields to skip.
* @access public
* @return object the sql object.
*/
public function data($data, $skipFields = '')
{
$data = (object) $data;
if($skipFields) $skipFields = ',' . str_replace(' ', '', $skipFields) . ',';
foreach($data as $field => $value)
{
if(!preg_match('|^\w+$|', $field))
{
unset($data->$field);
continue;
}
if(strpos($skipFields, ",$field,") !== false) continue;
$this->sql .= "`$field` = " . $this->quote($value) . ',';
}
$this->data = $data;
$this->sql = rtrim($this->sql, ','); // Remove the last ','.
return $this;
}
/**
* 在左边添加'('。
* Add an '(' at left.
*
* @param int $count
* @access public
* @return object the sql object.
*/
public function markLeft($count = 1)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= str_repeat('(', $count);
$this->inMark = true;
return $this;
}
/**
* 在右边增加')'。
* Add an ')' at right.
*
* @param int $count
* @access public
* @return object the sql object.
*/
public function markRight($count = 1)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= str_repeat(')', $count);
$this->inMark = false;
return $this;
}
/**
* SET部分。
* The set part.
*
* @param string $set
* @access public
* @return object the sql object.
*/
public function set($set)
{
/* Add ` to avoid keywords of mysql. */
if(strpos($set, '=') ===false)
{
$set = str_replace(',', '', $set);
$set = '`' . str_replace('`', '', $set) . '`';
}
$this->sql .= $this->isFirstSet ? " $set" : ", $set";
if($this->isFirstSet) $this->isFirstSet = false;
return $this;
}
/**
* 创建From部分。
* Create the from part.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public function from($table)
{
$this->sql .= "FROM $table";
return $this;
}
/**
* 创建Alias部分,Alias转为AS。
* Create the Alias part.
*
* @param string $alias
* @access public
* @return object the sql object.
*/
public function alias($alias)
{
$this->sql .= " AS $alias ";
return $this;
}
/**
* 创建LEFT JOIN部分。
* Create the left join part.
*
* @param string $table
* @access public
* @return object the sql object.
*/
public function leftJoin($table)
{
$this->sql .= " LEFT JOIN $table";
return $this;
}
/**
* 创建ON部分。
* Create the on part.
*
* @param string $condition
* @access public
* @return object the sql object.
*/
public function on($condition)
{
$this->sql .= " ON $condition ";
return $this;
}
/**
* 开始条件判断。
* Begin condition judge.
*
* @param bool $condition
* @access public
* @return object the sql object.
*/
public function beginIF($condition)
{
$this->inCondition = true;
$this->conditionIsTrue = $condition;
return $this;
}
/**
* 结束条件判断。
* End the condition judge.
*
* @access public
* @return object the sql object.
*/
public function fi()
{
$this->inCondition = false;
$this->conditionIsTrue = false;
return $this;
}
/**
* 创建WHERE部分。
* Create the where part.
*
* @param string $arg1 the field name
* @param string $arg2 the operator
* @param string $arg3 the value
* @access public
* @return object the sql object.
*/
public function where($arg1, $arg2 = null, $arg3 = null)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
if($arg3 !== null)
{
$value = $this->quote($arg3);
$condition = "`$arg1` $arg2 " . $this->quote($arg3);
}
else
{
$condition = $arg1;
}
if(!$this->inMark) $this->sql .= ' ' . DAO::WHERE ." $condition ";
if($this->inMark) $this->sql .= " $condition ";
return $this;
}
/**
* 创建AND部分。
* Create the AND part.
*
* @param string $condition
* @access public
* @return object the sql object.
*/
public function andWhere($condition, $addMark = false)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$mark = $addMark ? '(' : '';
$this->sql .= " AND {$mark} $condition ";
return $this;
}
/**
* 创建OR部分。
* Create the OR part.
*
* @param bool $condition
* @access public
* @return object the sql object.
*/
public function orWhere($condition)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " OR $condition ";
return $this;
}
/**
* 创建'='部分。
* Create the '='.
*
* @param string $value
* @access public
* @return object the sql object.
*/
public function eq($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " = " . $this->quote($value);
return $this;
}
/**
* 创建'!='。
* Create '!='.
*
* @param string $value
* @access public
* @return void the sql object.
*/
public function ne($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " != " . $this->quote($value);
return $this;
}
/**
* 创建'>'。
* Create '>'.
*
* @param string $value
* @access public
* @return object the sql object.
*/
public function gt($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " > " . $this->quote($value);
return $this;
}
/**
* 创建'>='
* Create '>='.
*
* @param string $value
* @access public
* @return object the sql object.
*/
public function ge($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " >= " . $this->quote($value);
return $this;
}
/**
* 创建'<'。
* Create '<'.
*
* @param mixed $value
* @access public
* @return object the sql object.
*/
public function lt($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " < " . $this->quote($value);
return $this;
}
/**
* 创建 '<='。
* Create '<='.
*
* @param mixed $value
* @access public
* @return object the sql object.
*/
public function le($value)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " <= " . $this->quote($value);
return $this;
}
/**
* 创建"between and"。
* Create "between and"
*
* @param string $min
* @param string $max
* @access public
* @return object the sql object.
*/
public function between($min, $max)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$min = $this->quote($min);
$max = $this->quote($max);
$this->sql .= " BETWEEN $min AND $max ";
return $this;
}
/**
* 创建IN部分。
* Create in part.
*
* @param string|array $ids ','分割的字符串或者数组 list string by ',' or an array
* @access public
* @return object the sql object.
*/
public function in($ids)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= helper::dbIN($ids);
return $this;
}
/**
* 创建'NOT IN'部分。
* Create not in part.
*
* @param string|array $ids list string by ',' or an array
* @access public
* @return object the sql object.
*/
public function notin($ids)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= ' NOT ' . helper::dbIN($ids);
return $this;
}
/**
* 创建LIKE部分。
* Create the like by part.
*
* @param string $string
* @access public
* @return object the sql object.
*/
public function like($string)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= " LIKE " . $this->quote($string);
return $this;
}
/**
* 创建NOT LIKE部分。
* Create the not like by part.
*
* @param string $string
* @access public
* @return object the sql object.
*/
public function notLike($string)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= "NOT LIKE " . $this->quote($string);
return $this;
}
/**
* 创建ORDER BY部分。
* Create the order by part.
*
* @param string $order
* @access public
* @return object the sql object.
*/
public function orderBy($order)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$order = str_replace(array('|', '', '_'), ' ', $order);
/* Add "`" in order string. */
/* When order has limit string. */
$pos = stripos($order, 'limit');
$orders = $pos ? substr($order, 0, $pos) : $order;
$limit = $pos ? substr($order, $pos) : '';
if(!empty($limit))
{
$trimedLimit = trim(str_replace('limit', '', $limit));
if(!preg_match('/^[0-9]+ *(, *[0-9]+)?$/', $trimedLimit)) die("Limit is bad query, The limit is " . htmlspecialchars($limit));
}
$orders = trim($orders);
if(empty($orders)) return $this;
if(!preg_match('/^(\w+\.)?(`\w+`|\w+)( +(desc|asc))?( *(, *(\w+\.)?(`\w+`|\w+)( +(desc|asc))?)?)*$/i', $orders)) die("Order is bad request, The order is " . htmlspecialchars($orders));
$orders = explode(',', $orders);
foreach($orders as $i => $order)
{
$orderParse = explode(' ', trim($order));
foreach($orderParse as $key => $value)
{
$value = trim($value);
if(empty($value) or strtolower($value) == 'desc' or strtolower($value) == 'asc') continue;
$field = $value;
/* such as t1.id field. */
if(strpos($value, '.') !== false) list($table, $field) = explode('.', $field);
if(strpos($field, '`') === false) $field = "`$field`";
$orderParse[$key] = isset($table) ? $table . '.' . $field : $field;
unset($table);
}
$orders[$i] = join(' ', $orderParse);
if(empty($orders[$i])) unset($orders[$i]);
}
$order = join(',', $orders) . ' ' . $limit;
$this->sql .= ' ' . DAO::ORDERBY . " $order";
return $this;
}
/**
* 创建LIMIT部分。
* Create the limit part.
*
* @param string $limit
* @access public
* @return object the sql object.
*/
public function limit($limit)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
if(empty($limit)) return $this;
/* filter limit. */
$limit = trim(str_ireplace('limit', '', $limit));
if(!preg_match('/^[0-9]+ *(, *[0-9]+)?$/', $limit))
{
$limit = htmlspecialchars($limit);
die("Limit is bad query, The limit is $limit");
}
$this->sql .= ' ' . DAO::LIMIT . " $limit ";
return $this;
}
/**
* 创建GROUP BY部分。
* Create the groupby part.
*
* @param string $groupBy
* @access public
* @return object the sql object.
*/
public function groupBy($groupBy)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
if(!preg_match('/^\w+[a-zA-Z0-9_`.]+$/', $groupBy))
{
$groupBy = htmlspecialchars($groupBy);
die("Group is bad query, The group is $groupBy");
}
$this->sql .= ' ' . DAO::GROUPBY . " $groupBy";
return $this;
}
/**
* 创建HAVING部分。
* Create the having part.
*
* @param string $having
* @access public
* @return object the sql object.
*/
public function having($having)
{
if($this->inCondition and !$this->conditionIsTrue) return $this;
$this->sql .= ' ' . DAO::HAVING . " $having";
return $this;
}
/**
* 获取SQL字符串。
* Get the sql string.
*
* @access public
* @return string
*/
public function get()
{
return $this->sql;
}
/**
* 对字段加转义。
* Quote a var.
*
* @param mixed $value
* @access public
* @return mixed
*/
public function quote($value)
{
if($this->magicQuote) $value = stripslashes($value);
return $this->dbh->quote((string)$value);
}
}