Всего: 2 1 2 >
 Поиск Sphinx в PrestaShop
Gorets


Cпециалист
: 143
: 19-10-2009


25-04-2011 12:08
Кто нибудь уже пробовал виполнить данное действие?, так как насколько я помню данная тема обсуждалась.
http://www.musicdestock.fr/
пример магазина на данной связке, поиск работает просто потрясно

 
  
ors
Профессионал
: 1885
: 09-11-2009


25-04-2011 15:36
Код для поиска через сфинкс еще проще, чем для стандартного поиска
 
  
Gorets


Cпециалист
: 143
: 19-10-2009


25-04-2011 15:58
хм.. может и так. но я какби сам не сделаю, навиков не хватает.
на веблансере запросили 300уе, что какби дорого
может здесь есть желающие реализовать за умеренную цену?
 
  
Fant63
Профессионал
: 543
: Тольятти
: 23-02-2010


25-04-2011 16:53
Вопрос по поводу поиска - в админке на вкладке модули есть строка для поиска - так вот она работает супер, даже с кириллическими символами а поисковик по магазину ужасен - хотя я думаю что код там один и тот же... или ошибаюсь?
 
 
Gorets


Cпециалист
: 143
: 19-10-2009


25-04-2011 18:11
Ошибаетесь.
Цитата:
даже с кириллическими символами
чтото, а проблем с кирилицой никогда не наблюдал (имею ввиду оф версию) -
разве что неправильная настройка Мускула
Вобще сама идея поиска в престе с использованием таблици ключевих слов очень неплоха, теоритически по такой схеме можно без проблем иметь базу и с миллионом товаров.
Но фактическая реализация просто убивает.. особенно для IT магазинов со сложными составными названиями это жесть. Найти чтото - счастливой совпадение.
Что странно - практически с каждой версией делал баг репорты на трекере, но уже почти год они просто игнорируются.."это не баг, это фича"
последний также оставлен без внимания http://forge.prestashop.com/browse/PSCFI-1582
Кого напрагает поиск в престе, подпишитесь типа I have some problem


Сообщение отредактировано Gorets 25-04-2011 19:16 ...
 
  
ors
Профессионал
: 1885
: 09-11-2009


25-04-2011 18:55
Вообще то это стандартный подход к поиску. Не думаю что в другом движке это сделано лучше. Если найдете такой, то можно попробовать портировать.

А вообще в случае поиска фразы 1010 в товаре Тонер ATT HP LJ1010/1012/1015 100г можно обойтись простым LIKE по названию товара
 
  
Gorets


Cпециалист
: 143
: 19-10-2009


25-04-2011 19:00
Цитата:
А вообще в случае поиска фразы 1010 в товаре Тонер ATT HP LJ1010/1012/1015 100г можно обойтись простым LIKE по названию товара
Ну это просто один из примеров - но просто LIKE какби не поможет.
При небольшой БД в 30-40мб, 2500 товаров, LIKE по поиску в описании и названии занимает около 10 секунд, при том что сайт находится на хорошем виделенном сервере 2*2.2х8 8 гб озу.
С чем собственно и связан мой вопрос со сфинксом.

Ну особо не помню как там реализован, но когдато пользовался virtuemart, с поиском никогда проблем небыло
http://demo.virtuemart.net/


Сообщение отредактировано Gorets 25-04-2011 20:05 ...
 
  
ors
Профессионал
: 1885
: 09-11-2009


