1
0
Fork 0
mirror of https://github.com/YunoHost-Apps/spip_ynh.git synced 2024-09-03 20:25:59 +02:00
spip_ynh/sources/ecrire/inc/prepare_recherche.php
2015-04-28 17:10:23 +02:00

160 lines
5.6 KiB
PHP
Raw Blame History

<?php
/***************************************************************************\
* SPIP, Systeme de publication pour l'internet *
* *
* Copyright (c) 2001-2014 *
* Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James *
* *
* Ce programme est un logiciel libre distribue sous licence GNU/GPL. *
* Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. *
\***************************************************************************/
if (!defined('_ECRIRE_INC_VERSION')) return;
include_spip('inc/rechercher');
if (!defined('_DELAI_CACHE_resultats')) define('_DELAI_CACHE_resultats', 600);
/**
* Preparer les listes id_article IN (...) pour les parties WHERE
* et points = des requetes du moteur de recherche
* http://doc.spip.org/@inc_prepare_recherche_dist
*
* Le parametre $serveur est utilise pour savoir sur quelle base on cherche
* mais l'index des resultats est toujours stock<63> sur le serveur principal
* car on ne sait pas si la base distante dispose d'une table spip_resultats
* ni meme si on aurait le droit d'ecrire dedans
*
* @param string $recherche
* chaine recherchee
* @param string $table
* table dans laquelle porte la recherche
* @param bool $cond
* critere conditionnel sur {recherche?}
* @param string $serveur
* serveur de base de donnees
* @param array $modificateurs
* modificateurs de boucle, ie liste des criteres presents
* @param string $primary
* cle primaire de la table de recherche
* @return array
*/
function inc_prepare_recherche_dist($recherche, $table='articles', $cond=false, $serveur='', $modificateurs = array(), $primary='') {
static $cache = array();
$delai_fraicheur = min(_DELAI_CACHE_resultats,time()-$GLOBALS['meta']['derniere_modif']);
// si recherche n'est pas dans le contexte, on va prendre en globals
// ca permet de faire des inclure simple.
if (!isset($recherche) AND isset($GLOBALS['recherche']))
$recherche = $GLOBALS['recherche'];
// traiter le cas {recherche?}
if ($cond AND !strlen($recherche))
return array("0 as points" /* as points */, /* where */ '');
$rechercher = false;
if (!isset($cache[$serveur][$table][$recherche])){
$hash_serv = ($serveur?substr(md5($serveur),0,16):'');
$hash = substr(md5($recherche . $table),0,16);
$where = "(resultats.recherche='$hash' AND resultats.table_objet=".sql_quote($table)." AND resultats.serveur='$hash_serv')";
$row = sql_fetsel('UNIX_TIMESTAMP(NOW())-UNIX_TIMESTAMP(resultats.maj) AS fraicheur','spip_resultats AS resultats',$where,'','fraicheur DESC','0,1');
if (!$row
OR ($row['fraicheur']>$delai_fraicheur)
OR (defined('_VAR_MODE') AND _VAR_MODE=='recalcul')){
$rechercher = true;
}
}
// si on n'a pas encore traite les donnees dans une boucle precedente
if ($rechercher) {
//$tables = liste_des_champs();
$x = objet_type($table);
$points = recherche_en_base($recherche,
$x,
array(
'score' => true,
'toutvoir' => true,
'jointures' => true
),
$serveur);
// pas de résultat, pas de point
$points = isset($points[$x]) ? $points[$x] : array();
// permettre aux plugins de modifier le resultat
$points = pipeline('prepare_recherche',array(
'args'=>array('type'=>$x,'recherche'=>$recherche,'serveur'=>$serveur,'modificateurs'=>$modificateurs),
'data'=>$points
));
// supprimer les anciens resultats de cette recherche
// et les resultats trop vieux avec une marge
// pas de AS resultats dans un delete (mysql)
$whered = str_replace(array("resultats.recherche","resultats.table_objet","resultats.serveur"),array("recherche","table_objet","serveur"),$where);
sql_delete('spip_resultats', 'NOT(' .sql_date_proche('maj', (0-($delai_fraicheur+100)), " SECOND") . ") OR ($whered)");
// inserer les resultats dans la table de cache des resultats
if (count($points)){
$tab_couples = array();
foreach ($points as $id => $p){
$tab_couples[] = array(
'recherche' => $hash,
'id' => $id,
'points' => $p['score'],
'table_objet' => $table,
'serveur' => $hash_serv,
);
}
sql_insertq_multi('spip_resultats',$tab_couples,array());
}
}
if (!isset($cache[$serveur][$table][$recherche])){
if (!$serveur)
$cache[$serveur][$table][$recherche] = array("resultats.points AS points",$where);
else {
if (sql_countsel('spip_resultats as resultats',$where))
$rows = sql_allfetsel('resultats.id,resultats.points','spip_resultats as resultats',$where);
$cache[$serveur][$table][$recherche] = generer_select_where_explicites($table, $primary, $rows, $serveur);
}
}
return $cache[$serveur][$table][$recherche];
}
/**
* Generer le select et where qui contiennent explicitement
* les id et points (ie comme dans SPIP 1.9.x)
* quand on fait une recherche sur une table externe
*
* @param string $table
* @param string $primary
* @param array $rows
* @param string $serveur
* @return array
*/
function generer_select_where_explicites($table, $primary, $rows, $serveur){
# calculer le {id_article IN()} et le {... as points}
if (!count($rows)){
return array("''", "0=1");
}
else {
$listes_ids = array();
$select = '0';
foreach ($rows as $r)
$listes_ids[$r['points']][] = $r['id'];
foreach ($listes_ids as $p => $ids)
$select .= "+$p*(".
sql_in("$table.$primary", $ids,'',$serveur)
.") ";
return array("$select AS points ",calcul_mysql_in("$table.$primary",array_map('reset',$rows),'',$serveur));
}
}
?>