Bifurcation depuis
spip / spip
15141 validations de retard le dépôt en amont.
-
Christian Lefebvre a rédigéChristian Lefebvre a rédigé
import.php 12,74 Kio
<?php
/***************************************************************************\
* SPIP, Systeme de publication pour l'internet *
* *
* Copyright (c) 2001-2007 *
* 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/presentation');
include_spip('inc/acces');
include_spip('inc/indexation'); // pour la fonction primary_index_table
include_spip('base/abstract_sql');
// NB: Ce fichier peut ajouter des tables (old-style)
// donc il faut l'inclure "en globals"
if ($f = include_spip('mes_fonctions', false)) {
global $dossier_squelettes;
@include_once ($f);
}
if (@is_readable(_DIR_TMP."charger_plugins_fonctions.php")){
// chargement optimise precompile
include_once(_DIR_TMP."charger_plugins_fonctions.php");
}
global $IMPORT_tables_noerase;
$IMPORT_tables_noerase[]='spip_meta';
$GLOBALS['flag_ob_flush'] = function_exists('ob_flush');
// Retourne la premiere balise XML figurant dans le buffet de la sauvegarde
// et avance dans ce buffet jusqu'au '>' de cette balise.
// Si le 2e argument (passe par reference) est non vide
// ce qui precede cette balise y est mis.
// Les balises commencant par <! sont ignorees
// $abs_pos est globale pour pouvoir etre reinitialisee a la meta
// status_restauration en cas d'interruption sur TimeOut.
// Evite au maximum les recopies
// http://doc.spip.org/@xml_fetch_tag
function xml_fetch_tag($f, &$before, $_fread='fread', $skip='!') {
global $abs_pos;
static $buf='';
static $ent = array('&','<');
static $brut = array('&','<');
while (($b=strpos($buf,'<'))===false) {
if (!($x = $_fread($f, 1024))) return '';
if ($before)
$buf .= $x;
else $buf = $x;
}
if ($before) $before = str_replace($ent,$brut,substr($buf,0,$b));
# else { spip_log("position: $abs_pos" . substr($buf,0,12));flush();}
// $b pour ignorer un > de raccourci Spip avant un < de balise XML
while (($e=strpos($buf,'>', $b))===false) {
if (!($x = $_fread($f, 1024))) return '';
$buf .= $x;
}
if ($buf[++$b]!=$skip) {
$tag = substr($buf, $b, $e-$b);
$buf = substr($buf,++$e);
$abs_pos += $e;
return $tag;
}
$buf = substr($buf,++$e);
$abs_pos += $e;
return xml_fetch_tag($f,$before,$_fread,$skip);
}
// http://doc.spip.org/@xml_parse_tag
function xml_parse_tag($texte) {
list($tag, $atts) = split('[[:space:]]+', $texte, 2);
$result[0] = $tag;
$result[1] = array();
if (!$atts) return $result;
if ($tag=='!--'){
$result[1]=preg_replace(",(.*?)--$,s",'\\1',$atts);
}
else {
while (ereg('^([^[:space:]]+)[[:space:]]*=[[:space:]]*"([^"]*)"([[:space:]]+(.*))?', $atts, $regs)) {
$result[1][$regs[1]] = $regs[2];
$atts = $regs[4];
}
}
return $result;
}
// Balise ouvrante:
// 'SPIP' si fait par spip, nom de la base source si fait par phpmyadmin
// http://doc.spip.org/@import_debut
function import_debut($f, $gz='fread') {
// Pour les anciennes archives, indiquer le charset par defaut:
$charset = 'iso-8859-1';
// les + recentes l'ont en debut de ce fichier
$flag_phpmyadmin = false;
$b = false;
while ($t = xml_fetch_tag($f, $b, $gz, '')) {
$r = xml_parse_tag($t);
if ($r[0] == '?xml' AND $r[1]['encoding'])
$charset = strtolower($r[1]['encoding']);
elseif ($r[0] == "SPIP") {$r[2] = $charset; return $r;}
if (($r[0] == "!--") && (preg_match(",phpmyadmin\sxml\sdump,is",$r[1]))){
// c'est un dump xml phpmyadmin
// on interprete le commentaire pour recuperer la version de phpmydadmin
$version = preg_replace(",(.*?)version\s*([0-9a-z\.\-]*)\s(.*),is","\\2",$r[1]);
$flag_phpmyadmin = true;
}
if (($r[0] != "!--") && ($flag_phpmyadmin == true)){
$r[1] = array('version_archive'=>"phpmyadmin::$version");
$r[2] = $charset;
return $r;
}
}
}
// on conserve ce tableau pour faire des translations
// de table eventuelles
$tables_trans = array(
);
// http://doc.spip.org/@import_init_tables
function import_init_tables($request)
{
global $IMPORT_tables_noerase, $connect_id_auteur;
// grand menage
// on vide toutes les tables dont la restauration est demandee
$tables = import_table_choix($request);
foreach($tables as $table){
// regarder si il y a au moins un champ impt='non'
if (($table!='spip_auteurs')&&(!in_array($table,$IMPORT_tables_noerase))){
$desc = description_table($table);
if (isset($desc['field']['impt']))
spip_query("DELETE FROM $table WHERE impt='oui'");
else
spip_query("DELETE FROM $table");
}
}
// Bidouille pour garder l'acces admin actuel pendant toute la restauration
spip_query("UPDATE spip_auteurs SET id_auteur=0, extra=$connect_id_auteur WHERE id_auteur=$connect_id_auteur");
spip_query("DELETE FROM spip_auteurs WHERE id_auteur!=0");
return $tables;
}
// Effacement de la bidouille ci-dessus
// Toutefois si la table des auteurs ne contient plus qu'elle
// c'est que la sauvegarde etait incomplete et on restaure le compte
// pour garder la connection au site (mais il doit pas etre bien beau)
// http://doc.spip.org/@detruit_restaurateur
function detruit_restaurateur()
{
$r = spip_fetch_array(spip_query("SELECT COUNT(*) AS n FROM spip_auteurs"));
if ($r['n'] > 1)
spip_query("DELETE FROM spip_auteurs WHERE id_auteur=0");
else {
spip_query("UPDATE spip_auteurs SET id_auteur=extra WHERE id_auteur=0");
}
}
// http://doc.spip.org/@import_tables
function import_tables($request, $dir) {
global $import_ok, $abs_pos, $affiche_progression_pourcent;
$abs_pos = (!isset($GLOBALS['meta']["status_restauration"])) ? 0 :
$GLOBALS['meta']["status_restauration"];
// au premier appel destruction des tables a restaurer
// ou initialisation de la table des translations,
// mais pas lors d'une reprise.
if ($request['insertion']=='on') {
include_spip('inc/import_insere');
$request['init'] = (!$abs_pos) ? 'insere_1_init' : 'insere_1bis_init';
$request['boucle'] = 'import_insere';
} elseif ($request['insertion']=='passe2') {
$request['init'] = 'insere_2_init';
$request['boucle'] = 'import_translate';
} else {
$request['init'] = (!$abs_pos) ? 'import_init_tables' : 'import_table_choix';
$request['boucle'] = 'import_replace';
}
$archive = $dir . ($request['archive'] ? $request['archive'] : $request['archive_perso']);
if (ereg("\.gz$", $archive)) {
$size = false;
$taille = taille_en_octets($abs_pos);
$file = gzopen($archive, 'rb');
$gz = 'gzread';
} else {
$size = @filesize($archive);
$taille = floor(100 * $abs_pos / $size)." %";
$file = fopen($archive, 'rb');
$gz = 'fread';
}
if ($abs_pos==0) {
list($tag, $atts, $charset) = import_debut($file, $gz);
// improbable: fichier correct avant debut_admin et plus apres
if (!$tag) return !($import_ok = true);
$version_archive = import_init_meta($tag, $atts, $charset, $request);
} else {
$version_archive = $GLOBALS['meta']['version_archive_restauration'];
$atts = unserialize($GLOBALS['meta']['attributs_archive_restauration']);
spip_log("Reprise de l'importation interrompue en $abs_pos");
$_fseek = ($gz=='gzread') ? 'gzseek' : 'fseek';
$_fseek($file, $abs_pos);
}
$fimport = import_charge_version($version_archive);
import_affiche_javascript($taille);
if ($GLOBALS['flag_ob_flush']) ob_flush();
flush();
$oldtable ='';
$cpt = 0;
$pos = $abs_pos;
while ($table = $fimport($file, $request, $gz, $atts)) {
// memoriser pour pouvoir reprendre en cas d'interrupt,
// mais pas d'ecriture sur fichier, ca ralentit trop
ecrire_meta("status_restauration", "$abs_pos",'non');
if ($oldtable != $table) {
if ($oldtable) spip_log("$cpt entrees");
spip_log("Analyse de $table (commence en $pos)");
affiche_progression_javascript($abs_pos,$size,$table);
$oldtable = $table;
$cpt = 0;
$pos = $abs_pos;
}
$cpt++;
}
spip_log("$cpt entrees");
if (!$import_ok)
$res = _T('avis_archive_invalide') . ' ' .
_T('taille_octets', array('taille' => $pos)) ;
else {
$res = '';
affiche_progression_javascript('100 %', $size);
}
return $res ;
}
// http://doc.spip.org/@import_init_meta
function import_init_meta($tag, $atts, $charset, $request)
{
$version_archive = $atts['version_archive'];
ecrire_meta('attributs_archive_restauration', serialize($atts),'non');
ecrire_meta('version_archive_restauration', $version_archive,'non');
ecrire_meta('tag_archive_restauration', $tag,'non');
if ( $i = $request['insertion'])
ecrire_meta('charset_insertion', $charset,'non');
else ecrire_meta('charset_restauration', $charset,'non');
ecrire_metas();
spip_log("Debut de l'importation (charset: $charset, format: $version_archive)" . ($i ? " insertion $i" : ''));
return $version_archive;
}
// http://doc.spip.org/@import_affiche_javascript
function import_affiche_javascript($taille)
{
$max_time = ini_get('max_execution_time')*1000;
echo debut_boite_alerte(),
"<span style='color: black;' class='verdana1 spip_large'><b>", _T('info_base_restauration'), "</b></span>",
"<form name='progression'><div style='text-align: center'><input type='text' size='10' name='taille' value='",
$taille,
"'></div><div style='text-align: center'><input type='text' class='forml' name='recharge' value='",
_T('info_recharger_page'),
"'></div></form>",
fin_boite_alerte(),
"<script language=\"JavaScript\" type=\"text/javascript\">window.setTimeout('location.href=\"",
self(),
"\";',",
$max_time,
");</script>\n";
}
// http://doc.spip.org/@affiche_progression_javascript
function affiche_progression_javascript($abs_pos,$size, $table="") {
include_spip('inc/charsets');
echo "\n<script type='text/javascript'><!--\n";
if ($abs_pos == '100 %') {
if ($x = $GLOBALS['erreur_restauration'])
echo "document.progression.recharge.value='".str_replace("'", "\\'", unicode_to_javascript(_T('avis_erreur'))).": $x ';\n";
else
echo "document.progression.recharge.value='".str_replace("'", "\\'", unicode_to_javascript(_T('info_fini')))."';\n";
echo "document.progression.taille.value='$abs_pos';\n";
echo "window.setTimeout('location.href=\"".self()."\";',0);";
}
else {
if (trim($table))
echo "document.progression.recharge.value='$table';\n";
if (!$size)
$taille = ereg_replace(" ", " ", taille_en_octets($abs_pos));
else
$taille = floor(100 * $abs_pos / $size)." %";
echo "document.progression.taille.value='$taille';\n";
}
echo "\n--></script>\n";
if ($GLOBALS['flag_ob_flush']) ob_flush();
flush();
}
// http://doc.spip.org/@import_table_choix
function import_table_choix($request)
{
// construction de la liste des tables pour le dump :
// toutes les tables principales
// + toutes les tables auxiliaires hors relations
// + les tables relations dont les deux tables liees sont dans la liste
$tables_for_dump = array();
$tables_pointees = array();
global $IMPORT_tables_noimport;
global $tables_principales;
global $tables_auxiliaires;
global $table_des_tables;
global $tables_jointures;
// on construit un index des tables de liens
// pour les ajouter SI les deux tables qu'ils connectent sont sauvegardees
$tables_for_link = array();
foreach($tables_jointures as $table=>$liste_relations)
if (is_array($liste_relations))
{
$nom = $table;
if (!isset($tables_auxiliaires[$nom])&&!isset($tables_principales[$nom]))
$nom = "spip_$table";
if (isset($tables_auxiliaires[$nom])||isset($tables_principales[$nom])){
foreach($liste_relations as $link_table){
if (isset($tables_auxiliaires[$link_table])/*||isset($tables_principales[$link_table])*/){
$tables_for_link[$link_table][] = $nom;
}
else if (isset($tables_auxiliaires["spip_$link_table"])/*||isset($tables_principales["spip_$link_table"])*/){
$tables_for_link["spip_$link_table"][] = $nom;
}
}
}
}
$liste_tables = array_merge(array_keys($tables_principales),array_keys($tables_auxiliaires));
foreach($liste_tables as $table){
$name = preg_replace("{^spip_}","",$table);
if ( !isset($tables_pointees[$table])
&& !in_array($table,$IMPORT_tables_noimport)
&& !isset($tables_for_link[$table])){
$tables_for_dump[] = $table;
$tables_pointees[$table] = 1;
}
}
foreach ($tables_for_link as $link_table =>$liste){
$connecte = true;
foreach($liste as $connect_table)
if (!in_array($connect_table,$tables_for_dump))
$connecte = false;
if ($connecte)
# on ajoute les liaisons en premier
# si une restauration est interrompue, cela se verra mieux si il manque des objets
# que des liens
array_unshift($tables_for_dump,$link_table);
}
return $tables_for_dump;
}
?>