25-04-2011 19:50
Для использования сфинкса заменить в classes\Search .php функцию find
Код:
public static function find($id_lang, $expr, $pageNumber = 1, $pageSize = 1, $orderBy = 'position', $orderWay = 'desc', $ajax = false)
    {
        global $cookie;
        $db = Db::getInstance(_PS_USE_SQL_SLAVE_);

        // TODO : smart page management
        if ($pageNumber < 1) $pageNumber = 1;
        if ($pageSize < 1) $pageSize = 1;

        if (!Validate::isOrderBy($orderBy) OR !Validate::isOrderWay($orderWay))
            die(Tools::displayError());

//==========sphinx fix===============
include('sphinxapi.php');
$cl = new SphinxClient();
$cl->SetServer( "localhost", 9312 );
$cl->SetMatchMode( SPH_MATCH_ANY  );
$result = $cl->Query($expr);
if (!$result)
    return false;
$eligibleProducts = $result["matches"];
if (empty($eligibleProducts))
    return false;

        $productPool = '';
        foreach ($eligibleProducts AS $product=>$info)
            if (!empty($product))
                $productPool .= (int)$product.',';
        if (empty($productPool))
            return ($ajax ? array() : array('total' => 0, 'result' => array()));
        $productPool = ((strpos($productPool, ',') === false) ? (' = '.(int)$productPool.' ') : (' IN ('.rtrim($productPool, ',').') '));
//==================================
        if ($ajax)
        {
            if (!$result = $db->ExecuteS('
            SELECT DISTINCT p.id_product, pl.name pname, cl.name cname,
                cl.link_rewrite crewrite, pl.link_rewrite prewrite '.$score.'
            FROM '._DB_PREFIX_.'product p
            INNER JOIN `'._DB_PREFIX_.'product_lang` pl ON (p.`id_product` = pl.`id_product` AND pl.`id_lang` = '.(int)$id_lang.')
            INNER JOIN `'._DB_PREFIX_.'category_lang` cl ON (p.`id_category_default` = cl.`id_category` AND cl.`id_lang` = '.(int)$id_lang.')
            WHERE p.`id_product` '.$productPool.'
            ORDER BY position DESC LIMIT 10'))
                return false;

            return $result;
        }

        $queryResults = '
        SELECT SQL_CALC_FOUND_ROWS p.*, pl.`description_short`, pl.`available_now`, pl.`available_later`, pl.`link_rewrite`, pl.`name`, pa.`id_product_attribute`,
            tax.`rate`, i.`id_image`, il.`legend`, m.`name` manufacturer_name '.$score.', DATEDIFF(p.`date_add`, DATE_SUB(NOW(), INTERVAL '.(Validate::isUnsignedInt(Configuration::get('PS_NB_DAYS_NEW_PRODUCT')) ? Configuration::get('PS_NB_DAYS_NEW_PRODUCT') : 20).' DAY)) > 0 new
        FROM '._DB_PREFIX_.'product p
        INNER JOIN `'._DB_PREFIX_.'product_lang` pl ON (p.`id_product` = pl.`id_product` AND pl.`id_lang` = '.(int)$id_lang.')
        LEFT JOIN `'._DB_PREFIX_.'tax_rule` tr ON (p.`id_tax_rules_group` = tr.`id_tax_rules_group`
                                                   AND tr.`id_country` = '.(int)Country::getDefaultCountryId().'
                                                      AND tr.`id_state` = 0)
        LEFT JOIN `'._DB_PREFIX_.'tax` tax ON (tax.`id_tax` = tr.`id_tax`)
        LEFT JOIN `'._DB_PREFIX_.'product_attribute` pa ON (p.`id_product` = pa.`id_product` AND default_on = 1)
        LEFT JOIN `'._DB_PREFIX_.'manufacturer` m ON m.`id_manufacturer` = p.`id_manufacturer`
        LEFT JOIN `'._DB_PREFIX_.'image` i ON (i.`id_product` = p.`id_product` AND i.`cover` = 1)
        LEFT JOIN `'._DB_PREFIX_.'image_lang` il ON (i.`id_image` = il.`id_image` AND il.`id_lang` = '.(int)$id_lang.')
        WHERE p.`id_product` '.$productPool.'
        '.($orderBy ? 'ORDER BY  '.$orderBy : '').($orderWay ? ' '.$orderWay : '').'
        LIMIT '.(int)(($pageNumber - 1) * $pageSize).','.(int)$pageSize;

        $result = $db->ExecuteS($queryResults);
        $total = $db->getValue('SELECT COUNT(*)
        FROM '._DB_PREFIX_.'product p
        INNER JOIN `'._DB_PREFIX_.'product_lang` pl ON (p.`id_product` = pl.`id_product` AND pl.`id_lang` = '.(int)$id_lang.')
        LEFT JOIN `'._DB_PREFIX_.'tax_rule` tr ON (p.`id_tax_rules_group` = tr.`id_tax_rules_group`
                                                   AND tr.`id_country` = '.(int)Country::getDefaultCountryId().'
                                                      AND tr.`id_state` = 0)
        LEFT JOIN `'._DB_PREFIX_.'tax` tax ON (tax.`id_tax` = tr.`id_tax`)
        LEFT JOIN `'._DB_PREFIX_.'product_attribute` pa ON (p.`id_product` = pa.`id_product` AND default_on = 1)
        LEFT JOIN `'._DB_PREFIX_.'manufacturer` m ON m.`id_manufacturer` = p.`id_manufacturer`
        LEFT JOIN `'._DB_PREFIX_.'image` i ON (i.`id_product` = p.`id_product` AND i.`cover` = 1)
        LEFT JOIN `'._DB_PREFIX_.'image_lang` il ON (i.`id_image` = il.`id_image` AND il.`id_lang` = '.(int)$id_lang.')
        WHERE p.`id_product` '.$productPool);
        
        return array('total' => $total,'result' => Product::getProductsProperties($id_lang, $result));
    }


Это код для версии 1.4. sphinxapi.php должен лежать в той же папке


Сообщение отредактировано ors 25-04-2011 20:52 ...
 
  
ors
Профессионал
: 1885
: 09-11-2009


25-04-2011 19:59
Вот конфиг сфинкса, который еще допиливать и допиливать, но с ним работает.

Код:
source product
{
    type            = mysql
    sql_host        = localhost
    sql_user        = root
    sql_pass        =homserv
    sql_db            = ps17
    sql_port        = 3306    # optional, default is 3306
     sql_query_pre        = SET NAMES utf8
    sql_query        = \
        SELECT * FROM ps_product_lang
    sql_attr_uint        = id_product
    sql_attr_timestamp    = date_add
    sql_ranged_throttle    = 0
    sql_query_info        = SELECT * FROM ps_product_lang WHERE id_product=$id
}

index product
{
    source            = product
    path            = D:\HomServ\sphinx/data/product
    morphology        = stem_ru
    min_word_len        = 1
    charset_type        = utf-8

}

searchd
{

    listen            = 9312
    log            = D:\HomServ\sphinx/log/searchd.log
    query_log        = D:\HomServ\sphinx/log/query.log
    read_timeout        = 5
    client_timeout        = 300
    max_children        = 30
    pid_file        = D:\HomServ\sphinx/log/searchd.pid
    max_matches        = 1000
    seamless_rotate        = 1
    preopen_indexes        = 1
    unlink_old        = 1
    workers            = threads # for RT to work
}

# --eof--
 
  
Gorets


Cпециалист
: 143
: 19-10-2009


25-04-2011 20:24
ок сенкс, попробую отпишусь.
 
  
   
Всего: 2 1 2 >