Файл: oc-includes/osclass/classes/database/DBCommandClass.php
Строк: 1039
<?php
/*
 * Copyright 2014 Osclass
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
    /**
     * Database command object
     *
     * @package Osclass
     * @subpackage Database
     * @since 2.3
     */
    class DBCommandClass
    {
        /**
         * Database connection object to Osclass database
         *
         * @access private
         * @since 2.3
         * @var mysqli
         */
        var $connId;
        /**
         * Database result object
         *
         * @access public
         * @since 2.3
         * @var MySQLi_Result
         */
        var $resultId;
        /**
         *
         * @var array
         */
        var $queries;
        /**
         *
         * @var array
         */
        var $queryTimes;
        /**
         *
         * @var int
         */
        var $queryCount;
        /**
         *
         * @var int
         */
        var $errorLevel;
        /**
         *
         * @var string
         */
        var $errorDesc;
        /**
         *
         * @var array
         */
        var $aSelect;
        /* var $aDistinct; */
        /**
         *
         * @var array
         */
        var $aFrom;
        /**
         *
         * @var array
         */
        var $aJoin;
        /**
         *
         * @var array
         */
        var $aWhere;
        /**
         *
         * @var array
         */
        var $aLike;
        /**
         *
         * @var array
         */
        var $aGroupby;
        /**
         *
         * @var array
         */
        var $aHaving;
        /* var $aKeys; */
        /**
         *
         * @var mixed
         */
        var $aLimit;
        /**
         *
         * @var mixed
         */
        var $aOffset;
        /**
         *
         * @var mixed
         */
        var $aOrder;
        /**
         *
         * @var array
         */
        var $aOrderby;
        /**
         *
         * @var array
         */
        var $aSet;
        /**
         *
         * @var array
         */
        var $aWherein;
        /* var $aAliasedTables; */
        /* var $aStoreArray; */
        /**
         *
         * @var LogDatabase
         */
        var $log;
        /**
         * Initializate variables
         *
         * @param mysqli $connId
         */
        function __construct(&$connId)
        {
            $this->connId     = &$connId;
            $this->resultId   = 0;
            $this->queries    = array();
            $this->queryTimes = array();
            $this->queryCount = 0;
            $this->errorLevel = 0;
            $this->errorDesc  = "";
            $this->aSelect    = array();
            $this->aFrom      = array();
            $this->aJoin      = array();
            $this->aWhere     = array();
            $this->aLike      = array();
            $this->aGroupby   = array();
            $this->aHaving    = array();
            $this->aLimit     = false;
            $this->aOffset    = false;
            $this->aOrder     = false;
            $this->aOrderby   = array();
            $this->aWherein   = array();
            if ( OSC_DEBUG_DB || OSC_DEBUG_DB_EXPLAIN ) $this->log = LogDatabase::newInstance();
        }
        /**
         * Unset connection and result objects
         */
        function __destruct()
        {
            unset($this->connId);
            unset($this->resultId);
        }
        /**
         * It creates a new DBCommandClass object or if it has been created before, it
         * returns the previous object
         *
         * @access public
         * @since 2.3
         * @return DBCommandClass
         */
        public static function newInstance()
        {
            if(!self::$instance instanceof self) {
                self::$instance = new self;
            }
            return self::$instance;
        }
        /**
         * Set SELECT clause
         *
         * @access public
         * @since 2.3
         * @param mixed $select It can be a string or array
         * @return DBCommandClass
         */
        function select($select = '*')
        {
            if( is_string($select) ) {
                $select = explode(',', $select);
            }
            foreach($select as $s) {
                $s = trim($s);
                if($s != '') {
                    $this->aSelect[] = $s;
                }
            }
            return $this;
        }
        /**
         * Set FROM clause
         *
         * @param mixed $from It can be a string or array
         * @return DBCommandClass
         */
        function from($from)
        {
            if( !is_array($from) ) {
                if( strpos($from, ',') !== false) {
                    $from = explode(',', $from);
                } else {
                    $from = array($from);
                }
            }
            foreach($from as $f) {
                $this->aFrom[] = $f;
            }
            return $this;
        }
        /**
         * Set JOIN clause
         *
         * @access public
         * @since 2.3
         * @param string $table
         * @param string $cond
         * @param string $type It can be: LEFT, RIGHT, OUTER, INNER, LEFT OUTER or RIGHT OUTER
         * @return DBCommandClass
         */
        function join($table, $cond, $type = '')
        {
            if($type != '') {
                $type = strtoupper(trim($type));
                if( !in_array($type, array('LEFT', 'RIGHT', 'OUTER', 'INNER', 'LEFT OUTER', 'RIGHT OUTER')) ) {
                    $type = '';
                } else {
                    $type .= ' ';
                }
            }
            $join = $type . ' JOIN ' . $table . ' ON ' . $cond;
            $this->aJoin[] = $join;
            return $this;
        }
        /**
         * Set WHERE clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $value
         * @return DBCommandClass
         */
        function where($key, $value = null)
        {
            return $this->_where($key, $value, 'AND ');
        }
        /**
         * Set WHERE clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $value
         * @return DBCommandClass
         */
        function orWhere($key, $value = null)
        {
            return $this->_where($key, $value, 'OR ');
        }
        /**
         * Set WHERE clause
         *
         * @access private
         * @since 2.3
         * @param mixed $key
         * @param mixed $value
         * @param string $type
         * @return DBCommandClass
         */
        function _where($key, $value = null, $type = 'AND ')
        {
            if( !is_array($key) ) {
                $key = array($key => $value);
            }
            foreach($key as $k => $v) {
                $prefix = (count($this->aWhere) > 0) ? $type : '';
                if( !$this->_hasOperator($k) ) {
                    $k .= ' =';
                }
                if(!is_null($v)) {
                    $v = ' ' . $this->escape($v);
                }
                $this->aWhere[] = $prefix . $k . $v;
            }
            return $this;
        }
        /**
         * Set WHERE IN clause using AND operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $values
         * @return DBCommandClass
         */
        function whereIn($key = null, $values = null)
        {
            return $this->_whereIn($key, $values, false, 'AND ');
        }
        /**
         * Set WHERE IN clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $values
         * @return DBCommandClass
         */
        function orWhereIn($key = null, $values = null)
        {
            return $this->_whereIn($key, $values, false, 'OR ');
        }
        /**
         * Set WHERE NOT IN clause using AND operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $values
         * @return DBCommandClass
         */
        function whereNotIn($key = null, $values = null)
        {
            return $this->_whereIn($key, $values, true, 'AND ');
        }
        /**
         * Set WHERE NOT IN clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $values
         * @return DBCommandClass
         */
        function orWhereNotIn($key = null, $values = null)
        {
            return $this->_whereIn($key, $values, true, 'OR ');
        }
        /**
         * Set WHERE IN clause
         *
         * @access private
         * @since 2.3
         * @param mixed $key
         * @param mixed $values
         * @param bool $not
         * @param string $type
         * @return DBCommandClass
         */
        function _whereIn($key = null, $values = null, $not = false, $type = 'AND ')
        {
            if( !is_array($values) ) {
                $values = array($values);
            }
            $not = ($not) ? ' NOT' : '';
            foreach($values as $value) {
                $this->aWherein[] = $this->escape($value);
            }
            $prefix  = (count($this->aWhere) > 0) ? $type : '';
            $whereIn = $prefix . $key . $not . ' IN (' . implode(', ', $this->aWherein) . ') ';
            $this->aWhere[] = $whereIn;
            $this->aWherein = array();
            return $this;
        }
        /**
         * Set LIKE clause
         *
         * @access public
         * @param type $field
         * @param type $match
         * @param type $side
         * @return DBCommandClass
         */
        function like($field, $match = '', $side = 'both')
        {
            return $this->_like($field, $match, 'AND ', $side);
        }
        /**
         * Set NOT LIKE clause using AND operator
         *
         * @access public
         * @since 2.3
         * @param string $field
         * @param string $match
         * @param string $side
         * @return DBCommandClass
         */
        function notLike($field, $match = '', $side = 'both')
        {
            return $this->_like($field, $match, 'AND ', $side, 'NOT');
        }
        /**
         * Set LIKE clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param string $field
         * @param string $match
         * @param type $side
         * @return string
         */
        function orLike($field, $match = '', $side = 'both')
        {
            return $this->_like($field, $match, 'OR ', $side);
        }
        /**
         * Set NOT LIKE clause using OR operator
         *
         * @access public
         * @since 2.3
         * @param string $field
         * @param string $match
         * @param string $side
         * @return DBCommandClass
         */
        function orNotLike($field, $match = '', $side = 'both')
        {
            return $this->_like($field, $match, 'OR ', $side, 'NOT');
        }
        /**
         * Set LIKE clause
         *
         * @access private
         * @since 2.3
         * @param string $field
         * @param string $match
         * @param string $type Types: AND, OR
         * @param string $side Options: before, after, both
         * @param string $not Two possibilities: blank or NOT
         * @return DBCommandClass
         */
        function _like($field, $match = '', $type = 'AND ', $side = 'both', $not = '')
        {
            $likeStatement = '';
            if( !is_array($field) ) {
                $field = array($field => $match);
            }
            foreach($field as $k => $v) {
                $prefix = (count($this->aLike) == 0) ? '' : $type;
                $v      = $this->escapeStr($v, true);
                switch ($side) {
                    case 'before':  $likeStatement = "$prefix $k $not LIKE '%$v'";
                    break;
                    case 'after':   $likeStatement = "$prefix $k $not LIKE '$v%'";
                    break;
                    default:        $likeStatement = "$prefix $k $not LIKE '%$v%'";
                    break;
                }
                $this->aLike[] = $likeStatement;
            }
            return $this;
        }
        /**
         * Fields for GROUP BY clause
         *
         * @access public
         * @since 2.3
         * @param mixed $by
         * @return DBCommandClass
         */
        function groupBy($by)
        {
            if( is_string($by) ) {
                $by = explode(',', $by);
            }
            foreach($by as $val) {
                $val = trim($val);
                if( $val != '' ) {
                    $this->aGroupby[] = $val;
                }
            }
            return $this;
        }
        /**
         *
         * @param type $key
         * @param type $value
         * @return type
         */
        function having($key, $value = '')
        {
            return $this->_having($key, $value, 'AND ');
        }
        /**
         *
         * @param type $key
         * @param type $value
         * @return type
         */
        function orHaving($key, $value = '')
        {
            return $this->_having($key, $value, 'OR ');
        }
        /**
         *
         * @param type $key
         * @param type $value
         * @param type $type
         */
        function _having($key, $value = '', $type = 'AND ')
        {
            if( !is_array($key) ) {
                $key = array($key => $value);
            }
            foreach($key as $k => $v) {
                $prefix = (count($this->aHaving) == 0) ? '' : $type;
                if( !$this->_hasOperator($k) ) {
                    $k .= ' = ';
                }
                $v = ' ' . $this->escapeStr($v);
                $this->aHaving[] = $prefix . $k . $v;
            }
        }
        /**
         * Set ORDER BY clause
         *
         * @access public
         * @since 2.3
         * @param string $orderby
         * @param string $direction Accepted directions: random, asc, desc
         */
        function orderBy($orderby, $direction = '')
        {
            if(strtolower($direction) == 'random') {
                $direction = ' RAND()';
            } elseif( trim($direction) != '' ) {
                $direction = (in_array(strtoupper(trim($direction)), array('ASC', 'DESC'))) ? ' ' . $direction : ' ASC';
            }
            $this->aOrderby[] = $orderby . $direction;
            return $this;
        }
        /**
         * Set LIMIT clause
         *
         * @access public
         * @since 2.3
         * @param int $value
         * @param int $offset
         * @return DBCommandClass
         */
        function limit($value, $offset = '')
        {
            if(is_numeric($value)) {
                $this->aLimit = intval($value);
            }
            if( $offset != '' ) {
                if(is_numeric($offset)) {
                    $this->aOffset = intval($offset);
                } else {
                    $this->aOffset = 0;
                }
            }
            return $this;
        }
        /**
         * Set the offset in the LIMIT clause
         *
         * @access public
         * @since 2.3
         * @param int $offset
         * @return DBCommandClass
         */
        function offset($offset)
        {
            if(is_numeric($offset)) {
                $this->aOffset = intval($offset);
            } else {
                $this->aOffset = 0;
            }
            return $this;
        }
        /**
         * Create the INSERT sql and perform the query
         *
         * @access public
         * @since 2.3
         * @param mixed $table
         * @param mixed $set
         * @return boolean
         */
        function insert($table = '', $set = null)
        {
            if( !is_null($set) ) {
                $this->set($set);
            }
            if( count($this->aSet) == 0 ) {
                return false;
            }
            if( $table == '') {
                if( !isset($this->aFrom[0]) ) {
                    return false;
                }
                $table = $this->aFrom[0];
            }
            $sql = $this->_insert($table, array_keys($this->aSet), array_values($this->aSet));
            $this->_resetWrite();
            return $this->query($sql);
        }
        /**
         * Create the INSERT sql string
         *
         * @access private
         * @since 2.3
         * @param string $table
         * @param array $keys
         * @param array $values
         * @return string
         */
        function _insert($table, $keys, $values)
        {
            return 'INSERT INTO ' . $table . ' (' . implode(', ', $keys). ') VALUES (' . implode(', ', $values) . ')';
        }
        /**
         * Create the REPLACE INTO sql and perform the query
         *
         * @access public
         * @since 2.3
         * @param mixed $table
         * @param mixed $set
         * @return boolean
         */
        function replace($table = '', $set = null)
        {
            if( !is_null($set) ) {
                $this->set($set);
            }
            if( count($this->aSet) == 0 ) {
                return false;
            }
            if( $table == '') {
                if( !isset($this->aFrom[0]) ) {
                    return false;
                }
                $table = $this->aFrom[0];
            }
            $sql = $this->_replace($table, array_keys($this->aSet), array_values($this->aSet));
            $this->_resetWrite();
            return $this->query($sql);
        }
        /**
         * Create the REPLACE INTO sql string
         *
         * @access private
         * @since 2.3
         * @param string $table
         * @param array $key
         * @param array $values
         * @return string
         */
        function _replace($table, $keys, $values)
        {
            return 'REPLACE INTO ' . $table . ' (' . implode(', ', $keys) . ') VALUES (' . implode(', ', $values) . ')';
        }
        /**
         * Create the UPDATE sql and perform the query
         *
         * @access public
         * @since 2.3
         * @param mixed $table
         * @param mixed $set
         * @param mixed $where
         * @return mixed
         */
        function update($table = '', $set = null, $where = null)
        {
            if( !is_null($set) ) {
                $this->set($set);
            }
            if( count($this->aSet) == 0 ) {
                return false;
            }
            if( $table == '') {
                if( !isset($this->aFrom[0]) ) {
                    return false;
                }
                $table = $this->aFrom[0];
            }
            if( $where != null ) {
                $this->where($where);
            }
            $sql = $this->_update($table, $this->aSet, $this->aWhere);
            $this->_resetWrite();
            $result = $this->query($sql);
            if( $result == false ) {
                return false;
            }
            return $this->affectedRows();
        }
        /**
         * Create the UPDATE sql string
         *
         * @access private
         * @since 2.3
         * @param string $table
         * @param array $values
         * @param array $where
         * @return string
         */
        function _update($table, $values, $where)
        {
            foreach($values as $k => $v) {
                $valstr[] = $k . ' = ' . $v;
            }
            $sql = 'UPDATE ' . $table . ' SET ' . implode(', ', $valstr);
            $sql .= ($where != '' && count($where) > 0) ? " WHERE " . implode(" ", $where) : '';
            return $sql;
        }
        /**
         * Create the DELETE sql and perform the query
         *
         * @access public
         * @since 2.3
         * @param mixed $table
         * @param mixed $where
         * @return mixed
         */
        function delete($table = '', $where = '')
        {
            if( $table == '') {
                if( !isset($this->aFrom[0]) ) {
                    return false;
                }
                $table = $this->aFrom[0];
            }
            if( $where != null ) {
                $this->where($where);
            }
            if( count($this->aWhere) == 0 && count($this->aWherein) == 0 && count($this->aLike) == 0 ) {
                return false;
            }
            $sql = $this->_delete($table, $this->aWhere, $this->aLike);
            $this->_resetWrite();
            $result = $this->query($sql);
            if( $result == false ) {
                return false;
            }
            return $this->affectedRows();
        }
        /**
         * Create the DELETE sql string
         *
         * @access private
         * @since 2.3
         * @param string $table
         * @param array $where
         * @param array $like
         * @return string
         */
        function _delete($table, $where, $like)
        {
            $conditions = '';
            if( count($where) > 0 || count($like) > 0 ) {
                $conditions  = "nWHERE ";
                $conditions .= implode("n", $where);
                if( count($where) > 0 && count($like) > 0 ) {
                    $conditions .= ' AND ';
                }
                $conditions .= implode("n", $like);
            }
            $sql = 'DELETE FROM ' . $table . $conditions;
            return $sql;
        }
        /**
         * Compile the select sql string and perform the query. Quick method for
         * getting the rows of one table
         *
         * @access public
         * @since 2.3
         * @param mixed $table
         * @param mixed $limit
         * @param mixed $offset
         * @return mixed
         */
        function get($table = '', $limit = null, $offset = null)
        {
            if($table != '') {
                $this->from($table);
            }
            if( !is_null($limit) ) {
                $this->limit($limit, $offset);
            }
            $sql = $this->_getSelect();
            $result = $this->query($sql);
            $this->_resetSelect();
            return $result;
        }
        /**
         * Performs a query on the database
         *
         * @access public
         * @since 2.3
         * @param string $sql
         * @return mixed
         */
        function query($sql)
        {
            if($sql == '') {
                return false;
            }
            if( OSC_DEBUG_DB_EXPLAIN && $this->isSelectType($sql) ) {
                $this->query_debug($sql);
            }
            $this->queries[] = $sql;
            $timeStart = list($sm, $ss) = explode(' ', microtime());
            $this->resultId = $this->_execute($sql);
            $this->errorReport();
            if( false === $this->resultId ) {
                if( OSC_DEBUG_DB ) {
                    $this->log->addMessage($sql, 0, $this->errorLevel, $this->errorDesc);
                }
                return false;
            }
            $timeEnd = list($em, $es) = explode(' ', microtime());
            $this->queryTimes[] = ($em + $es) - ($sm + $ss);
            $this->queryCount++;
            if( OSC_DEBUG_DB ) {
                $this->log->addMessage($sql, ($em + $es) - ($sm + $ss), $this->errorLevel, $this->errorDesc);
            }
            if( $this->isWriteType($sql) === true ) {
                return true;
            }
            $rs           = new DBRecordsetClass();
            $rs->connId   = $this->connId;
            $rs->resultId = $this->resultId;
            $rs->numRows  = $rs->numRows();
            return $rs;
        }
        function query_debug($sql)
        {
            if($sql == '') {
                return false;
            }
            $sql  = 'EXPLAIN ' . $sql;
            $rsID = $this->_execute($sql);
            if( false === $rsID ) {
                return false;
            }
            $rs           = new DBRecordsetClass();
            $rs->connId   = $this->connId;
            $rs->resultId = $rsID;
            $rs->numRows  = $rs->numRows();
            if( $rs->numRows() == 0 ) {
                return false;
            }
            $this->log->addExplainMessage($sql, $rs->result());
            return true;
        }
        /**
         * Performs a query on the database
         *
         * @access private
         * @since 2.3
         * @param string $sql
         * @return mixed
         */
        function _execute($sql)
        {
            return $this->connId->query($sql);
        }
        /**
         * Execute queries sql. We replace TABLE_PREFIX for the real prefix: DB_TABLE_PREFIX
         * The executions is stopped if some query throws an error.
         *
         * @access public
         * @since 2.3
         * @param string $sql
         * @return boolean true if it's succesful, false if not
         */
        function importSQL($sql)
        {
            $sql     = str_replace( '/*TABLE_PREFIX*/', DB_TABLE_PREFIX, $sql);
            $sql     = preg_replace('#/*(?:[^*]*(?:*(?!/))*)**/#','',($sql));
            $queries = $this->splitSQL($sql, ';');
            if( count($queries) == 0 ) {
                return false;
            }
            foreach($queries as $q) {
                $q = trim($q);
                if( !empty($q) ) {
                    if( !$this->query($q) ) {
                        return false;
                    }
                }
            }
            return true;
        }
        /**
         * Split sql queries, allowing DELIMITER blocks. We clean DELIMITER statements.
         *
         * @param string $sql
         * @param string $explodeChars
         * @return array
         */
        private function splitSQL($sql, $explodeChars)
        {
            if (preg_match('|^(.*)DELIMITER (S+)s(.*)$|isU', $sql, $matches)) {
                $queries = explode($explodeChars, $matches[1]);
                $recursive = $this->splitSQL($matches[3], $matches[2]);
                return array_merge($queries, $recursive);
            }
            else {
                return explode($explodeChars, $sql);
            }
        }
        /**
         * Prepare and separe the queries, and save into data or struct queries
         *
         * @param array $queries
         * @param array $data_queries
         * @param array $struct_queries
         */
        private function prepareAndSepareQueries($queries, &$data_queries, &$struct_queries)
        {
            foreach($queries as $query) {
                if(preg_match('|CREATE DATABASE ([^ ]*)|', $query, $match)) {
                    array_unshift($struct_queries, $query);
                } else if(preg_match('|CREATE TABLE ([^ ]*)|', $query, $match)) {
                    $struct_queries[trim(strtolower($match[1]), '`')] = $query;
                } else if(preg_match('|INSERT INTO ([^ ]*)|', $query, $match)) {
                    $data_queries[] = $query;
                } else if(preg_match('|UPDATE ([^ ]*)|', $query, $match)) {
                    $data_queries[] = $query;
                }
            }
        }
        /**
         * Check if $table exist into array $struct_queries
         *
         * @param string $table
         * @param array $struct_queries
         */
        private function existTableIntoStruct($table, $struct_queries)
        {
            return array_key_exists(strtolower($table), $struct_queries);
        }
        /**
         * Get fields from struct_queries (struct.sql)
         *
         * @param string $table
         * @param array $struct_queries
         */
        private function getTableFieldsFromStruct($table, &$struct_queries)
        {
            if(preg_match('|((.*))|ms', $struct_queries[strtolower($table)], $match)) {
                $fields = explode("n", trim($match[1]));
                foreach($fields as $key => $value) {
                    $fields[$key] = trim(preg_replace('/,$/','', $value));
                }
            } else {
                $fields = false;
            }
            return $fields;
        }
        /**
         * Classify fields, inside arrays $normal_fields, $indexes, $constrains (foreign key's)
         *
         * @param type $fields
         * @param type $normal_fields
         * @param type $indexes
         * @param type $constrains
         * @param type $lastTable
         */
        private function classifyFieldsSql($fields, &$normal_fields, &$indexes, &$constrains, &$lastTable)
        {
            foreach($fields as $field) {
                if(preg_match('|([^ ]+)|', trim($field), $field_name)) {
                    switch (strtolower($field_name[1])) {
                        case '':
                        case 'on':
                            if($lastTable){
                                $constrains[$lastTable] = $constrains[$lastTable].' '.trim($field);
                            }
                            break;
                        case 'foreign':
                            if( preg_match("|FOREIGN KEYs+(.*)s+REFERENCESs+(.*)|mi", $field, $match) ) {
                                $_table = $match[1];
                                $refere = $match[2];
                                $refere = str_replace(',', '', $refere);
                                $lastTable = $_table;
                                $constrains[$_table] = trim($refere);
                            }
                            break;
                        case 'primary':
                        case 'index':
                        case 'fulltext':
                        case 'unique':
                        case 'key':
                            $added = false;
                            if( preg_match("|PRIMARY KEYs+((.*))|mi", $field, $match) ) {
                                $_field = strtolower($match[1]);
                                if(isset($normal_fields[$_field])) {
                                    $normal_fields[ $_field ] = $normal_fields[ $_field ] . ' PRIMARY KEY';
                                    $added = true;
                                }
                            }
                            if(!$added) {
                                $indexes[] = trim($field, ", n");
                            }
                            break;
                        default :
                            $normal_fields[strtolower($field_name[1])] = trim($field, ", n");
                            break;
                    }
                }
            }
        }
        /**
         * Build alter sql, ADD COLUMN, CHANGE COLUMN, ALTER COLUMN
         *
         * @param array $tbl_fields, contain all fields inside database
         * @param type $table
         * @param type $normal_fields
         * @param type $struct_queries
         */
        private function createAlterTable($tbl_fields, $table, &$normal_fields, &$struct_queries )
        {
            foreach($tbl_fields as $tbl_field) {
                //Every field should we on the definition, so else SHOULD never happen, unless a very aggressive plugin modify our tables
                if(array_key_exists(strtolower($tbl_field['Field']), $normal_fields)) {
                    // Take the type of the field
                    if(preg_match("|".$tbl_field['Field']." (ENUMs*(([^)]*)))|i", $normal_fields[strtolower($tbl_field['Field'])], $match) || preg_match("|".$tbl_field['Field']." ([^ ]*( unsigned)?)|i", $normal_fields[strtolower($tbl_field['Field'])], $match)) {
                        $field_type = $match[1];
                        // Are they the same?
                        if(strtolower($field_type)!=strtolower($tbl_field['Type']) && str_replace(' ', '', strtolower($field_type))!=str_replace(' ', '', strtolower($tbl_field['Type']))) {
                            $struct_queries[] = "ALTER TABLE ".$table." CHANGE COLUMN ".$tbl_field['Field']." ".$normal_fields[strtolower($tbl_field['Field'])];
                        }
                    }
                    error_log(" --- ".$normal_fields[strtolower($tbl_field['Field'])]);
                    // Have we changed the default value? [with quotes]
                    if(preg_match("| DEFAULTs+'(.*)'|i", $normal_fields[strtolower($tbl_field['Field'])], $default_match)) {
                        // alter column only if default value has been changed
                        if($tbl_field['Default'] != $default_match[1]) {
                            $struct_queries[] = "ALTER TABLE ".$table." ALTER COLUMN ".$tbl_field['Field']." SET DEFAULT '".$default_match[1]."'";
                        }
                    // Have we changed the default value? [without quotes]
                    } else if(preg_match("| DEFAULTs+(.*)|i", $normal_fields[strtolower($tbl_field['Field'])], $default_match)) {
                        if(isset($tbl_field['Default'])) {
                            // alter column only if default value has been changed
                            if( $tbl_field['Default'] != $default_match[1] ) {
                                $struct_queries[] = "ALTER TABLE ".$table." ALTER COLUMN ".$tbl_field['Field']." SET DEFAULT ".$default_match[1];
                            }
                        } else {
                            // check NULL default values
                            // if new default value is diferent, alter column ...
                            if($default_match[1] != 'NULL' ) {
                                $struct_queries[] = "ALTER TABLE ".$table." ALTER COLUMN ".$tbl_field['Field']." SET DEFAULT ".$default_match[1];
                            }
                        }
                    }
                    // Remove it from the list, so it will not be added
                    unset($normal_fields[strtolower($tbl_field['Field'])]);
                }
            }
            // For the rest of normal fields (they are not in the table) we add them.
            foreach($normal_fields as $k => $v) {
                $struct_queries[] = "ALTER TABLE ".$table." ADD COLUMN ".$v;
            }
        }
        /**
         * With all the indexes from struct.sql, remove indexes which actually
         * exist into database
         *
         * @param type $tbl_indexes
         * @param type $indexes
         */
        private function createNewIndex($tbl_indexes, &$indexes, $table, &$struct_queries)
        {
            if($tbl_indexes) {
                unset($indexes_array);
                foreach($tbl_indexes as $tbl_index) {
                    $indexes_array[$tbl_index['Key_name']]['columns'][]  = array('fieldname' => $tbl_index['Column_name'], 'subpart' => $tbl_index['Sub_part']);
                    $indexes_array[$tbl_index['Key_name']]['unique']     = ($tbl_index['Non_unique'] == 0)?true:false;
                    $indexes_array[$tbl_index['Key_name']]['index_type'] = $tbl_index['Index_type'];
                    $indexes_array[$tbl_index['Key_name']]['Key_name']   = $tbl_index['Key_name'];
                }
                foreach($indexes_array as $k => $v) {
                    // if PRIMARY KEY already exist
                    $exist_primary = false;
                    if($k == 'PRIMARY') {
                        if(isset($indexes_array['PRIMARY'])) {
                            if(count($indexes_array['PRIMARY']['columns'])>0) {
                                $exist_primary = true;
                            }
                        }
                    }
                    $string = '';
                    if ($k=='PRIMARY') {
                        $string .= 'PRIMARY KEY ';
                    } else if($v['unique']) {
                        $string .= 'UNIQUE KEY ';
                    } else if($v['index_type'] == 'FULLTEXT') {  // FULLTEXT INDEX MUST HAVE KEY_NAME
                        $string .= 'FULLTEXT '.$k.' ';
                    } else {
                        if( ( count($v['columns']) == 1 && $v['columns'][0]['fieldname'] !=  $k ) || ( preg_match('/^idx/', $k, $coincidencias) > 0 ) ) {
                            $string .= 'INDEX '.$k.' ';
                        } else {
                            $string .= 'INDEX ' . $v['Key_name'] . ' ';
                        }
                    }
                    $columns = '';
                    // For each column in the index
                    foreach ($v['columns'] as $column) {
                        if ($columns != '') $columns .= ', ';
                        // Add the field to the column list string
                        $columns .= "".$column['fieldname']."";
                        if ($column['subpart'] != '') {
                            $columns .= '('.$column['subpart'].')';
                        }
                    }
                    // Add the column list to the index create string
                    $string .= '('.$columns.')';
                    $var_index = array_search($string, $indexes);
                    if (!($var_index===false)) {
                        unset($indexes[$var_index]);
                    } else {
                        $var_index = array_search(str_replace(', ', ',', $string), $indexes);
                        if (!($var_index===false)) {
                            unset($indexes[$var_index]);
                        }
                    }
                }
            }
            // alter table
            foreach($indexes as $v ) {
                if(preg_match('/primary key/i', $v, $coincidencias) > 0 ) {
                    $struct_queries[] = "ALTER TABLE ".$table." DROP PRIMARY KEY, ADD ".$v;
                } else {
                    $struct_queries[] = "ALTER TABLE ".$table." ADD ".$v;
                }
            }
        }
        /**
         * Create alter table if foreign key don't exist into database structure
         *
         * @param array $tbl_constraint
         * @param string $table
         * @param array $struct_queries
         * @param array $constrains
         */
        private function createForeignKey($tbl_constraint, $table, &$struct_queries, $constrains)
        {
            $constrainsDB  = $foreignRepited =  array();
            if(preg_match_all("| CONSTRAINTs+(.*)s+FOREIGN KEYs+(.*)s+REFERENCESs+(.*),?n|i", $tbl_constraint['Create Table'], $default_match)) {
                $aKeyName = $default_match[1];
                $aTables = $default_match[2];
                $aRefere = $default_match[3];
                foreach($aTables as $index => $value) {
                    $_refere = str_replace('`', '', $aRefere[$index]);
                    $_keyName = str_replace('`', '', $aKeyName[$index]);
                    $_refere = str_replace(',', '', $_refere);
                    $_value  = str_replace('`', '', $value);
                    if( in_array($_refere, $constrainsDB) ) {
                        $foreignRepited[] = $_keyName;
                    }
                    $constrainsDB[$_value] = $_refere;
                }
            }
            $delete_foreign = array();
            if( count($foreignRepited) > 0 ) {
                foreach( $foreignRepited as $_key ) {
                    echo "ALTER TABLE ".$table." DROP FOREIGN KEY ".$_key ."<br>";
                    $struct_queries[] = "ALTER TABLE ".$table." DROP FOREIGN KEY ".$_key;
                }
            }
            $keys = array_keys($constrainsDB);
            foreach($constrains as $k => $v ) {
                if( in_array($k, $keys) && $constrainsDB[$k] == $v ) {
                    // nothing to do
                } else {
                    // alter table
                    $index = 'FOREIGN KEY '.$k.' REFERENCES '.$v;
                    $struct_queries[] = "ALTER TABLE ".$table." ADD ".$index;
                }
            }
        }
        /**
         * Given some queries, it will check against the installed database if the information is the same
         *
         * _______pseudocode_______
         *
         *   _separeQueries()
         *   showTables_DataBase()
         *   foreach(table)
         *     if(table exist into struct.sql)
         *       _getTableFields()
         *       _createAlterTables() (save info into array)
         *       _createNewIndex()    (save info into array)
         *       _createForeignKeys() (save info into array)
         *     endif
         *   enforeach
         *
         *   foreach(query created before)
         *     exec(query)
         *   endforeach
         *
         * ______endpseudocode______
         *
         * @param mixed array or string with the SQL queries.
         * @return BOOLEAN true on success, false on fail
        */
        function updateDB($queries = '')
        {
            error_log(' ----- START updateDB ----- ');
            if(!is_array($queries)) {
                $queries = $this->splitSQL($queries, ';');
            }
            // Prepare and separate the queries
            $struct_queries = array();
            $data_queries = array();
            $this->prepareAndSepareQueries($queries, $data_queries, $struct_queries);
            // hack
            $this->query("SET FOREIGN_KEY_CHECKS = 0");
            // Get tables from DB (already installed)
            $result = $this->query('SHOW TABLES');
            $tables = $result->result();
            foreach($tables as $v) {
                $table = current($v);
                if( $this->existTableIntoStruct($table, $struct_queries) ) {
                    $lastTable = NULL;
                    $normal_fields = $indexes = $constrains = array();
                    $fields = $this->getTableFieldsFromStruct($table, $struct_queries);
                    if($fields) {
                        // classify fields (into sql file)
                        $this->classifyFieldsSql($fields, $normal_fields, $indexes, $constrains, $lastTable);
                        // Take fields from the DB (now into database)
                        $result = $this->query('DESCRIBE '.$table);
                        $tbl_fields = $result->result();
                        // compare and create alter statments
                        $this->createAlterTable($tbl_fields, $table, $normal_fields, $struct_queries);
                        // Go for the index part
                        $result = $this->query("SHOW INDEX FROM ".$table);
                        $tbl_indexes = $result->result();
                        // compare table index and struct.sql index for the same table, and only add the new ones
                        $this->createNewIndex($tbl_indexes, $indexes, $table, $struct_queries);
                        // show create table TABLE_NAME constrains
                        $result = $this->query("SHOW CREATE TABLE ".$table);
                        $tbl_constraint = $result->row();
                        // create foreign keys
                        $this->createForeignKey($tbl_constraint, $table, $struct_queries, $constrains);
                        // No need to create the table, so we delete it SQL
                        unset($struct_queries[strtolower($table)]);
//                        error_log(' --- struct_queries ---');
                        foreach($struct_queries as $q) {
//                            error_log(' --- ' . $q );
                        }
                    }
                }
            }
            error_log(' --- last_struct_queries ---');
            foreach($struct_queries as $q) {
                error_log(' --- ' . $q );
            }
            // HACK: AUTO_INCREMENT fields needs to be also a PRIMARY KEY
            foreach($struct_queries as $k => $v) {
                if(stripos($v, "auto_increment")!==FALSE && stripos($v, "primary key")===FALSE) {
                    $struct_queries[$k] = $v." PRIMARY KEY";
                }
            }
            $queries = array_merge($struct_queries, $data_queries);
            $ok = true;
            $error_queries = array();
            foreach($queries as $query) {
                // hack for 2.4
                if( stripos($query, 'country ADD PRIMARY KEY') ) {
                   continue;
                }
                $res = $this->query($query);
                if(!$res) {
                    $ok = false;
                    $error_queries[] = $query;
                }
            }
            if(!empty($error_queries)) {
                error_log(' --- error_queries ---');
                foreach($struct_queries as $q) {
                    error_log(' --- ' . $q );
                }
            }
            // hack
            $this->query("SET FOREIGN_KEY_CHECKS = 1");
            error_log(' ----- END updateDB ----- ');
            return array($ok, $queries, $error_queries);
        }
        /**
         * Set aSet array
         *
         * @access public
         * @since 2.3
         * @param mixed $key
         * @param mixed $value
         * @return DBCommandClass
         */
        function set($key, $value = '', $escape = true)
        {
            if( !is_array($key) ) {
                $key = array($key => $value);
            }
            foreach($key as $k => $v) {
                if( $escape ) {
                    $this->aSet[$k] = $this->escape($v);
                } else {
                    $this->aSet[$k] = $v;
                }
            }
            return $this;
        }
        /**
         * Create SELECT sql statement
         *
         * @access private
         * @since 2.3
         * @return string
         */
        function _getSelect()
        {
            $sql = 'SELECT ';
            // "SELECT" portion of the query
            if( count($this->aSelect) == 0 ) {
                $sql .= '*';
            } else {
                $sql .= implode(', ', $this->aSelect);
            }
            // "FROM" portion of the query
            if( count($this->aFrom) > 0 ) {
                $sql .= "nFROM ";
                if( !is_array($this->aFrom) ) {
                    $this->a_from = array($this->aFrom);
                }
                $sql .= '(' . implode(', ', $this->aFrom) . ')';
            }
            // "JOIN" portion of the query
            if( count($this->aJoin) > 0 ) {
                $sql .= "n";
                $sql .= implode("n", $this->aJoin);
            }
            // "WHERE" portion of the query
            if( count($this->aWhere) > 0 || count($this->aLike) > 0 ) {
                $sql .= "n";
                $sql .= "WHERE ";
            }
            $sql .= implode("n", $this->aWhere);
            // "LIKE" portion of the query
            if( count($this->aLike) > 0 ) {
                if( count($this->aWhere) > 0 ) {
                    $sql .= "nAND";
                }
                $sql .= implode("n", $this->aLike);
            }
            // "GROUP BY" portion of the query
            if( count($this->aGroupby) > 0 ) {
                $sql .= "nGROUP BY ";
                $sql .= implode(', ', $this->aGroupby);
            }
            // "HAVING" portion of the query
            if( count($this->aHaving) > 0 ) {
                $sql .= "nHAVING ";
                $sql .= implode(', ', $this->aHaving);
            }
            // "ORDER BY" portion of the query
            if( count($this->aOrderby) > 0 ) {
                $sql .= "nORDER BY ";
                $sql .= implode(', ', $this->aOrderby);
                if($this->aOrder !== false) {
                    $sql .= ($this->aOrder == 'desc') ? ' DESC' : ' ASC';
                }
            }
            // "LIMIT" portion of the query
            if( is_numeric($this->aLimit) ) {
                $sql .= "n";
                $sql .= "LIMIT " . $this->aLimit;
                if( $this->aOffset > 0 ) {
                    $sql .= ", " . $this->aOffset;
                }
            }
            return $sql;
        }
        /**
         * Gets the number of affected rows in a previous MySQL operation
         *
         * @access public
         * @since 2.3
         * @return int
         */
        function affectedRows()
        {
            return $this->connId->affected_rows;
        }
        /**
         * Get last SQL query
         *
         * @access public
         * @since 2.3
         * @return string
         */
        function lastQuery()
        {
            return end($this->queries);
        }
        /**
         * Get the ID generated from the previous INSERT operation
         *
         * @access public
         * @since 2.3
         * @return mixed
         */
        function insertedId()
        {
            return $this->connId->insert_id;
        }
        /**
         * Check if the string has an operator
         *
         * @access private
         * @since 2.3
         * @param string $str
         * @return bool
         */
        function _hasOperator($str)
        {
            $str = trim($str);
            if ( ! preg_match('/(s|<|>|!|=|is null|is not null)/i', $str)) {
                return false;
            }
            return true;
        }
        /**
         * Check if the sql is a select
         *
         * @access private
         * @since 2.3
         * @param string $sql
         * @return bool
         */
        function isSelectType($sql)
        {
            if ( ! preg_match('/^s*"?(SELECT)s+/i', $sql)) {
                return false;
            }
            return true;
        }
        /**
         * Check if the sql is a write type such as INSERT, UPDATE, UPDATE...
         *
         * @access private
         * @since 2.3
         * @param string $sql
         * @return bool
         */
        function isWriteType($sql)
        {
            if ( ! preg_match('/^s*"?(SET|INSERT|UPDATE|DELETE|REPLACE|CREATE|DROP|TRUNCATE|LOAD DATA|COPY|ALTER|GRANT|REVOKE|LOCK|UNLOCK|RENAME)s+/i', $sql)) {
                return false;
            }
            return true;
        }
        /**
         * Add the apostrophe if it's an string; 0 or 1 if it's a number; NULL
         *
         * @access private
         * @since 2.3
         * @param string $str
         * @return string
         */
        function escape($str)
        {
            if( is_string($str) ) {
                $str = "'" . $this->escapeStr($str) . "'";
            } elseif ( is_bool($str) ) {
                $str = ($str === false) ? 0 : 1;
            } elseif( is_null($str) ) {
                $str = 'NULL';
            }
            return $str;
        }
        /**
         * Escape the string if it's necessary
         *
         * @access private
         * @since 2.3
         * @param string $str
         * @return string
         */
        function escapeStr($str, $like = false)
        {
            if( is_object($this->connId) ) {
                $str = $this->connId->real_escape_string($str);
            } else {
                $str = addslashes($str);
            }
            if( $like ) {
                $str = str_replace(array('%', '_'), array('\%', '\_'), $str);
            }
            return $str;
        }
        /**
         * Reset variables used in write sql: aSet, aFrom, aWhere, aLike, aOrderby, aLimit, aOrder
         *
         * @access private
         * @since 2.3
         */
        function _resetWrite()
        {
            $aReset = array('aSet'     => array(),
                            'aFrom'    => array(),
                            'aWhere'   => array(),
                            'aLike'    => array(),
                            'aOrderby' => array(),
                            'aLimit'   => false,
                            'aOrder'   => false );
            $this->_resetRun($aReset);
        }
        /**
         * Reset variables used in select sql: aSelect, aFrom, aJoin, aWhere, aLike, aGroupby, aHaving,
         * aOrderby, aWherein, aLimit, aOffset, aOrder
         *
         * @access private
         * @since 2.3
         */
        function _resetSelect()
        {
            $aReset = array('aSelect'  => array(),
                            'aFrom'    => array(),
                            'aJoin'    => array(),
                            'aWhere'   => array(),
                            'aLike'    => array(),
                            'aGroupby' => array(),
                            'aHaving'  => array(),
                            'aOrderby' => array(),
                            'aWherein' => array(),
                            'aLimit'   => false,
                            'aOffset'  => false,
                            'aOrder'   => false );
            $this->_resetRun($aReset);
        }
        /**
         * Initializate $aReset variables
         *
         * @access private
         * @since 2.3
         * @param array $aReset
         */
        function _resetRun($aReset){
            foreach ($aReset as $item => $defaultValue) {
                $this->$item = $defaultValue;
            }
        }
        /**
         * Set last error code and descriptionfor the most recent mysqli function call
         *
         * @access private
         * @since 2.3
         */
        function errorReport()
        {
            $this->errorLevel = $this->connId->errno;
            $this->errorDesc  = $this->connId->error;
        }
        /**
         * Returns the last error code for the most recent mysqli function call
         *
         * @access public
         * @since 2.3
         * @return int
         */
        function getErrorLevel()
        {
            return $this->errorLevel;
        }
        /**
         * Returns a string description of the last error for the most recent MySQLi function call
         *
         * @access public
         * @since 2.3
         * @return string
         */
        function getErrorDesc()
        {
            return $this->errorDesc;
        }
    }
    /* file end: ./oc-includes/osclass/classes/database/DBCommandClass.php */
?>