Query.php 7.25 KB
Newer Older
zYne's avatar
zYne committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
<?php
/*
 *  $Id: Hook.php 1939 2007-07-05 23:47:48Z zYne $
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 * This software consists of voluntary contributions made by many individuals
 * and is licensed under the LGPL. For more information, see
 * <http://www.phpdoctrine.com>.
 */

/**
 * Doctrine_Search_Query
 *
 * @package     Doctrine
26 27
 * @subpackage  Search
 * @author      Konsta Vesterinen <kvesteri@cc.hut.fi>
zYne's avatar
zYne committed
28
 * @license     http://www.opensource.org/licenses/lgpl-license.php LGPL
29
 * @version     $Revision$
zYne's avatar
zYne committed
30 31 32 33 34 35 36 37 38 39
 * @link        www.phpdoctrine.com
 * @since       1.0
 */
class Doctrine_Search_Query
{
    /**
     * @var Doctrine_Query $query           the base query
     */
    protected $_query;
    /**
zYne's avatar
zYne committed
40
     * @var Doctrine_Table $_table          the index table
zYne's avatar
zYne committed
41
     */
zYne's avatar
zYne committed
42 43 44
    protected $_table = array();
    
    protected $_sql = '';
zYne's avatar
zYne committed
45
    
46 47
    protected $_params = array();
    
48

zYne's avatar
zYne committed
49
    protected $_condition;
zYne's avatar
zYne committed
50
    /**
zYne's avatar
zYne committed
51
     * @param octrine_Table $_table         the index table
zYne's avatar
zYne committed
52
     */
zYne's avatar
zYne committed
53
    public function __construct($table)
zYne's avatar
zYne committed
54
    {
zYne's avatar
zYne committed
55 56 57
        if (is_string($table)) {
           $table = Doctrine_Manager::table($table);
        }
zYne's avatar
zYne committed
58 59 60 61

        $this->_table = $table;

        $this->_query = new Doctrine_Query();
zYne's avatar
zYne committed
62 63
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));

64
        $this->_condition = $foreignId . ' %s (SELECT ' . $foreignId . ' FROM ' . $this->_table->getTableName() . ' WHERE ';
zYne's avatar
zYne committed
65 66 67 68 69 70 71 72 73 74 75 76
    }
    /**
     * getQuery
     *
     * @return Doctrine_Query       returns the query object associated with this object
     */
    public function getQuery()
    {
        return $this->_query;
    }

    public function search($text)
zYne's avatar
zYne committed
77
    {
78 79 80
        $text = trim($text);
        
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));
zYne's avatar
zYne committed
81

zYne's avatar
zYne committed
82 83 84 85 86 87 88 89 90 91 92
        $weighted = false;
        if (strpos($text, '^') === false) {
            $select = 'SELECT COUNT(keyword) AS relevance, ' . $foreignId;
            $from = 'FROM ' . $this->_table->getTableName();
        } else {
            // organize terms according weights
            $weighted = true;

            $select = 'SELECT SUM(sub_relevance) AS relevance, ' . $foreignId;
            $from = 'FROM ' ;
        }
zYne's avatar
zYne committed
93
        
zYne's avatar
zYne committed
94 95 96
        $where = 'WHERE ';
        $where .= $this->parseClause($text);

zYne's avatar
zYne committed
97 98 99
        $groupby = 'GROUP BY ' . $foreignId;
        $orderby = 'ORDER BY relevance';

zYne's avatar
zYne committed
100 101
        $this->_sql = $select . ' ' . $from . ' ' . $where . ' ' . $groupby . ' ' . $orderby;
    }
zYne's avatar
zYne committed
102

103
    public function parseClause($originalClause, $recursive = false)
zYne's avatar
zYne committed
104 105 106 107
    {
        $clause = Doctrine_Tokenizer::bracketTrim($originalClause);
        
        $brackets = false;
zYne's avatar
zYne committed
108

zYne's avatar
zYne committed
109 110 111
        if ($clause !== $originalClause) {
            $brackets = true;
        }
zYne's avatar
zYne committed
112

zYne's avatar
zYne committed
113 114 115
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));
        
        $terms = Doctrine_Tokenizer::sqlExplode($clause, ' OR ', '(', ')');
116

zYne's avatar
zYne committed
117 118
        $ret = array();

