Query.php 7.2 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
 * @link        www.phpdoctrine.com
 * @since       1.0
 */
class Doctrine_Search_Query
{
zYne's avatar
zYne committed
35

zYne's avatar
zYne committed
36
    /**
zYne's avatar
zYne committed
37
     * @var Doctrine_Table $_table          the index table
zYne's avatar
zYne committed
38
     */
zYne's avatar
zYne committed
39 40 41
    protected $_table = array();
    
    protected $_sql = '';
zYne's avatar
zYne committed
42
    
43 44
    protected $_params = array();
    
zYne's avatar
zYne committed
45 46
    protected $_words = array();
    
47
    protected $_tokenizer;
48

zYne's avatar
zYne committed
49
    protected $_condition;
50

zYne's avatar
zYne committed
51
    /**
zYne's avatar
zYne committed
52
     * @param Doctrine_Table $_table        the index table
zYne's avatar
zYne committed
53
     */
zYne's avatar
zYne committed
54
    public function __construct($table)
zYne's avatar
zYne committed
55
    {
zYne's avatar
zYne committed
56 57
        if (is_string($table)) {
           $table = Doctrine_Manager::table($table);
zYne's avatar
zYne committed
58 59 60 61
        } else {
            if ( ! $table instanceof Doctrine_Table) {
                throw new Doctrine_Search_Exception('Invalid argument type. Expected instance of Doctrine_Table.');
            }
zYne's avatar
zYne committed
62
        }
zYne's avatar
zYne committed
63

64
        $this->_tokenizer = new Doctrine_Query_Tokenizer();
zYne's avatar
zYne committed
65 66
        $this->_table = $table;

zYne's avatar
zYne committed
67 68
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));

69
        $this->_condition = $foreignId . ' %s (SELECT ' . $foreignId . ' FROM ' . $this->_table->getTableName() . ' WHERE ';
zYne's avatar
zYne committed
70 71
    }

zYne's avatar
zYne committed
72 73

    public function query($text)
zYne's avatar
zYne committed
74
    {
75
        $text = trim($text);
zYne's avatar
zYne committed
76

77
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));
zYne's avatar
zYne committed
78

zYne's avatar
zYne committed
79 80 81 82 83 84 85 86 87 88 89
        $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
90
        
zYne's avatar
zYne committed
91 92 93
        $where = 'WHERE ';
        $where .= $this->parseClause($text);

zYne's avatar
zYne committed
94
        $groupby = 'GROUP BY ' . $foreignId;
95
        $orderby = 'ORDER BY relevance DESC';
zYne's avatar
zYne committed
96

zYne's avatar
zYne committed
97 98
        $this->_sql = $select . ' ' . $from . ' ' . $where . ' ' . $groupby . ' ' . $orderby;
    }
zYne's avatar
zYne committed
99

100
    public function parseClause($originalClause, $recursive = false)
zYne's avatar
zYne committed
101
    {
102
        $clause = $this->_tokenizer->bracketTrim($originalClause);
zYne's avatar
zYne committed
103 104
        
        $brackets = false;
zYne's avatar
zYne committed
105

zYne's avatar
zYne committed
106 107 108
        if ($clause !== $originalClause) {
            $brackets = true;
        }
zYne's avatar
zYne committed
109

zYne's avatar
zYne committed
110 111
        $foreignId = current(array_diff($this->_table->getColumnNames(), array('keyword', 'field', 'position')));
        
112
        $terms = $this->_tokenizer->sqlExplode($clause, ' OR ', '(', ')');
113

zYne's avatar
zYne committed
114 115
        $ret = array();

zYne's avatar
zYne committed
116 117
        if (count($terms) > 1) {
            $leavesOnly = true;
118

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

zYne's avatar
zYne committed
128
            $return = implode(' OR ', $ret);
zYne's avatar
zYne committed
129

130 131 132
            if ($leavesOnly && $recursive) {
                $return = sprintf($this->_condition, 'IN') . $return . ')';
                $brackets = false;
zYne's avatar
zYne committed
133 134
            }
        } else {
135
            $terms = $this->_tokenizer->sqlExplode($clause, ' ', '(', ')');
136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
            
            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
159
                }
160
                $return = implode(' AND ', $ret);
zYne's avatar
zYne committed
161 162
            }
        }
163

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

zYne's avatar
zYne committed
181 182
    public function parseTerm($term)
    {
183
        $negation = false;
zYne's avatar
zYne committed
184

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

190
            $terms = $this->_tokenizer->quoteExplode($term);
191 192
            $where = $this->parseWord($terms[0]);

zYne's avatar
zYne committed
193 194 195 196
            foreach ($terms as $k => $word) {
                if ($k === 0) {
                    continue;
                }
197
                $where .= ' AND (position + ' . $k . ') = (SELECT position FROM ' . $this->_table->getTableName() . ' WHERE ' . $this->parseWord($word) . ')';
zYne's avatar
zYne committed
198
            }
zYne's avatar
zYne committed
199
        }
zYne's avatar
zYne committed
200
        return $where;
zYne's avatar
zYne committed
201
    }
zYne's avatar
zYne committed
202
    public function parseWord($word)
203
    {
zYne's avatar
zYne committed
204 205
        $this->_words[] = str_replace('*', '', $word);

206 207
        if (strpos($word, '?') !== false ||
            strpos($word, '*') !== false) {
zYne's avatar
zYne committed
208

209
            $word = str_replace('*', '%', $word);
zYne's avatar
zYne committed
210

211
            $where = 'keyword LIKE ?';
zYne's avatar
zYne committed
212

213 214 215 216
            $params = array($word);
        } else {
            $where = 'keyword = ?';
        }
zYne's avatar
zYne committed
217

218 219 220 221
        $this->_params[] = $word;

        return $where;
    }
zYne's avatar
zYne committed
222 223 224 225 226

    public function getWords()
    {
        return $this->_words;
    }
227 228 229 230
    public function getParams()
    {
        return $this->_params;
    }
zYne's avatar
zYne committed
231 232 233 234
    public function getSql()
    {
        return $this->_sql;
    }
zYne's avatar
zYne committed
235
}