dbhost, $globals->dbuser, $globals->dbpwd, $globals->dbdb); if ($globals->debug & 1) { $bt = new PlBacktrace('MySQL'); if (mysqli_connect_errno()) { $bt->newEvent("MySQLI connection", 0, mysqli_connect_error()); return false; } } XDB::$mysqli->autocommit(true); XDB::$mysqli->set_charset($globals->dbcharset); return true; } public static function _prepare($args) { $query = array_map(Array('XDB', '_db_escape'), $args); $query[0] = str_replace('{?}', '%s', str_replace('%', '%%', $args[0])); return call_user_func_array('sprintf', $query); } public static function _reformatQuery($query) { $query = preg_split("/\n\\s*/", trim($query)); $length = 0; foreach ($query as $key=>$line) { $local = -2; if (preg_match('/^([A-Z]+(?:\s+(?:JOIN|BY|FROM|INTO))?)\s+(.*)/u', $line, $matches) && $matches[1] != 'AND' && $matches[1] != 'OR') { $local = strlen($matches[1]); $line = $matches[1] . ' ' . $matches[2]; $length = max($length, $local); } $query[$key] = array($line, $local); } $res = ''; foreach ($query as $array) { list($line, $local) = $array; $local = max(0, $length - $local); $res .= str_repeat(' ', $local) . $line . "\n"; $length += 2 * (substr_count($line, '(') - substr_count($line, ')')); } return $res; } public static function _query($query) { global $globals; if (!XDB::$mysqli && !XDB::connect()) { return false; } if ($globals->debug & 1) { $explain = array(); if (strpos($query, 'FOUND_ROWS()') === false) { $res = XDB::$mysqli->query("EXPLAIN $query"); if ($res) { while ($row = $res->fetch_assoc()) { $explain[] = $row; } $res->free(); } } PlBacktrace::$bt['MySQL']->start(XDB::_reformatQuery($query)); } $res = XDB::$mysqli->query($query); if ($globals->debug & 1) { PlBacktrace::$bt['MySQL']->stop(@$res->num_rows ? $res->num_rows : XDB::$mysqli->affected_rows, XDB::$mysqli->error, $explain); } return $res; } public static function query() { return new XOrgDBResult(XDB::_prepare(func_get_args())); } public static function execute() { return XDB::_query(XDB::_prepare(func_get_args())); } public static function iterator() { return new XOrgDBIterator(XDB::_prepare(func_get_args())); } public static function iterRow() { return new XOrgDBIterator(XDB::_prepare(func_get_args()), MYSQL_NUM); } public static function insertId() { return XDB::$mysqli->insert_id; } public static function errno() { return XDB::$mysqli->errno; } public static function error() { return XDB::$mysqli->error; } public static function affectedRows() { return XDB::$mysqli->affected_rows; } public static function _db_escape($var) { switch (gettype($var)) { case 'boolean': return $var ? 1 : 0; case 'integer': case 'double': case 'float': return $var; case 'string': return "'".addslashes($var)."'"; case 'NULL': return 'NULL'; case 'object': case 'array': return "'".addslashes(serialize($var))."'"; default: die(var_export($var, true).' is not a valid for a database entry'); } } } class XOrgDBResult { private $_res; function XOrgDBResult($query) { $this->_res = XDB::_query($query); } function free() { $this->_res->free(); unset($this); } function _fetchRow() { return $this->_res->fetch_row(); } function _fetchAssoc() { return $this->_res->fetch_assoc(); } function fetchAllRow() { $result = Array(); while ($result[] = $this->_res->fetch_row()); array_pop($result); $this->free(); return $result; } function fetchAllAssoc() { $result = Array(); while ($result[] = $this->_res->fetch_assoc()); array_pop($result); $this->free(); return $result; } function fetchOneAssoc() { $tmp = $this->_fetchAssoc(); $this->free(); return $tmp; } function fetchOneRow() { $tmp = $this->_fetchRow(); $this->free(); return $tmp; } function fetchOneCell() { $tmp = $this->_fetchRow(); $this->free(); return $tmp[0]; } function fetchColumn($key = 0) { $res = Array(); if (is_numeric($key)) { while($tmp = $this->_fetchRow()) { $res[] = $tmp[$key]; } } else { while($tmp = $this->_fetchAssoc()) { $res[] = $tmp[$key]; } } $this->free(); return $res; } function fetchOneField() { return $this->_res->fetch_field(); } function fetchFields() { $res = array(); while ($res[] = $this->fetchOneField()); return $res; } function numRows() { return $this->_res->num_rows; } function fieldCount() { return $this->_res->field_count; } } require_once dirname(__FILE__) . '/pliterator.php'; class XOrgDBIterator implements PlIterator { private $_result; private $_pos; private $_total; private $_fpos; private $_fields; private $_mode = MYSQL_ASSOC; function __construct($query, $mode = MYSQL_ASSOC) { $this->_result = new XOrgDBResult($query); $this->_pos = 0; $this->_total = $this->_result->numRows(); $this->_fpost = 0; $this->_fields = $this->_result->fieldCount(); $this->_mode = $mode; } function next() { $this->_pos ++; if ($this->_pos > $this->_total) { $this->_result->free(); unset($this); return null; } return $this->_mode != MYSQL_ASSOC ? $this->_result->_fetchRow() : $this->_result->_fetchAssoc(); } function first() { return $this->_pos == 1; } function last() { return $this->_pos == $this->_total; } function total() { return $this->_total; } function nextField() { $this->_fpos++; if ($this->_fpos > $this->_fields) { return null; } return $this->_result->fetchOneField(); } function firstField() { return $this->_fpos == 1; } function lastField() { return $this->_fpos == $this->_fields; } function totalFields() { return $this->_fields; } } // vim:set et sw=4 sts=4 sws=4 foldmethod=marker enc=utf-8: ?>