zYne's avatar
zYne committed
119 120
        if (count($terms) > 1) {
            $leavesOnly = true;
121

zYne's avatar
zYne committed
122 123
            foreach ($terms as $k => $term) {
                if ($this->isExpression($term)) {
124
                    $ret[$k] = $this->parseClause($term, true);
zYne's avatar
zYne committed
125
                    $leavesOnly = false;
zYne's avatar
zYne committed
126
                } else {
zYne's avatar
zYne committed
127
                    $ret[$k] = $this->parseTerm($term);
zYne's avatar
zYne committed
128 129 130
                }
            }

zYne's avatar
zYne committed
131
            $return = implode(' OR ', $ret);
zYne's avatar
zYne committed
132

133 134 135
            if ($leavesOnly && $recursive) {
                $return = sprintf($this->_condition, 'IN') . $return . ')';
                $brackets = false;
zYne's avatar
zYne committed
136 137 138
            }
        } else {
            $terms = Doctrine_Tokenizer::sqlExplode($clause, ' ', '(', ')');
139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
            
            if (count($terms) === 1 && ! $recursive) {
                $return = $this->parseTerm($clause);
            } else {
                foreach ($terms as $k => $term) {
                    $term = trim($term);
    
                    if ($term === 'AND') {
                        continue;
                    }
    
                    if (substr($term, 0, 1) === '-') {
                        $operator = 'NOT IN';
                        $term = substr($term, 1);
                    } else {
                        $operator = 'IN';
                    }
    
                    if ($this->isExpression($term)) {
                        $ret[$k] = $this->parseClause($term, true);
                    } else {
                        $ret[$k] = sprintf($this->_condition, $operator) . $this->parseTerm($term) . ')';
                    }
zYne's avatar
zYne committed
162
                }
163
                $return = implode(' AND ', $ret);
zYne's avatar
zYne committed
164 165
            }
        }
166

zYne's avatar
zYne committed
167 168
        if ($brackets) {
            return '(' . $return . ')';
zYne's avatar
zYne committed
169
        } else {
zYne's avatar
zYne committed
170 171 172 173 174 175 176 177 178 179 180
            return $return;
        }
    }
    public function isExpression($term)
    {
        if (strpos($term, '(') !== false) {
            return true;
        } else {
            $terms = Doctrine_Tokenizer::quoteExplode($term);
            
            return (count($terms) > 1);
zYne's avatar
zYne committed
181 182
        }
    }
zYne's avatar
zYne committed
183

zYne's avatar
zYne committed
184 185
    public function parseTerm($term)
    {
186
        $negation = false;
zYne's avatar
zYne committed
187

zYne's avatar
zYne committed
188
        if (strpos($term, "'") === false) {
189
            $where = $this->parseWord($term);
zYne's avatar
zYne committed
190 191
        } else {
            $term = trim($term, "' ");
192

zYne's avatar
zYne committed
193
            $terms = Doctrine_Tokenizer::quoteExplode($term);
194 195
            $where = $this->parseWord($terms[0]);

zYne's avatar
zYne committed
196 197 198 199
            foreach ($terms as $k => $word) {
                if ($k === 0) {
                    continue;
                }
200
                $where .= ' AND (position + ' . $k . ') = (SELECT position FROM ' . $this->_table->getTableName() . ' WHERE ' . $this->parseWord($word) . ')';
zYne's avatar
zYne committed
201
            }
zYne's avatar
zYne committed
202
        }
zYne's avatar
zYne committed
203
        return $where;
zYne's avatar
zYne committed
204
    }
205 206 207 208 209 210
    public function parseWord($word) 
    {
        if (strpos($word, '?') !== false ||
            strpos($word, '*') !== false) {
            
            $word = str_replace('*', '%', $word);
zYne's avatar
zYne committed
211

212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
            $where = 'keyword LIKE ?';
            
            $params = array($word);
        } else {
            $where = 'keyword = ?';
        }
        
        $this->_params[] = $word;

        return $where;
    }
    public function getParams()
    {
        return $this->_params;
    }
zYne's avatar
zYne committed
227 228 229 230
    public function getSql()
    {
        return $this->_sql;
    }
zYne's avatar
zYne committed
231 232 233 234 235 236
    public function execute()
    {
        $resultSet = $this->_query->execute(); 
        
        return $resultSet;
    }
237
}