<?php
if (ini_get(‘short_open_tag’) == 0 && strtoupper(ini_get(‘short_open_tag’)) != ‘ON’)
die(‘Error: short_open_tag parameter must be turned on in php.ini’);
?><?
error_reporting(E_ALL & ~E_NOTICE & ~E_STRICT);
@ini_set(‘pcre.backtrack_limit’, 1024*1024);
define(‘IP_LIMIT_DEFAULT’, ‘#IP’.‘_LIMIT_PLACEHOLDER#’);
define(‘IP_LIMIT’, ‘#IP_LIMIT_PLACEHOLDER#’);
define(‘INIT_TIMESTAMP’, ‘#INIT_TIMESTAMP#’);
if (getenv(‘BITRIX_VA_VER’))
define(‘VMBITRIX’, ‘defined’);
if (version_compare(phpversion(),‘5.6’,‘<‘)) // 5.6 для старых версий Битрикс, без поддержки PHP 7
die(‘Error: PHP version 5.6 or higher is required’);
if(realpath(dirname(__FILE__)) != realpath($_SERVER[‘DOCUMENT_ROOT’]))
die(‘Error: this script must be started from Web Server’s DOCUMENT ROOT’);
if(isset($_SERVER[«BX_PERSONAL_ROOT«]) && $_SERVER[«BX_PERSONAL_ROOT«] <> «»)
define(«BX_PERSONAL_ROOT«, $_SERVER[«BX_PERSONAL_ROOT«]);
else
define(«BX_PERSONAL_ROOT«, «/bitrix«);
if(!defined(«START_EXEC_TIME«))
define(«START_EXEC_TIME«, microtime(true));
$START_TIME = START_EXEC_TIME;
define(«STEP_TIME«, defined(‘VMBITRIX’) ? 30 : 15);
# xdebug_start_trace();
define(‘RESTORE_FILE_LIST’, $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/tmp/restore.file_list.php’);
define(‘RESTORE_FILE_DIR’, $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/tmp/restore.removed’);
if (file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.txt’) && file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.php’))
unlink($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.txt’);
define(‘RESTORE_CLOUD_FILE_LIST’, file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.txt’) ? $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.txt’ : $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/restore_cloud.php’);
$strWarning = »;
if (function_exists(‘mb_internal_encoding’))
{
switch (ini_get(«mbstring.func_overload«))
{
case 0:
$bUTF_serv = false;
break;
case 2:
$bUTF_serv = mb_internal_encoding() == ‘UTF-8’;
break;
default:
die(‘PHP parameter mbstring.func_overload=’.ini_get(«mbstring.func_overload«).‘. The only supported values are 0 or 2.’);
break;
}
mb_internal_encoding(‘ISO-8859-1’);
}
else
$bUTF_serv = false;
if (!function_exists(‘htmlspecialcharsbx’))
{
function htmlspecialcharsbx($string, $flags = ENT_COMPAT)
{
//shitty function for php 5.4 where default encoding is UTF-8
return htmlspecialchars($string, $flags, «ISO-8859-1«);
}
}
define(‘DEBUG’, file_exists(dirname(__FILE__).‘/restore.debug’));
#@set_time_limit(0);
ob_start();
if (@preg_match(‘#ru#i’,$_SERVER[‘HTTP_ACCEPT_LANGUAGE’]))
$lang = ‘ru’;
elseif (@preg_match(‘#de#i’,$_SERVER[‘HTTP_ACCEPT_LANGUAGE’]))
$lang = ‘de’;
if ($_REQUEST[‘lang’])
$lang = $_REQUEST[‘lang’];
if (!in_array($lang,array(‘ru’,‘en’,‘de’)))
$lang = ‘en’;
define(«LANG«, $lang);
if (!headers_sent())
header(«Content-type:text/html; charset=utf-8«);
$dbconn = $_SERVER[‘DOCUMENT_ROOT’].»/bitrix/php_interface/dbconn.php«;
$arc_name = $_REQUEST[«arc_name«];
if (LANG == ‘ru’)
{
$MESS = array(
«BEGIN» => «
<p>
<ul>
<li>Перейдите в административную панель своего сайта на страницу <b>Настройки > Инструменты > Резервное копирование</b>
<li>Создайте полную резервную копию, которая будет включать <b>публичную часть</b>, <b>ядро</b> и <b>базу данных</b>
</ul>
<a href=’https://dev.1c-bitrix.ru/~4thxV’ target=’_blank’>Документация</a>
</p>
«,
«ARC_DOWN» => «Скачать резервную копию с дальнего сайта«,
«ARC_DOWN_BITRIXCLOUD» => «Развернуть резервную копию из облака "1С-Битрикс"«,
«BITRIXCLOUD_KEYS» => «Обновление ключей доступа к файлам в облаке«,
«LICENSE_KEY» => «Ваш лицензионный ключ:«,
«ARC_LOCAL_NAME» => «Имя архива:«,
«DB_SELECT» => «Выберите дамп БД:«,
«DB_SETTINGS» => «Параметры подключения к базе данных«,
«DB_SKIP» => «Пропустить восстановление базы«,
«SKIP» => «Пропустить«,
«DELETE_FILES» => «Удалить локальную резервную копию и служебные скрипты«,
«ARC_DOWN_URL» => «Ссылка на архив:«,
«TITLE0» => «Подготовка архива«,
«TITLE1» => «Загрузка резервной копии«,
«TITLE_PROCESS1» => «Распаковка архива«,
«FILE_IS_ENC» => «Архив зашифрован, для продолжения распаковки необходимо ввести пароль (с учетом регистра и пробелов): «,
«WRONG_PASS» => «Введенный пароль неверен«,
«ENC_KEY» => «Пароль: «,
«TITLE_PROCESS2» => «Выполняется восстановление базы данных«,
«TITLE2» => «Восстановление базы данных«,
«TITLE3» => «Загрузка файлов из облака«,
«ARC_SKIP» => «Архив уже распакован«,
«ARC_SKIP_DESC» => «переход к восстановлению базы данных«,
«ARC_NAME» => «Архив загружен в корневую папку сервера«,
«ARC_DOWN_PROCESS» => «Загружается:«,
«ERR_LOAD_FILE_LIST» => «Ошибочный ответ от сервиса 1С-Битрикс«,
«ARC_LOCAL» => «Загрузить с локального диска«,
«ARC_LOCAL_WARN» => «Загрузите все части многотомного архива«,
«ERR_NO_PARTS» => «Доступны не все части многотомного архива.<br>Общее число частей: «,
«BUT_TEXT1» => «Далее«,
«BUT_TEXT_BACK» => «Назад«,
«DUMP_RETRY» => «Попробовать снова«,
«USER_NAME» => «Имя пользователя«,
«USER_PASS» => «Пароль«,
«SEARCHING_UNUSED» => «Поиск посторонних файлов в ядре…«,
«BASE_NAME» => «Имя базы данных«,
«BASE_HOST» => «Сервер баз данных«,
«BASE_RESTORE» => «Восстановить«,
«ERR_EXTRACT» => «Ошибка«,
«ERR_MSG» => «Ошибка!«,
«LICENSE_NOT_FOUND» => «Лицензионный ключ не найден«,
«SELECT_ARC» => «Выберите архив«,
«CNT_PARTS» => «частей«,
«ARC_LIST_EMPTY» => «Нет резервных копий, связанных с этим ключом«,
«ERR_UNKNOWN» => «Неизвестный ответ сервера«,
«ERR_UPLOAD» => «Не удалось загрузить файл на сервер«,
«ERR_DUMP_RESTORE» => «Ошибка восстановления базы данных«,
«ERR_CREATE_DB» => «Ошибка создания базы«,
«ERR_TAR_TAR» => «Присутствуют файлы с расширением tar.tar. Вместо них должны быть архивы с номерами: tar.1, tar.2 и т.д.«,
«FINISH» => «Операция выполнена успешно«,
«FINISH_MSG» => «Операция восстановления системы завершена.«,
«FINISH_BTN» => «Перейти на сайт«,
«BASE_CREATE_DB» => «Создать базу данных если не существует«,
«BASE_CLOUDS» => «Файлы из облачных хранилищ:«,
«BASE_CLOUDS_Y» => «сохранить локально«,
«BASE_CLOUDS_N» => «оставить в облаке«,
«FINISH_ERR_DELL» => «Не удалось удалить все временные файлы! Обязательно удалите их вручную.«,
«FINISH_ERR_DELL_TITLE» => «Ошибка удаления файлов«,
«NO_READ_PERMS» => «Нет прав на чтение корневой папки сайта«,
«UTF8_ERROR1» => «Сайт работал в кодировке UTF-8. Конфигурация сервера не соответствует требованиям.<br>Для продолжения установите настройки PHP: mbstring.func_overload=2 и mbstring.internal_encoding=UTF-8.«,
«UTF8_ERROR2» => «Сайт работал в однобайтовой кодировке, а конфигурация сервера рассчитана на кодировку UTF-8.<br>Для продолжения установите настройки PHP: mbstring.func_overload=0 или mbstring.internal_encoding=ISO-8859-1.«,
«DOC_ROOT_WARN» => «Во избежание проблем с доступом был переписан путь к корню сайта в настройках сайтов. Проверьте настройки сайтов.«,
«CDN_WARN» => «Ускорение CDN было отключено т.к. текущий домен не соответствует домену из настроек CDN.«,
«HOSTS_WARN» => «Было отключено ограничение по доменам в модуле проактивной защиты т.к. текущий домен попадает под ограничения.«,
«WARN_CLEARED» => «При распаковке ядра были обнаружены файлы, которых не было в архиве. Эти файлы перенесены в /bitrix/tmp/restore.removed«,
«WARN_SITES» => «Вы распаковали многосайтовый архив, файлы дополнительных сайтов следует скопировать вручную из папки /bitrix/backup/sites«,
«WARNING» => «Внимание!«,
«DBCONN_WARN» => «Данные подключения взяты из dbconn.php. Если их не изменить, будет переписана база данных текущего сайта.«,
«HTACCESS_RENAMED_WARN» => «Файл .htaccess из архива был сохранен в корне сайта под именем .htaccess.restore, т.к. он может содержать директивы, недопустимые на данном сервере.«,
«HTACCESS_WARN» => «Файл .htaccess из архива был сохранен в корне сайта под именем .htaccess.restore, т.к. он может содержать директивы, недопустимые на данном сервере. В корне сайта создан .htaccess по умолчанию. Измените его вручную через FTP.«,
«HTACCESS_ERR_WARN» => «Файл .htaccess из архива был сохранен в корне сайта под именем .htaccess.restore, т.к. он может содержать директивы, недопустимые на данном сервере. <br> Не удалось создать корне сайта .htaccess по умолчанию. Переименуйте файл .htaccess.restore в .htaccess через FTP.«,
«ERR_CANT_DECODE» => «Невозможно восстановить архив т.к. он содержит файлы, имена которых нужно перекодировать, а модуль mbstring недоступен.«,
«ERR_CANT_DETECT_ENC» => «Невозможно восстановить архив т.к. он содержит файлы с именами в неизвестной кодировке:«,
‘TAR_ERR_FILE_OPEN’ => ‘Не удалось открыть файл: ‘,
«ARC_DOWN_OK» => «Все части архива загружены«,
«LOADER_SUBTITLE1» => «Загрузка резервной копии«,
«LOADER_SUBTITLE1_ERR» => «Ошибка загрузки«,
«LOADER_LOAD_QUERY_DISTR» => «Запрашиваю файл #DISTR#«,
«LOADER_LOAD_CONN2HOST» => «Подключение к серверу #HOST#«,
«LOADER_LOAD_NO_CONN2HOST» => «Не могу соединиться с #HOST#:«,
«LOADER_LOAD_SERVER_ANSWER» => «Ошибка загрузки. Сервер ответил: #ANS#«,
«LOADER_LOAD_SERVER_ANSWER1» => «Ошибка загрузки. У вас нет прав на доступ к этому файлу. Сервер ответил: #ANS#«,
«LOADER_LOAD_LOAD_DISTR» => «Загружаю файл #DISTR#«,
«LOADER_LOAD_ERR_RENAME» => «Не могу переименовать файл #FILE1# в файл #FILE2#«,
«ERROR_CANT_WRITE» => «Не могу записать файл #FILE#. Место на диске: #SPACE#«,
«ERROR_IP_CHANGED» => «IP адрес клиента изменился, продолжение невозможно.«,
«ERROR_INIT_TIMESTAMP» => «Время работы скрипта восстановления истекло. Загрузите новую версию.«,
«LOADER_LOAD_CANT_REDIRECT» => «Ошибочное перенаправление на адрес #URL#. Проверьте адрес для скачивания.«,
«LOADER_LOAD_CANT_OPEN_READ» => «Не могу открыть файл #FILE# на чтение«,
«LOADER_LOAD_LOADING» => «Загружаю файл, дождитесь окончания загрузки…«,
«LOADER_LOAD_FILE_SAVED» => «Файл сохранен: #FILE# [#SIZE# байт]«,
«UPDATE_SUCCESS» => «Обновлено успешно. <a href=’?’>Открыть</a>.«,
«LOADER_NEW_VERSION» => «Доступна новая версия скрипта восстановления, но загрузить её не удалось«,
);
}
elseif (LANG == ‘de’)
{
$MESS = array(
«BACK» => «Zurück«,
«BEGIN» => «
<p>
<ul>
<li>Öffnen Sie den Administrativen Bereich Ihrer alten Website und wählen Sie <b>Einstellungen > Tools > Backup</b>
<li>Erstellen Sie ein vollständiges Archiv mit <b>öffentlichen Website-Dateien</b>, <b>Kernel-Dateien</b> und <b>Datenbank-Dump</b>
</ul>
<b>Dokumentation:</b> <a href=’https://training.bitrix24.com/support/training/course/?COURSE_ID=12&LESSON_ID=5913&LESSON_PATH=3884.5489.5913′ target=’_blank’>Trainingskurs</a>
</p>
«,
«ARC_DOWN» => «Von Remote-Server herunterladen«,
«ARC_DOWN_BITRIXCLOUD» => «Backup aus der Bitrix Cloud wiederherstellen«,
«BITRIXCLOUD_KEYS» => «Refreshing Bitrix Cloud access keys«,
«LICENSE_KEY» => «Ihr Lizenzschlüssel:«,
«ARC_LOCAL_NAME» => «Archivname:«,
«DB_SELECT» => «Datenbank-Dump auswählen:«,
«DB_SETTINGS» => «Datenbank-Einstellungen«,
«DB_SKIP» => «Überspringen«,
«SKIP» => «Überspringen«,
«DELETE_FILES» => «Archiv und temporäre Scripts löschen«,
«ARC_DOWN_URL» => «Archiv-URL:«,
«TITLE0» => «Archiv erstellen«,
«TITLE1» => «Archiv herunterladen«,
«TITLE_PROCESS1» => «Archiv wird entpackt«,
«TITLE_PROCESS2» => «Datenbank wird wiederhergestellt…«,
«FILE_IS_ENC» => «Archiv ist verschlüsselt. Passwort eingeben: «,
«WRONG_PASS» => «Passwort ist falsch«,
«ENC_KEY» => «Passwort: «,
«TITLE2» => «Datenbank wiederherstellen«,
«TITLE3» => «Herunterladen von cloud-Dateien«,
«ARC_SKIP» => «Archiv wurde bereits entpackt«,
«ARC_SKIP_DESC» => «Wiederherstellung der Datenbank starten«,
«ARC_NAME» => «Archiv ist im Dokumenten-Root abgespeichert«,
«ARC_DOWN_PROCESS» => «Wird herunterladen:«,
«ERR_LOAD_FILE_LIST» => «Falsche Antwort vom Bitrixsoft Server«,
«ARC_LOCAL» => «Vom lokalen Speicher hochladen«,
«ARC_LOCAL_WARN» => «Vergessen Sie nicht, alle Teile eines mehrbändigen Archivs hochzuladen.«,
«ERR_NO_PARTS» => «Einige Teile des mehrbändigen Archivs fehlen.<br>Teile gesamt: «,
«BUT_TEXT1» => «Fortfahren«,
«BUT_TEXT_BACK» => «Zurück«,
«DUMP_RETRY» => «Wiederholen«,
«USER_NAME» => «Name des datenbank-Nutzers«,
«USER_PASS» => «Passwort«,
«BASE_NAME» => «Datenbankname«,
«SEARCHING_UNUSED» => «Ungenutzte Kernel-Dateien werden gesucht…«,
«BASE_HOST» => «Datenbank-Host«,
«BASE_RESTORE» => «Wiederherstellen«,
«ERR_EXTRACT» => «Fehler«,
«ERR_MSG» => «Fehler!«,
«LICENSE_NOT_FOUND» => «Lizenz wurde nicht gefunden«,
«SELECT_ARC» => «Backup auswählen«,
«CNT_PARTS» => «Teile«,
«ARC_LIST_EMPTY» => «Backup-Liste ist leer für den aktuellen Lizenzschlüssel«,
«ERR_UNKNOWN» => «Unbekannte Server-Antwort«,
«ERR_UPLOAD» => «Datei kann nicht hochgeladen werden«,
«ERR_DUMP_RESTORE» => «Fehler bei Wiederherstellung der Datenbank:«,
«ERR_CREATE_DB» => «Fehler bei Erstellung der Datenbank«,
«ERR_TAR_TAR» => «Es gibt Dateien mit Erweiterung tar.tar. Es müssten tar.1, tar.2 und so weiter sein«,
«FINISH» => «Erfolgreich abgeschlossen«,
«FINISH_MSG» => «Wiederherstellung des Systems wurde abgeschlossen.«,
«FINISH_BTN» => «Website öffnen«,
«BASE_CREATE_DB» => «Datenbank erstellen«,
«BASE_CLOUDS» => «Cloud-Dateien:«,
«BASE_CLOUDS_Y» => «lokal speichern«,
«BASE_CLOUDS_N» => «in der Cloud lassen«,
«FINISH_ERR_DELL» => «Löschen von temporären Dateien ist fehlgeschlagen. Sie sollten diese manuell löschen«,
«FINISH_ERR_DELL_TITLE» => «Fehler beim Löschen von dateien«,
«NO_READ_PERMS» => «Sie haben nicht genügend Rechte, um Web-Server Root zu lesen«,
«UTF8_ERROR1» => «Ihr Server ist für die Codierung UTF-8 nicht konfiguriert. Definieren Sie bitte mbstring.func_overload=2 und mbstring.internal_encoding=UTF-8 um fortzufahren.«,
«UTF8_ERROR2» => «Ihr Server ist für die Codierung UTF-8 konfiguriert. Definieren Sie bitte mbstring.func_overload=0 oder mbstring.internal_encoding=ISO-8859-1 um fortzufahren.«,
«DOC_ROOT_WARN» => «Um Probleme mit Zugriffsrechten zu vermeiden, wurde das Dokumenten-Root in den Einstellungen der Website aufgeräumt.«,
«CDN_WARN» => «CDN Web-Accelerator wurde deaktiviert, weil die aktuelle Domain sich von der unterscheidet, die in den CDN-Einstellungen gespeichrt ist.«,
«HOSTS_WARN» => «Domain-Einschränkung wurde deaktiviert (Sicherheitsmodul), weil die aktuelle Domain den Einstellungen nicht entspricht.«,
«WARN_CLEARED» => «Einige Dateien wurden in /bitrix gefunden, sie sind in der Backup nicht enthalten. Sie wurden nach /bitrix/tmp/restore.removed verschoben«,
«WARN_SITES» => «Sie haben das Multisite-Archiv entpackt, kopieren Sie bitte die Dateien zusätzlicher Websites von /bitrix/backup/sites in einen entsprechenden Ort«,
«WARNING» => «Warnung!«,
«DBCONN_WARN» => «Die Verbindungseinstellungen werden von dbconn.php gelesen. Wenn Sie sie nicht ändern, wird aktuelle Datenbank überschrieben.«,
«HTACCESS_RENAMED_WARN» => «Die Datei .htaccess wurde unter .htaccess.restore gespeichert, weil sie Anweisungen enthalten kann, die auf diesem Server nicht erlaubt sind.«,
«HTACCESS_WARN» => «Die Datei .htaccess wurde unter .htaccess.restore gespeichert, weil sie die Anweisungen enthalten kann, die auf diesem Server nicht erlaubt sind. Standarddatei .htaccess wurde im Dokumenten-Root erstellt. Sie sollten sie manuell via FTP aktualisieren.«,
«HTACCESS_ERR_WARN» => «Die Datei .htaccess wurde unter .htaccess.restore gespeichert, weil sie Anweisungen enthalten kann, die auf diesem Server nicht erlaubt sind. Es gab einen Fehler beim Erstellen der Standarddatei .htaccess. Sie sollten .htaccess.restore in .htaccess via FTP umbenennen.«,
«ERR_CANT_DECODE» => «Unmöglich fortzufahren, weil das Modul MBString nicht verfügbar ist.«,
«ERR_CANT_DETECT_ENC» => «Unmöglich fortzufahren wegen eines Fehlers in der Erkennung der Codierung des Dateinamen: «,
‘TAR_ERR_FILE_OPEN’ => ‘Datei kann nicht geöffnet werden: ‘,
«ARC_DOWN_OK» => «Alle Archivteile wurden heruntergeladen«,
«LOADER_SUBTITLE1» => «Wird geladen«,
«LOADER_SUBTITLE1_ERR» => «Fehler beim Laden«,
«LOADER_LOAD_QUERY_DISTR» => «Paket #DISTR# wird angefragt«,
«LOADER_LOAD_CONN2HOST» => «Verbindung mit #HOST#«,
«LOADER_LOAD_NO_CONN2HOST» => «Keine Verbindung mit #HOST#:«,
«LOADER_LOAD_SERVER_ANSWER» => «Fehler beim Herunterladen. Die Antwort vom Server war: #ANS#«,
«LOADER_LOAD_SERVER_ANSWER1» => «Fehler beim Herunterladen. Sie können dieses Paket nicht herunterladen. Die Antwort vom Server war: #ANS#«,
«LOADER_LOAD_LOAD_DISTR» => «Das Paket #DISTR# wird heruntergeladen«,
«LOADER_LOAD_ERR_RENAME» => «Die Datei #FILE1# kann nicht in #FILE2# umbenannt werden«,
«ERROR_CANT_WRITE» => «In der Datei #FILE# kann nicht geschrieben werden. Freier Speicherplatz: #SPACE#«,
«ERROR_IP_CHANGED» => «IP address has changed. Permission denied.«,
«ERROR_INIT_TIMESTAMP» => «This script is outdated. Please upload the new version.«,
«LOADER_LOAD_CANT_REDIRECT» => «Inkorrekte Weiterleitung an #URL#. Überprüfen Sie die Download-URL.«,
«LOADER_LOAD_CANT_OPEN_READ» => «Die Datei #FILE# kann nicht zum Lesen geöffnet werden«,
«LOADER_LOAD_LOADING» => «Es wird nun heruntergeladen. Bitte warten…«,
«LOADER_LOAD_FILE_SAVED» => «Datei gespeichert: #FILE# [#SIZE# bytes]«,
«UPDATE_SUCCESS» => «Aktualisierung war erfolgreich. <a href=’?’>Öffnen</a>.«,
«LOADER_NEW_VERSION» => «Beim Aktualisieren des Scripts restore.php ist ein Fehler aufgetreten.«,
);
}
else
{
$MESS = array(
«BEGIN» => «
<p>
<ul>
<li>Open Control Panel section of your old site and select <b>Settings > Tools > Backup</b>
<li>Create full archive which contains <b>public site files</b>, <b>kernel files</b> and <b>database dump</b>
</ul>
<b>Documentation:</b> <a href=’https://training.bitrix24.com/support/training/course/?COURSE_ID=12&LESSON_ID=5913&LESSON_PATH=3884.5489.5913′ target=’_blank’>learning course</a>
</p>
«,
«ARC_DOWN» => «Download from remote server«,
«ARC_DOWN_BITRIXCLOUD» => «Restore the backup from the Bitrix Cloud«,
«BITRIXCLOUD_KEYS» => «Refreshing Bitrix Cloud access keys«,
«LICENSE_KEY» => «Your license key:«,
«ARC_LOCAL_NAME» => «Archive name:«,
«DB_SELECT» => «Select Database Dump:«,
«DB_SETTINGS» => «Database settings«,
«DB_SKIP» => «Skip«,
«SKIP» => «Skip«,
«DELETE_FILES» => «Delete archive and temporary scripts«,
«ARC_DOWN_URL» => «Archive URL:«,
«TITLE0» => «Archive Creation«,
«TITLE1» => «Archive download«,
«TITLE_PROCESS1» => «Extracting an archive«,
«TITLE_PROCESS2» => «Restoring database…«,
«FILE_IS_ENC» => «Archive is encrypted. Enter password: «,
«WRONG_PASS» => «Wrong password«,
«ENC_KEY» => «Password: «,
«TITLE2» => «Database restore«,
«TITLE3» => «Downloading cloud files«,
«ARC_SKIP» => «Archive is already extracted«,
«ARC_SKIP_DESC» => «Starting database restore«,
«ARC_NAME» => «Archive is stored in document root folder«,
«ARC_DOWN_PROCESS» => «Downloading:«,
«ERR_LOAD_FILE_LIST» => «Wrong Bitrixsoft server response«,
«ARC_LOCAL» => «Upload from local disk«,
«ARC_LOCAL_WARN» => «Don’t forget to upload all the parts of multi-volume archive.«,
«ERR_NO_PARTS» => «Some parts of the multivolume archive are missed.<br>Total number of parts: «,
«BUT_TEXT1» => «Continue«,
«BUT_TEXT_BACK» => «Back«,
«DUMP_RETRY» => «Retry«,
«USER_NAME» => «Database User Name«,
«USER_PASS» => «Password«,
«BASE_NAME» => «Database Name«,
«SEARCHING_UNUSED» => «Searching for unused kernel files…«,
«BASE_HOST» => «Database Host«,
«BASE_RESTORE» => «Restore«,
«ERR_EXTRACT» => «Error«,
«ERR_MSG» => «Error!«,
«LICENSE_NOT_FOUND» => «License not found«,
«SELECT_ARC» => «Select backup«,
«CNT_PARTS» => «parts«,
«ARC_LIST_EMPTY» => «Backup list is empty for current license key«,
«ERR_UNKNOWN» => «Unknown server response«,
«ERR_UPLOAD» => «Unable to upload file«,
«ERR_DUMP_RESTORE» => «Error restoring the database:«,
«ERR_CREATE_DB» => «Error creating the database«,
«ERR_TAR_TAR» => «There are files with tar.tar extension presents. Should be tar.1, tar.2 and so on«,
«FINISH» => «Successfully completed«,
«FINISH_MSG» => «Restoring of the system was completed.«,
«FINISH_BTN» => «Open site«,
«BASE_CREATE_DB» => «Create database«,
«BASE_CLOUDS» => «Cloud files:«,
«BASE_CLOUDS_Y» => «store locally«,
«BASE_CLOUDS_N» => «leave in the cloud«,
«FINISH_ERR_DELL» => «Failed to delete temporary files! You should delete them manually«,
«FINISH_ERR_DELL_TITLE» => «Error deleting the files«,
«NO_READ_PERMS» => «No permissions for reading Web Server root«,
«UTF8_ERROR1» => «Your server is not configured for UTF-8 encoding. Please set mbstring.func_overload=2 and mbstring.internal_encoding=UTF-8 to continue.«,
«UTF8_ERROR2» => «Your server is configured for UTF-8 encoding. Please set mbstring.func_overload=0 or mbstring.internal_encoding=ISO-8859-1 to continue.«,
«DOC_ROOT_WARN» => «To prevent access problems the document root has been cleared in the site settings.«,
«CDN_WARN» => «CDN Web Accelerator has been disabled because current domain differs from the one stored in CDN settings.«,
«HOSTS_WARN» => «Domain restriction has beed disabled (security module) because current domain doesn’t correspond settings.«,
«WARN_CLEARED» => «Some files were found in /bitrix which don’t present in the backup. They were moved to /bitrix/tmp/restore.removed«,
«WARN_SITES» => «You have extracted the multisite archive, please copy files of additional sites from /bitrix/backup/sites to an appropriate place«,
«WARNING» => «Warning!«,
«DBCONN_WARN» => «The connection settings are read from dbconn.php. If you don’t change them, current database will be overwriten.«,
«HTACCESS_RENAMED_WARN» => «The file .htaccess was saved as .htaccess.restore, because it may contain directives which are not permitted on this server.«,
«HTACCESS_WARN» => «The file .htaccess was saved as .htaccess.restore, because it may contain directives which are not permitted on this server. Default .htaccess file has been created at the document root. Please modify it manually using FTP.«,
«HTACCESS_ERR_WARN» => «The file .htaccess was saved as .htaccess.restore, because it may contain directives which are not permitted on this server. There was an error in creating default .htaccess file. Please rename .htaccess.restore to .htaccess using FTP.«,
«ERR_CANT_DECODE» => «Unable to continue because the module MBString is not available.«,
«ERR_CANT_DETECT_ENC» => «Unable to continue due to error in encoding detection of file name: «,
‘TAR_ERR_FILE_OPEN’ => ‘Can’t open file: ‘,
«ARC_DOWN_OK» => «All archive parts have been downloaded«,
«LOADER_SUBTITLE1» => «Loading«,
«LOADER_SUBTITLE1_ERR» => «Loading Error«,
«LOADER_MENU_UNPACK» => «Unpack file«,
«LOADER_LOAD_QUERY_DISTR» => «Requesting package #DISTR#«,
«LOADER_LOAD_CONN2HOST» => «Connection to #HOST#…«,
«LOADER_LOAD_NO_CONN2HOST» => «Cannot connect to #HOST#:«,
«LOADER_LOAD_SERVER_ANSWER» => «Error while downloading. Server reply was: #ANS#«,
«LOADER_LOAD_SERVER_ANSWER1» => «Error while downloading. Your can not download this package. Server reply was: #ANS#«,
«LOADER_LOAD_LOAD_DISTR» => «Downloading package #DISTR#«,
«LOADER_LOAD_ERR_RENAME» => «Cannot rename file #FILE1# to #FILE2#«,
«ERROR_CANT_WRITE» => «Cannot write to file #FILE#. Free disk space: #SPACE#«,
«ERROR_IP_CHANGED» => «IP address has changed. Permission denied.«,
«ERROR_INIT_TIMESTAMP» => «This script is outdated. Please upload the new version.«,
«LOADER_LOAD_CANT_REDIRECT» => «Wrong redirect to #URL#. Check download url.«,
«LOADER_LOAD_CANT_OPEN_READ» => «Cannot open file #FILE# for reading«,
«LOADER_LOAD_LOADING» => «Download in progress. Please wait…«,
«LOADER_LOAD_FILE_SAVED» => «File saved: #FILE# [#SIZE# bytes]«,
«UPDATE_SUCCESS» => «Successful update. <a href=’?’>Open</a>.«,
«LOADER_NEW_VERSION» => «Error occured while updating restore.php script!«,
);
}
$bClearUnusedStep = (bool) $_REQUEST[‘clear’];
$bSelectDumpStep = $_REQUEST[‘source’] == ‘dump’;
$bCloudDownloadStep = $_REQUEST[‘cloud_download’];
$Step = IntVal($_REQUEST[«Step«]);
$strErrMsg = »;
if (!DEBUG && !$Step && $_SERVER[‘REQUEST_METHOD’] == ‘GET’)
{
$this_script_name = basename(__FILE__);
$bx_host = ‘www.1c-bitrix.ru’;
$bx_url = ‘/download/files/scripts/’.$this_script_name;
$form = »;
// Check for updates
$res = fsockopen(‘ssl://’.$bx_host, 443, $errno, $errstr, 3);
if($res)
{
$strRequest = «HEAD «.$bx_url.» HTTP/1.1rn»;
$strRequest.= «Host: «.$bx_host.»rn»;
$strRequest.= «rn»;
fputs($res, $strRequest);
while ($line = fgets($res, 4096))
{
if (preg_match(«/Content-Length: *([0-9]+)/i«, $line, $regs))
{
if (filesize(__FILE__) != trim($regs[1]))
{
$tmp_name = $this_script_name.‘.tmp’;
if (LoadFile(‘https://’.$bx_host.$bx_url, $tmp_name))
{
if (rename($_SERVER[‘DOCUMENT_ROOT’].‘/’.$tmp_name,__FILE__))
{
bx_accelerator_reset();
echo ‘<script>document.location=»?lang=’.LANG.‘»;</script>’.getMsg(‘UPDATE_SUCCESS’);
die();
}
else
$strErrMsg = getMsg(«ERROR_CANT_WRITE«, [«#FILE#» => $this_script_name, ‘#SPACE#’ => freeSpace()]);
}
else
$strErrMsg = getMsg(‘LOADER_NEW_VERSION’);
}
break;
}
}
fclose($res);
}
}
@ini_set(‘pcre.backtrack_limit’, 1024*1024);
if (!DEBUG)
{
if (IP_LIMIT == IP_LIMIT_DEFAULT)
{
($str = file_get_contents(__FILE__)) || die(getMsg(«LOADER_LOAD_CANT_OPEN_READ«, [«#FILE#» => __FILE__]));
$size = strlen($str);
$str = str_replace(IP_LIMIT, $_SERVER[‘REMOTE_ADDR’], $str);
$str = str_replace(INIT_TIMESTAMP, time(), $str);
$str = str_pad($str, $size, ‘ ‘);
file_put_contents(__FILE__, $str) || die(getMsg(«ERROR_CANT_WRITE«, [«#FILE#» => __FILE__, ‘#SPACE#’ => freeSpace()]));
bx_accelerator_reset();
header(‘Location: ‘.$_SERVER[‘REQUEST_URI’]);
die();
}
elseif ($_SERVER[‘REMOTE_ADDR’] != IP_LIMIT)
$strErrMsg = getMsg(‘ERROR_IP_CHANGED’);
elseif (time() — INIT_TIMESTAMP > 86400)
$strErrMsg = getMsg(‘ERROR_INIT_TIMESTAMP’);
if ($strErrMsg)
{
echo ‘<div style=»color:red»>’.getMsg(‘ERR_MSG’).‘ ‘.$strErrMsg.‘</div>’;
die();
}
}
if ($_REQUEST[‘LoadFileList’])
{
$strLog = »;
if (LoadFile(«https://www.1c-bitrix.ru/buy_tmp/backup.php?license=«.md5(trim($_REQUEST[‘license_key’])).»&lang=«.LANG.»&action=get_info«, $file = $_SERVER[‘DOCUMENT_ROOT’].‘/file_list.xml’) && ($str = file_get_contents($file)))
{
if (preg_match_all(‘/<file name=»([^»]+)» size=»([^»]+)».*?/>/’, $str, $regs))
{
$arFiles = array();
$arParts = array();
foreach($regs[0] as $i => $wholeMatch)
{
$name = CTar::getFirstName($regs[1][$i]);
$arFiles[$name] += $regs[2][$i];
$arParts[$name]++;
}
krsort($arFiles);
echo getMsg(‘SELECT_ARC’).‘: <select name=»bitrixcloud_backup»>’;
foreach($arFiles as $name => $size)
echo ‘<option value=»‘.htmlspecialcharsbx($name).‘» ‘.($_REQUEST[‘bitrixcloud_backup’] == $name ? ‘selected’ : »).‘>’.htmlspecialcharsbx($name).‘ (‘.floor($size/1024/1024), ‘ Mb’.($arParts[$name] > 1 ? ‘, ‘.getMsg(‘CNT_PARTS’).‘: ‘.$arParts[$name] : »).‘)</option>’;
echo ‘</select><br>’;
echo getMsg(‘ENC_KEY’).‘ <input type=»password» size=30 name=»EncryptKey» autocomplete=»off»>’;
}
else
{
if (strpos($str, ‘<files>’) !== false) // valid answer
$strErrMsg = getMsg(‘ARC_LIST_EMPTY’);
elseif (preg_match(‘#error#i’,$str))
{
$code = strip_tags($str);
if ($code == ‘LICENSE_NOT_FOUND’)
$strErrMsg = getMsg(‘LICENSE_NOT_FOUND’);
else
$strErrMsg = $code;
}
else
$strErrMsg = getMsg(‘ERR_UNKNOWN’);
echo ‘<div style=»color:red»>’.getMsg(‘ERR_MSG’).‘ ‘.$strErrMsg.‘</div>’;
}
bx_unlink($file);
}
else
echo ‘<div style=»color:red»>’.getMsg(‘ERR_LOAD_FILE_LIST’).‘</div><div style=»text-align:left;color:#CCC»>’.nl2br($strLog).‘</div>’;
die();
}
elseif ($Step == 2 && !$bSelectDumpStep)
{
if (is_array($_REQUEST[‘arHeaders’]))
$arHeaders = $_REQUEST[‘arHeaders’];
else
$arHeaders = array();
$source = $_REQUEST[‘source’];
if ($source == ‘bitrixcloud’ && !$_REQUEST[‘arc_down_url’])
{
$strLog = »;
if (LoadFile(‘https://www.1c-bitrix.ru/buy_tmp/backup.php?license=’.md5(trim($_REQUEST[‘license_key’])).‘&lang=’.LANG.‘&action=read_file&file_name=’.urlencode($_REQUEST[‘bitrixcloud_backup’]).‘&check_word=’.CTar::getCheckword($_REQUEST[‘EncryptKey’]), $file = $_SERVER[‘DOCUMENT_ROOT’].‘/file_info.xml’) && ($str = file_get_contents($file)))
{
bx_unlink($file);
$host = preg_match(‘#<host>([^<]+)</host>#i’,$str,$regs) ? $regs[1] : false;
$path = preg_match(‘#<path>([^<]+)</path>#i’,$str,$regs) ? $regs[1] : false;
if (preg_match_all(‘/<header name=»([^»]+)» value=»([^»]+)».*?/>/’, $str, $regs))
{
foreach($regs[0] as $i => $wholeMatch)
$arHeaders[$regs[1][$i]] = $regs[2][$i];
}
if ($host && $path)
{
$_REQUEST[‘arc_down_url’] = $host.$path;
}
elseif (strpos($str, ‘WRONG_FILE_NAME_OR_CHECKWORD’) !== false)
$strErrMsg = ‘<div style=»color:red»>’.getMsg(‘WRONG_PASS’).‘</div>’;
else
$strErrMsg = ‘<div style=»color:red»>’.getMsg(‘ERR_LOAD_FILE_LIST’).‘</div>’;
}
else
$strErrMsg = ‘<div style=»color:red»>’.getMsg(‘ERR_LOAD_FILE_LIST’).‘</div><div style=»text-align:left;color:#CCC»>’.nl2br($strLog).‘</div>’;
if (!$_REQUEST[‘try_next’] && $strErrMsg)
{
$text = $strErrMsg.
getMsg(‘ENC_KEY’).‘<input type=»password» size=30 name=»EncryptKey» autocomplete=»off» value=»»>’.
‘<input type=»hidden» name=»license_key» value=»‘.htmlspecialcharsbx($_REQUEST[‘license_key’]).‘»>’.
‘<input type=»hidden» name=»source» value=»bitrixcloud»>’.
‘<input type=»hidden» name=»bitrixcloud_backup» value=»‘.htmlspecialcharsbx($_REQUEST[‘bitrixcloud_backup’]).‘»>’.
‘<input type=»hidden» name=»Step» value=»2″>’;
$bottom .= ‘
<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>
<input type=»button» id=»start_button» value=»‘.getMsg(«BUT_TEXT1«).‘» onClick=»reloadPage(2)»>’;
showMsg(getMsg(‘TITLE1’),$text,$bottom);
die();
}
}
if ($source == ‘download’ || $source == ‘bitrixcloud’)
{
$strUrl = $_REQUEST[‘arc_down_url’];
$res = false;
if ($strUrl)
{
if (!preg_match(‘#https?://#’,$strUrl))
$strUrl = ‘http://’.$strUrl;
$arc_name = trim(basename($strUrl));
if ($arc_name == CTar::getFirstName($arc_name))
{
$full = false;
$tar = new CTar();
$n = CTar::getLastNum($_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name);
$arc_name1 = $arc_name;
while(file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name1))
{
if ($n && $arc_name1 == $arc_name.‘.’.$n)
{
$full = true;
break;
}
$arc_name1 = $tar->getNextName($arc_name1);
}
if (!$full)
{
$strUrl = str_replace($arc_name, $arc_name1, $strUrl);
$_REQUEST[‘bitrixcloud_backup’] = $arc_name = $arc_name1;
}
}
$strLog = »;
$status = »;
if ($_REQUEST[‘continue’])
{
$res = LoadFile($strUrl, $_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name, $arHeaders);
if (file_exists($file = $_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name))
{
if ($res == 1)
{
$f = fopen($_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name, ‘rb’);
$id = fread($f, 2);
fclose($f);
if ($id != chr(31).chr(139)) // not gzip
{
$s = filesize($_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name);
if ($s == 0 || $s%512 > 0) // not tar
{
bx_unlink($_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name);
$res = false;
}
}
}
}
}
else
{
$res = 2;
SetCurrentProgress(0);
}
}
if ($res)
{
$text = getMsg(‘ARC_DOWN_PROCESS’).‘ <b>’.htmlspecialcharsbx($arc_name).‘</b>’ . $status .
‘<input type=hidden name=Step value=2>’.
‘<input type=hidden name=continue value=Y>’.
‘<input type=hidden name=»EncryptKey» value=»‘.htmlspecialcharsbx($_REQUEST[‘EncryptKey’]).‘»>’.
‘<input type=hidden name=»license_key» value=»‘.htmlspecialcharsbx($_REQUEST[‘license_key’]).‘»>’;
if ($res == 2)
{
$text .= ‘<input type=hidden name=arc_down_url value=»‘.htmlspecialcharsbx($strUrl).‘»>’;
$text .= ‘<input type=hidden name=source value=download>’;
$text .= ‘<input type=hidden name=»bitrixcloud_backup» value=»‘.htmlspecialcharsbx($_REQUEST[‘bitrixcloud_backup’]).‘»>’;
}
else
{
$tar = new CTar();
$text .= ‘<input type=hidden name=try_next value=Y>’;
if (count($arHeaders)) // bitrixcloud
{
$text .= ‘<input type=hidden name=source value=bitrixcloud>’;
$text .= ‘<input type=hidden name=»bitrixcloud_backup» value=»‘.htmlspecialcharsbx($tar->getNextName($_REQUEST[‘bitrixcloud_backup’])).‘»>’;
}
else
{
$text .= ‘<input type=hidden name=source value=download>’;
$text .= ‘<input type=hidden name=arc_down_url value=»‘.htmlspecialcharsbx($tar->getNextName($strUrl)).‘»>’;
}
}
if (count($arHeaders))
{
foreach($arHeaders as $k=>$v)
$text .= ‘<input type=hidden name=»arHeaders[‘.htmlspecialcharsbx($k).‘]» value=»‘.htmlspecialcharsbx($v).‘»>’;
}
}
elseif ($_REQUEST[‘try_next’])
{
$text = getMsg(‘ARC_DOWN_OK’).
‘<input type=hidden name=Step value=2>’.
‘<input type=hidden name=»EncryptKey» value=»‘.htmlspecialcharsbx($_REQUEST[‘EncryptKey’]).‘»>’;
if ($_REQUEST[‘bitrixcloud_backup’])
$text .= ‘<input type=hidden name=arc_name value=»‘.htmlspecialcharsbx(CTar::getFirstName($_REQUEST[‘bitrixcloud_backup’])).‘»>’;
else
$text .= ‘<input type=hidden name=arc_name value=»‘.htmlspecialcharsbx(CTar::getFirstName($arc_name)).‘»>’;
}
else
{
if ($_REQUEST[‘bitrixcloud_backup’] && $replycode == 403 && count($arHeaders) && (!$_REQUEST[‘timestamp’] || time() — $_REQUEST[‘timestamp’] > 10)) // Retry for bitrixcloud
{
$status = ‘<div>’.getMsg(‘BITRIXCLOUD_KEYS’).‘</div>’;
$text = getMsg(‘ARC_DOWN_PROCESS’).‘ <b>’.htmlspecialcharsbx($arc_name).‘</b>’ . $status .
‘<input type=hidden name=Step value=2>’.
‘<input type=hidden name=timestamp value=’.time().‘>’.
‘<input type=hidden name=continue value=Y>’.
‘<input type=hidden name=»EncryptKey» value=»‘.htmlspecialcharsbx($_REQUEST[‘EncryptKey’]).‘»>’.
‘<input type=hidden name=»license_key» value=»‘.htmlspecialcharsbx($_REQUEST[‘license_key’]).‘»>’;
$text .= ‘<input type=hidden name=source value=bitrixcloud>’;
$text .= ‘<input type=hidden name=»bitrixcloud_backup» value=»‘.htmlspecialcharsbx($_REQUEST[‘bitrixcloud_backup’]).‘»>’;
}
else
{
$ar = array(
‘TITLE’ => getMsg(‘LOADER_SUBTITLE1_ERR’),
‘TEXT’ => nl2br($strLog),
‘BOTTOM’ => ‘<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>’
);
html($ar);
die();
}
}
$bottom = ‘<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>’;
showMsg(getMsg(‘LOADER_SUBTITLE1’),$text,$bottom);
?><script>reloadPage(2, 1);</script><?
die();
}
elseif($source == ‘upload’)
{
if (!count($_FILES[‘archive’][‘tmp_name’]))
{
$ar = array(
‘TITLE’ => getMsg(‘ERR_EXTRACT’),
‘TEXT’ => getMsg(‘ERR_UPLOAD’),
‘BOTTOM’ => ‘<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>’
);
html($ar);
die();
}
foreach($_FILES[‘archive’][‘tmp_name’] as $k => $v)
{
if (!$v)
continue;
$arc_name = $_FILES[‘archive’][‘name’][$k];
if (!@move_uploaded_file($v, $_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name))
{
$ar = array(
‘TITLE’ => getMsg(‘ERR_EXTRACT’),
‘TEXT’ => getMsg(‘ERR_UPLOAD’),
‘BOTTOM’ => ‘<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>’
);
html($ar);
die();
}
}
$text =
‘<input type=hidden name=Step value=2>’.
‘<input type=hidden name=arc_name value=»‘.htmlspecialcharsbx(CTar::getFirstName($arc_name)).‘»>’;
showMsg(getMsg(‘LOADER_SUBTITLE1’),$text);
?><script>reloadPage(2, 1);</script><?
die();
}
}
elseif($Step == 3)
{
$d_pos = (double) $_REQUEST[«d_pos«];
if ($d_pos < 0)
$d_pos = 0;
$oDB = new CDBRestore($_REQUEST[«DBHost«], $_REQUEST[«DBName«], $_REQUEST[«DBLogin«], $_REQUEST[«DBPassword«], $_REQUEST[«dump_name«], $d_pos);
$oDB->LocalCloud = $_REQUEST[‘LocalCloud’];
if(!$oDB->Connect())
{
$strErrMsg = $oDB->getError();
$Step = 2;
$bSelectDumpStep = true;
}
}
##################################### GUI #################################
if(!$Step)
{
$ar = array(
‘TITLE’ => getMsg(«TITLE0«),
‘TEXT’ =>
($strErrMsg ? ‘<div style=»color:red;padding:10px;border:1px solid red»>’.$strErrMsg.‘</div>’ : »).
getMsg(‘BEGIN’),
‘BOTTOM’ =>
(defined(‘VMBITRIX’) ? ‘<a href=»/?lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘ : »).
‘<input type=»button» value=»‘.getMsg(«BUT_TEXT1«).‘» onClick=»reloadPage(1)»>’
);
html($ar);
}
elseif($Step == 1)
{
$arc_down_url = $_REQUEST[‘arc_down_url’] ? $_REQUEST[‘arc_down_url’] : »;
$local_arc_name = htmlspecialcharsbx(ltrim($_REQUEST[‘local_arc_name’],‘/’));
if ($_REQUEST[‘bitrixcloud_backup’])
{
@include($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/license_key.php’);
$license_key = $LICENSE_KEY;
}
$option = getArcList();
$ar = array(
‘TITLE’ => getMsg(«TITLE1«),
‘TEXT’ =>
$local_arc_name
?
‘<div class=t_div><input type=hidden name=arc_name value=»‘.$local_arc_name.‘»> ‘.getMsg(«ARC_LOCAL_NAME«).‘ <b>’.$local_arc_name.‘</div>’
:
($strErrMsg ? ‘<div style=»color:red»>’.$strErrMsg.‘</div>’ : »).
‘<input type=»hidden» name=»Step» value=»2″>’.
‘<div class=t_div>
<label><input type=radio name=x_source onclick=»div_show(0)» ‘.($_REQUEST[‘bitrixcloud_backup’] ? ‘checked’ : »).‘>’.getMsg(«ARC_DOWN_BITRIXCLOUD«).‘</label>
<div id=div0 class=»div-tool» style=»display:none»>
<nobr>’.getMsg(«LICENSE_KEY«).‘</nobr> <input name=license_key type=»text» id=license_key size=30 value=»‘.htmlspecialcharsbx($license_key).‘»> <input type=»button» value=» OK » onclick=»LoadFileList()»><br>
<div id=file_list></div>
</div>
</div>
<div class=t_div>
<label><input type=radio name=x_source onclick=»div_show(1)» ‘.($_REQUEST[‘arc_down_url’] ? ‘checked’ : »).‘>’.getMsg(«ARC_DOWN«).‘</label>
<div id=div1 class=»div-tool» style=»display:none»><nobr>’.getMsg(«ARC_DOWN_URL«).‘</nobr> <input name=arc_down_url type=»text» size=40 value=»‘.htmlspecialcharsbx($arc_down_url).‘»></div>
</div>
<div class=t_div>
<label><input type=radio name=x_source onclick=»div_show(2)»>’. getMsg(«ARC_LOCAL«).‘</label>
<div id=div2 class=»div-tool» style=»display:none»><span style=»color:#666″>’.getMsg(«ARC_LOCAL_WARN«).‘</span><br/><input type=file name=»archive[]» size=40 multiple onchange=»addFileField()»></div>
</div>
‘
.(strlen($option) ?
‘<div class=t_div>
<label><input type=radio name=x_source onclick=»div_show(3)»>’.getMsg(«ARC_NAME«).‘</label>
<div id=div3 class=»div-tool» style=»display:none»>
<select name=»arc_name»>’.$option.‘</select>
</div>’.
‘</div>’
: »)
.($option === false ? ‘<div style=»color:red»>’.getMsg(‘NO_READ_PERMS’).‘</div>’ : »)
.(count(getDumpList()) ?
‘<div class=t_div>’.
‘<label><input type=radio name=x_source onclick=»div_show(4)»>’.getMsg(«ARC_SKIP«).‘</label>
<div id=div4 class=»div-tool» style=»display:none;color:#999999″>’.getMsg(‘ARC_SKIP_DESC’).‘</div>
</div>’ : »)
,
‘BOTTOM’ =>
‘<a href=»/restore.php?Step=0&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=»button» id=»start_button» value=»‘.getMsg(«BUT_TEXT1«).‘» onClick=»reloadPage(2)» ‘.($local_arc_name ? » : ‘disabled’).‘>’
);
html($ar);
}
elseif($Step == 2)
{
if(!$bSelectDumpStep && !$bClearUnusedStep && !$bCloudDownloadStep)
{
$tar = new CTarRestore;
$tar->path = $_SERVER[‘DOCUMENT_ROOT’];
$tar->ReadBlockCurrent = intval($_REQUEST[‘ReadBlockCurrent’]);
$tar->EncryptKey = $_REQUEST[‘EncryptKey’];
$bottom = ‘<a href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘;
if ($rs = $tar->openRead($file1 = $file = $_SERVER[‘DOCUMENT_ROOT’].‘/’.$arc_name))
{
$DataSize = intval($_REQUEST[‘DataSize’]);
$skip = »;
if(!$DataSize) // first step
{
bx_unlink(RESTORE_FILE_LIST);
DeleteDirRec(RESTORE_FILE_DIR);
$Block = $tar->Block;
if (!$ArchiveSize = $tar->getDataSize($file))
$ArchiveSize = filesize($file) * 2; // for standard gzip files
$DataSize = $ArchiveSize;
while(file_exists($file1 = $tar->getNextName($file1)))
$DataSize += $ArchiveSize;
$r = true;
SetCurrentProgress(0);
if ($n = CTar::getLastNum($file))
{
for($i=1;$i<=$n;$i++)
{
if (!file_exists($file.‘.’.$i))
{
$strErrMsg = getMsg(‘ERR_NO_PARTS’).‘ <b>’.($n + 1).‘</b>’;
$r = false;
break;
}
}
}
}
else
{
$Block = intval($_REQUEST[‘Block’]);
$skip = ‘ <input type=button value=»‘.getMsg(‘SKIP’).‘» onClick=»document.forms.restore.skip.value=1;reloadPage(2)»>’;
if ($r = $tar->SkipTo($Block))
{
if ($_REQUEST[‘skip’])
{
while($tar->readHeader() === false);
$tar->SkipFile();
}
while(($r = $tar->extractFile()) && haveTime());
}
$strErrMsg = implode(‘<br>’,$tar->err);
}
if ($r === 0) // Finish
$bClearUnusedStep = true;
else
{
SetCurrentProgress(($tar->BlockHeader + $tar->ReadBlockCurrent) * 512,$DataSize, $red=false);
$hidden = ‘<input type=»hidden» name=»Block» value=»‘.$tar->BlockHeader.‘»>’.
‘<input type=»hidden» name=»ReadBlockCurrent» value=»‘.$tar->ReadBlockCurrent.‘»>’.
‘<input type=»hidden» name=»EncryptKey» value=»‘.htmlspecialcharsbx($tar->EncryptKey).‘»>’.
‘<input type=»hidden» name=»DataSize» value=»‘.$DataSize.‘»>’.
‘<input type=»hidden» name=»arc_name» value=»‘.htmlspecialcharsbx($arc_name).‘»>’;
if($r === false) // Error
{
showMsg(getMsg(«ERR_EXTRACT«), $status.$hidden.‘<div style=»color:red»>’.$strErrMsg.‘</div>’, $bottom.$skip);
die();
}
else
{
showMsg(getMsg(‘TITLE_PROCESS1’),$status.$hidden,$bottom);
?><script>reloadPage(2, 1);</script><?
die();
}
}
$tar->close();
}
elseif ($tar->LastErrCode == ‘ENC_KEY’)
{
$text = ($tar->EncryptKey ? ‘<div style=»color:red»>’.getMsg(‘WRONG_PASS’).‘</div>’ : »).
getMsg(‘FILE_IS_ENC’).
‘<input type=»password» size=30 name=»EncryptKey» autocomplete=»off»>’.
‘<input type=»hidden» name=»arc_name» value=»‘.htmlspecialcharsbx($arc_name).‘»>’.
‘<input type=»hidden» name=»Step» value=»2″>’;
$bottom .= ‘ <input type=»button» id=»start_button» value=»‘.getMsg(«BUT_TEXT1«).‘» onClick=»reloadPage(2)»>’;
showMsg(getMsg(‘TITLE_PROCESS1’),$text,$bottom);
die();
}
else
{
showMsg(getMsg(«ERR_EXTRACT«), getMsg(‘TAR_ERR_FILE_OPEN’).‘ ‘.implode(‘<br>’,$tar->err),$bottom);
die();
}
}
if ($bClearUnusedStep)
{
if (file_exists(RESTORE_FILE_LIST) && $f = fopen(RESTORE_FILE_LIST, ‘rb’))
{
$modules = $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/modules’;
$components = $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/components/bitrix’;
fgets($f); // «die» line
$a = array();
while(false !== $l = fgets($f))
$a[trim($l)] = 1;
fclose($f);
$ds = new CDirRealScan;
$ds->startPath = $_REQUEST[‘nextPath’];
if (strpos($_REQUEST[‘nextPath’], $components) === 0)
$init_path = $components;
else
$init_path = $modules;
$res = $ds->Scan($init_path);
if ($res === true && $init_path == $modules)
{
$ds->nextPath = $components;
$res = ‘BREAK’;
}
if ($res === ‘BREAK’)
{
$status = getMsg(«SEARCHING_UNUSED«);
$hidden = ‘<input type=»hidden» name=»nextPath» value=»‘.$ds->nextPath.‘»>’.
‘<input type=»hidden» name=»clear» value=»1″>’.GetHidden(array(‘dump_name’, ‘arc_name’));
$bottom = ‘<a href=»javascript:reloadPage(1)»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘;
showMsg(getMsg(‘TITLE_PROCESS1’),$status.$hidden,$bottom);
?><script>reloadPage(2, 1);</script><?
die();
}
bx_unlink(RESTORE_FILE_LIST);
}
if (file_exists(RESTORE_FILE_DIR))
$strWarning.= ‘<li>’.getMsg(‘WARN_CLEARED’);
if (file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/backup/sites’))
$strWarning.= ‘<li>’.getMsg(‘WARN_SITES’);
$bSelectDumpStep = true;
}
if ($strWarning)
{
$status = ‘<div style=»color:red;text-align:center»><b>’.getMsg(‘WARNING’).‘</b></div> <ul style=»color:red»>’.$strWarning.‘</ul>’;
$hidden = ‘<input type=»hidden» name=»source» value=»dump»>’.GetHidden(array(‘dump_name’, ‘arc_name’));
$bottom = ‘<a href=»javascript:reloadPage(1)»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=»button» value=»‘.getMsg(«BUT_TEXT1«).‘» onClick=»reloadPage(2)»>’;
showMsg(getMsg(‘TITLE_PROCESS1’),$status.$hidden,$bottom);
die();
}
if (file_exists(RESTORE_CLOUD_FILE_LIST))
{
$skip = $_REQUEST[‘skip’];
if ($skip)
{
}
elseif ($rsFile = fopen(RESTORE_CLOUD_FILE_LIST, ‘rb’))
{
if ($start_position = intval($_REQUEST[‘position’]))
fseek($rsFile, $start_position);
if (is_array($_REQUEST[‘mirror_list’]) && time() — intval($_REQUEST[‘init_time’]) < 600) // раз в 10 минут снова измеряем зеркала
{
$mirror_list = $_REQUEST[‘mirror_list’];
arsort($mirror_list); // выбираем самое быстрое зеркало
}
else
{
$mirror_list = [
‘cdn.bitrix24.ru’ => 999999,
‘cdn.bitrix24.com’ => 999999,
‘cdn.bitrix24.de’ => 999999,
‘cdn-ru.bitrix24.ru’ => 999999,
‘cdn-ru.bitrix24.de’ => 999999,
];
$_REQUEST[‘init_time’] = $init_time = time();
}
$mirror = key($mirror_list);
if (DEBUG)
{
echo time() — intval($_REQUEST[‘init_time’]);
echo ‘<pre>’;print_r($mirror_list);echo ‘</pre>’;
}
$file_list = [];
if (is_array($_REQUEST[‘file_list’]))
{
foreach($_REQUEST[‘file_list’] as $file => $str)
{
if (!$f = unserialize($str))
{
$strErrMsg = ‘Unserialize failure: ‘.$str;
break;
}
$file_list[$file] = $f;
$url = ‘https://’.$mirror.‘/’.$f[‘path’];
CMultiGet::startLoad($url, $file);
}
}
$concurrency = 20;
$num = 0;
while(!$strErrMsg)
{
while (haveTime() && count(CMultiGet::$connections) < $concurrency && (false !== $line = fgets($rsFile)))
{
$num++;
if (strpos($line, ‘<‘.‘?php’) === 0)
continue;
if (!preg_match(‘#^(d{4}-d{2}-d{2} [d:]{8})[ t]+(d+)[ t]+(.+)$#’, trim($line), $regs))
{
$strErrMsg = ‘File parse error ‘.RESTORE_CLOUD_FILE_LIST.‘ for line ‘.$num.‘: ‘.$line;
break;
}
$date = $regs[1];
$size = $regs[2];
$path = str_replace(‘ ‘, ‘%20’, $regs[3]);
$save_path = preg_replace(‘#^[^/]+/#’, », $regs[3]);
if (preg_match(‘#^[^/]*(cache/|tmp/|test.txt)#’, $save_path))
continue;
$url = ‘https://’.$mirror.‘/’.$path;
$file = $_SERVER[‘DOCUMENT_ROOT’].‘/upload/’.$save_path;
if (file_exists($file))
{
if (filesize($file) == $size)
continue;
if (filesize($file) > $size)
bx_unlink($file);
}
if (!CMultiGet::startLoad($url, $file))
{
$strErrMsg = nl2br(htmlspecialcharsbx(CMultiGet::$error));
break;
}
$file_list[$file] = [
‘date’ => $date,
‘size’ => $size,
‘path’ => $path,
];
}
if (!CMultiGet::getPart())
{
debug(CMultiGet::$error);
$connection = CMultiGet::$connections[CMultiGet::$current];
$url = $connection[‘url’];
CMultiGet::dropConnection(CMultiGet::$current);
foreach($mirror_list as $mirror => $speed)
{
if (strpos($url, ‘https://’.$mirror) === 0)
{
unset($mirror_list[$mirror]);
CMultiGet::$bytes = 0;
$START_TIME = microtime(1);
break;
}
}
if (count($mirror_list))
{
$mirror = key($mirror_list);
continue;
}
$strErrMsg = nl2br(htmlspecialcharsbx(CMultiGet::$error));
}
if (!haveTime())
break;
}
if (!$strErrMsg && (!feof($rsFile) || count(CMultiGet::$connections)))
{
$position = ftell($rsFile);
$size = filesize(RESTORE_CLOUD_FILE_LIST);
SetCurrentProgress($position, $size);
if (strlen($file) > 80)
$file = ‘…’.substr($file, —80);
$text = getMsg(‘ARC_DOWN_PROCESS’).‘ <b>’.htmlspecialcharsbx($file).‘</b>’.
$status.
GetHidden(array(‘source’, ‘dump_name’, ‘arc_name’, ‘init_time’)).‘
<input type=»hidden» name=»cloud_download» value=»Y»>
<input type=»hidden» name=»position» value=»‘.$position.‘»>’;
foreach($file_list as $file => $f)
{
if (!file_exists($file) || filesize($file) < $f[‘size’])
$text .= ‘<input type=»hidden» name=»file_list[‘.htmlspecialcharsbx($file).‘]» value=»‘.htmlspecialcharsbx(serialize($f)).‘»>’;
}
$mirror_list[$mirror] = round(CMultiGet::$bytes * 8/1024/1024/STEP_TIME, 2); // Mbit/s
foreach($mirror_list as $mirror => $speed)
{
$text .= ‘<input type=»hidden» name=»mirror_list[‘.htmlspecialcharsbx($mirror).‘]» value=»‘.$speed.‘»>’;
}
$bottom = ‘<a href=»javascript:reloadPage(1)»>’.getMsg(‘BUT_TEXT_BACK’).‘</a>’.
‘ <input type=button value=»‘.getMsg(‘SKIP’).‘» onClick=»document.forms.restore.skip.value=1;reloadPage(2)»>’;
showMsg(getMsg(‘TITLE3’),$text,$bottom);
?><script>reloadPage(2, 1);</script><?
die();
}
fclose($rsFile);
}
else
$strErrMsg = getMsg(«LOADER_LOAD_CANT_OPEN_READ«, [«#FILE#» => RESTORE_CLOUD_FILE_LIST]);
if (!$strErrMsg)
{
bx_unlink(RESTORE_CLOUD_FILE_LIST) || $strErrMsg = getMsg(‘FINISH_ERR_DELL_TITLE’).‘: ‘.RESTORE_CLOUD_FILE_LIST;
$bSelectDumpStep = true;
}
if ($strErrMsg)
{
$ar = array(
‘TITLE’ => getMsg(«TITLE3«),
‘TEXT’ => ‘<div style=»color:red»>’.$strErrMsg.‘</div>’,
‘BOTTOM’ =>
‘<input type=»hidden» name=»cloud_download» value=»Y»>’.
GetHidden(array(‘source’, ‘dump_name’, ‘arc_name’)).
‘<a href=»javascript:reloadPage(1)»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=»button» value=»‘.getMsg(«DUMP_RETRY«).‘» onClick=»reloadPage(2)»> ‘
);
html($ar);
?><script>reloadPage(2, 300);</script><?
die();
}
}
if ($bSelectDumpStep)
{
if (file_exists($dbconn) && $strFile = file_get_contents($dbconn))
{
$bUTF_conf = preg_match(‘#^[ t]*define(.BX_UTF.+true)#mi’, $strFile);
if ($bUTF_conf && !$bUTF_serv)
$strErrMsg = getMsg(‘UTF8_ERROR1’).‘<br><br>’.$strErrMsg;
elseif (!$bUTF_conf && $bUTF_serv)
$strErrMsg = getMsg(‘UTF8_ERROR2’).‘<br><br>’.$strErrMsg;
}
if ($strErrMsg)
{
$ar = array(
‘TITLE’ => getMsg(«TITLE2«),
‘TEXT’ => ‘<div style=»color:red»>’.$strErrMsg.‘</div>’,
‘BOTTOM’ =>
‘<input type=»hidden» name=»source» value=»dump»>’.GetHidden(array(‘dump_name’, ‘arc_name’)).
‘<a href=»javascript:reloadPage(1)»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=»button» value=»‘.getMsg(«DUMP_RETRY«).‘» onClick=»reloadPage(2)»> ‘
);
html($ar);
}
else
{
if (!$_REQUEST[‘DBName’])
{
$DBName = »;
if (file_exists($dbconn) && $str = file_get_contents($dbconn))
{
$ar = explode(«n», $str);
foreach($ar as $l)
if (preg_match(‘#^[ t]*$(DBHost|DBLogin|DBPassword|DBName)[ t]*=[ t]*[«‘]([^»‘]+)[«‘]#’, $l))
eval($l);
}
if ($DBName && !preg_match(‘#^*+$#’, $DBName))
{
$strWarning .= ‘<li>’.getMsg(‘DBCONN_WARN’);
$create_db = false;
}
else
{
$DBHost = ‘localhost’.(file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/../BitrixEnv.exe’) ? ‘:31006’ : »);
$DBLogin = ‘root’;
$DBPassword = »;
$DBName = ‘bitrix_’.(rand(11,99));
$create_db = «Y«;
}
}
else
{
$DBHost = $_REQUEST[«DBHost«];
$DBLogin = $_REQUEST[«DBLogin«];
$DBPassword = $_REQUEST[«DBPassword«];
$DBName = $_REQUEST[«DBName«];
$create_db = $_REQUEST[«create_db«] == «Y«;
}
$arDName = getDumpList();
$strDName = »;
foreach($arDName as $db)
$strDName .= ‘<option value=»‘.htmlspecialcharsbx($db).‘»>’.htmlspecialcharsbx($db).‘</option>’;
if(count($arDName))
{
$ar = array(
‘TITLE’ => getMsg(«TITLE2«),
‘TEXT’ =>
($strWarning ? ‘<div style=»color:red;text-align:center»><b>’.getMsg(‘WARNING’).‘</b></div> <ul style=»color:red»>’.$strWarning.‘</ul>’ : »).
‘<input type=»hidden» name=»arc_name» value=»‘.htmlspecialcharsbx($arc_name).‘»>’.
(count($arDName)>1 ? getMsg(«DB_SELECT«).‘ <select name=»dump_name»>’.$strDName.‘</select>’ : ‘<input type=hidden name=dump_name value=»‘.htmlspecialcharsbx($arDName[0]).‘»>’).
‘<div style=»border:1px solid #aeb8d7;padding:5px;margin-top:4px;margin-bottom:4px;»>
<div style=»text-align:center;color:#aeb8d7;margin:4px»><b>’.getMsg(«DB_SETTINGS«).‘</b></div>
<table width=100% cellspacing=0 cellpadding=2 border=0 class=»content-table»>
<tr><td align=right>’. getMsg(«BASE_HOST«).‘:</td><td><input autocomplete=off type=»text» name=»DBHost» value=»‘.htmlspecialcharsbx($DBHost).‘»></td></tr>
<tr><td align=right>’. getMsg(«USER_NAME«).‘:</td><td><input autocomplete=off type=»text» name=»DBLogin» value=»‘.htmlspecialcharsbx($DBLogin).‘»></td></tr>
<tr><td align=right>’. getMsg(«USER_PASS«).‘:</td><td><input type=»password» autocomplete=off name=»DBPassword» value=»‘.htmlspecialcharsbx($DBPassword).‘»></td></tr>
<tr><td align=right>’. getMsg(«BASE_NAME«).‘:</td><td><input autocomplete=off type=»text» name=»DBName» value=»‘.htmlspecialcharsbx($DBName).‘»></td></tr>
<tr><td align=right>’. getMsg(«BASE_CREATE_DB«).‘</td><td><input type=»checkbox» name=»create_db» value=»Y» ‘.($create_db ? ‘checked’ : »).‘></td></tr>
</table>
</div>’.
(
file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/backup/clouds’) ?
‘<div>’.getMsg(«BASE_CLOUDS«).‘
<select name=»LocalCloud»>
<option value=»Y»>’.getMsg(«BASE_CLOUDS_Y«).‘</option>
<option value=»»>’.getMsg(«BASE_CLOUDS_N«).‘</option>
</select>
</div>’
:
»
)
,
‘BOTTOM’ =>
‘<a style=»padding: 0 13px;» href=»/restore.php?Step=1&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=»button» style=»padding: 0 13px;» value=»‘.getMsg(«DB_SKIP«).‘» onClick=»reloadPage(4)»> ‘.
‘<input type=»button» style=»padding: 0 13px;» value=»‘.getMsg(«BASE_RESTORE«).‘» onClick=»reloadPage(3)»>’
);
html($ar);
}
else
showMsg(getMsg(‘FINISH’),GetHidden(array(‘dump_name’, ‘arc_name’)).‘<script>reloadPage(4);</script>’);
}
}
}
elseif($Step == 3)
{
$d_pos = (double) $_REQUEST[«d_pos«];
if ($d_pos < 0)
$d_pos = 0;
if (!isset($_REQUEST[‘d_pos’])) // start
{
if(!file_exists($dbconn))
{
if (!is_dir($dir = dirname($dbconn)))
mkdir($dir, 0777, true);
file_put_contents($dbconn, ‘<?’.»n».
‘define(«DBPersistent», false);’.»n».
‘$DBType = «mysql»;’.»n».
‘$DBHost = «»;’.»n».
‘$DBLogin = «»;’.»n».
‘$DBPassword = «»;’.»n».
‘$DBName = «»;’.»n».
«n».
‘$DBDebug = false;’.»n».
‘$DBDebugToFile = false;’.»n».
‘?>’);
}
if (file_exists($tmp = str_replace(‘dbconn.php’,‘dbconn.restore.php’,$dbconn)))
{
bx_unlink($dbconn);
rename($tmp, $dbconn);
}
$arFile = file($dbconn);
foreach($arFile as $line)
{
$line = str_replace(«rn», «n», $line);
if (preg_match(‘#^[ t]*$(DBHost|DBLogin|DBPassword|DBName)#’,$line,$regs))
{
$key = $regs[1];
$line = ‘$’.$key.‘ = «‘.str_replace(‘$’,‘$’,addslashes($_REQUEST[$key])).‘»;’.»n»;
}
$strFile .= $line;
}
if (defined(‘VMBITRIX’) && !preg_match(‘#^[ t]*define..BX_CRONTAB_SUPPORT#mi’, $strFile))
$strFile = ‘<‘.‘?define(«BX_CRONTAB_SUPPORT», true);?’.‘>’.$strFile;
file_put_contents($dbconn, $strFile);
if (file_exists($config = $_SERVER[‘DOCUMENT_ROOT’].»/bitrix/.settings.php«))
{
ob_start();
$ar = include($config);
ob_end_clean();
if (is_array($ar))
{
if (is_array($ar[‘connections’][‘value’][‘default’]))
{
$ar[‘connections’][‘value’][‘default’][‘host’] = $_REQUEST[‘DBHost’];
$ar[‘connections’][‘value’][‘default’][‘database’] = $_REQUEST[‘DBName’];
$ar[‘connections’][‘value’][‘default’][‘login’] = $_REQUEST[‘DBLogin’];
$ar[‘connections’][‘value’][‘default’][‘password’] = $_REQUEST[‘DBPassword’];
$data = var_export($ar, true);
file_put_contents($config, «<«.»?phpnreturn «.$data.»;n»);
}
}
else
rename($config, $_SERVER[‘DOCUMENT_ROOT’].»/bitrix/.settings.restore.php«); // workaround for bug #47641
}
SetCurrentProgress(0);
$r = true;
}
else
$r = $oDB->restore();
$bottom = ‘<a href=»/restore.php?Step=2&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘;
if($r && !$oDB->is_end())
{
$d_pos = $oDB->getPos();
$oDB->close();
$arc_name = $_REQUEST[«arc_name«];
$dump_name = preg_replace(‘#.[0-9]+$#’, », $_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/backup/’.$_REQUEST[‘dump_name’]);
$dump_size = 0;
while(file_exists($dump_name))
{
$dump_size += filesize($dump_name);
$dump_name = CDBRestore::getNextName($dump_name);
}
SetCurrentProgress($d_pos, $dump_size);
$text =
$status . ‘
<input type=»hidden» name=»arc_name» value=»‘.htmlspecialcharsbx($arc_name).‘»>
<input type=»hidden» name=»dump_name» value=»‘. htmlspecialcharsbx($_REQUEST[«dump_name«]).‘»>
<input type=»hidden» name=»check_site_path» value=»Y»>
<input type=»hidden» name=»d_pos» value=»‘.$d_pos.‘»>
<input type=»hidden» name=»DBLogin» value=»‘.htmlspecialcharsbx($_REQUEST[«DBLogin«]).‘»>
<input type=»hidden» name=»DBPassword» value=»‘. (strlen($_REQUEST[«DBPassword«]) > 0 ? htmlspecialcharsbx($_REQUEST[«DBPassword«]) : «»).‘»>
<input type=»hidden» name=»DBName» value=»‘. htmlspecialcharsbx($_REQUEST[«DBName«]).‘»>
<input type=»hidden» name=»DBHost» value=»‘. htmlspecialcharsbx($_REQUEST[«DBHost«]).‘»>
<input type=»hidden» name=»LocalCloud» value=»‘. ($_REQUEST[«LocalCloud«] ? ‘Y’ : »).‘»>
‘;
showMsg(getMsg(‘TITLE_PROCESS2’),$text,$bottom);
?><script>reloadPage(3, 1);</script><?
}
else
{
if($oDB->getError() != «»)
showMsg(getMsg(«ERR_DUMP_RESTORE«), ‘<div style=»color:red»>’.$oDB->getError().‘</div>’, $bottom);
else
{
if (!$oDB->ft_index)
{
$oDB->Query(‘UPDATE b_option SET VALUE=»‘.$oDB->escapeString(serialize(array())).‘» WHERE MODULE_ID=»main» AND NAME LIKE «~ft_%»‘);
}
showMsg(getMsg(‘FINISH’),GetHidden(array(‘DBLogin’,‘DBPassword’,‘DBHost’,‘DBName’,‘dump_name’, ‘arc_name’, ‘check_site_path’)).‘<script>reloadPage(4);</script>’);
}
}
}
elseif($Step == 4)
{
$strWarning .= CheckHtaccessAndWarn();
if ($_REQUEST[‘check_site_path’])
{
$oDB = new CDBRestore($_REQUEST[«DBHost«], $_REQUEST[«DBName«], $_REQUEST[«DBLogin«], $_REQUEST[«DBPassword«], $_REQUEST[«dump_name«], $d_pos);
if ($oDB->Connect())
{
if ($rs = $oDB->Query(‘SELECT * FROM b_lang WHERE DOC_ROOT != «‘.$oDB->escapeString($_SERVER[‘DOCUMENT_ROOT’]).‘» AND DOC_ROOT IS NOT NULL AND DOC_ROOT != «»‘))
{
if ($oDB->Fetch($rs))
{
$oDB->Query(‘UPDATE b_lang SET DOC_ROOT = «» ‘);
$strWarning .= ‘<li>’.getMsg(‘DOC_ROOT_WARN’);
}
}
$rs = $oDB->Query(‘SHOW TABLES LIKE «b_bitrixcloud_option»‘);
if ($oDB->Fetch($rs))
{
$rs = $oDB->Query(‘SELECT * FROM b_bitrixcloud_option WHERE NAME=»cdn_config_active» AND PARAM_VALUE=1’);
if ($oDB->Fetch($rs))
{
$rs = $oDB->Query(‘SELECT * FROM b_bitrixcloud_option WHERE NAME=»cdn_config_domain»‘);
if (($f = $oDB->Fetch($rs)) && $f[‘PARAM_VALUE’] != $_SERVER[‘HTTP_HOST’])
{
$oDB->Query(‘UPDATE b_bitrixcloud_option SET PARAM_VALUE=0 WHERE NAME=»cdn_config_active»‘);
$oDB->Query(‘UPDATE b_bitrixcloud_option SET PARAM_VALUE=’.(time() + 86400 * 3650).‘ WHERE NAME=»cdn_config_expire_time»‘);
$strWarning .= ‘<li>’.getMsg(‘CDN_WARN’);
}
}
}
if ($rs = $oDB->Query(‘SELECT * FROM b_module_to_module WHERE FROM_MODULE_ID=»main» AND MESSAGE_ID=»OnPageStart» AND TO_CLASS=»Bitrix\Security\HostRestriction»‘))
{
if ($f = $oDB->Fetch($rs)) // host restriction is turned on
{
$rs0 = $oDB->Query(‘SELECT * FROM b_option WHERE MODULE_ID=»security» AND NAME=»restriction_hosts_hosts»‘);
if ($f0 = $oDB->Fetch($rs0))
{
if (strpos($f0[‘VALUE’], $_SERVER[‘HTTP_HOST’]) === false)
{
$oDB->Query(‘DELETE FROM b_module_to_module WHERE ID=’.$f[‘ID’]);
$strWarning .= ‘<li>’.getMsg(‘HOSTS_WARN’);
}
}
}
}
}
else
$strWarning .= ‘<li>’.$oDB->getError();
}
$text =
($strWarning ? ‘<div style=»color:red;padding:10px;text-align:center»><b>’.getMsg(‘WARNING’).‘</b></div> <ul style=»color:red»>’.$strWarning.‘</ul>’ : »).
getMsg(«FINISH_MSG«).GetHidden(array(‘dump_name’, ‘arc_name’));
$bottom = ‘<a style=»padding:0 13px;font-size:13px;» href=»/restore.php?Step=2&source=dump&lang=’.LANG.‘»>’.getMsg(‘BUT_TEXT_BACK’).‘</a> ‘.
‘<input type=button style=»padding:0 13px;font-size:13px;» value=»‘.getMsg(‘DELETE_FILES’).‘» onClick=»reloadPage(5)»>’;
showMsg(getMsg(«FINISH«), $text, $bottom);
}
elseif($Step == 5)
{
$ok = true;
$ok = bx_unlink($_SERVER[‘DOCUMENT_ROOT’].‘/bitrixsetup.php’) && $ok;
if ($_REQUEST[‘dump_name’])
{
$ok = bx_unlink($_SERVER[«DOCUMENT_ROOT«].»/bitrix/backup/«.$_REQUEST[«dump_name«]) && $ok;
$ok = bx_unlink($_SERVER[«DOCUMENT_ROOT«].»/bitrix/backup/«.str_replace(‘.sql’,‘_after_connect.sql’,$_REQUEST[«dump_name«])) && $ok;
}
if($_REQUEST[‘arc_name’] && strpos($_REQUEST[‘arc_name’],‘bitrix/’) === false)
{
$ok = bx_unlink($_SERVER[«DOCUMENT_ROOT«].»/«.$_REQUEST[«arc_name«]) && $ok;
$i = 0;
while(file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/’.$_REQUEST[‘arc_name’].‘.’.++$i))
$ok = bx_unlink($_SERVER[‘DOCUMENT_ROOT’].‘/’.$_REQUEST[‘arc_name’].‘.’.$i) && $ok;
}
foreach(array(‘cache’, ‘stack_cache’, ‘managed_cache’, ‘resize_cache’, ‘tmp’) as $dir)
{
$ok = DeleteDirRec($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/’.$dir) && $ok;
}
$ok = bx_unlink($_SERVER[«DOCUMENT_ROOT«].»/restore.php«) && $ok;
if (!$ok)
showMsg(getMsg(«FINISH_ERR_DELL_TITLE«), ‘<span style=»color:red»>’.getMsg(«FINISH_ERR_DELL«).‘</span>’);
else
{
showMsg(getMsg(«FINISH«), getMsg(«FINISH_MSG«), ‘<input type=button onclick=»document.location=’/’» value=»‘.getMsg(«FINISH_BTN«).‘»>’);
?><script>window.setTimeout(function(){document.location=«/»;},5000);</script><?
}
}
#################### END ############
class CDBRestore
{
var $type = «»;
var $DBHost =»»;
var $DBName = «»;
var $DBLogin = «»;
var $DBPassword = «»;
var $DBdump = «»;
var $db_Conn = «»;
var $db_Error = «»;
var $f_end = false;
var $start;
var $d_pos;
var $_dFile;
var $mysqli;
public function __construct($DBHost, $DBName, $DBLogin, $DBPassword, $DBdump, $d_pos)
{
$this->DBHost = $DBHost;
$this->DBLogin = $DBLogin;
$this->DBPassword = $DBPassword;
$this->DBName = $DBName;
$this->DBdump = $_SERVER[«DOCUMENT_ROOT«].»/bitrix/backup/«.$DBdump;
$this->d_pos = $d_pos;
$this->mysqli = function_exists(‘mysqli_connect’);
$this->ft_index = null;
}
function Query($sql)
{
if ($this->ft_index === false && preg_match(«#^CREATE TABLE.*FULLTEXT KEY#ms«, $sql))
{
$sql = preg_replace(«#[rns]*FULLTEXT KEY[^rn]*[rn]*#m«, «», $sql);
$sql = str_replace(«),)«, «))«, $sql);
}
$rs = $this->mysqli ? mysqli_query($this->db_Conn, $sql) : mysql_query($sql, $this->db_Conn);
if ($this->ft_index === null)
{
if (preg_match(«#^CREATE TABLE.*FULLTEXT KEY#ms«, $sql))
{
$this->ft_index = (bool) $rs;
return $rs ? $rs : $this->Query($sql);
}
}
if (!$rs)
{
$this->db_Error = «<font color=#ff0000>MySQL query error!</font><br>«.($this->mysqli ? mysqli_error($this->db_Conn) : mysql_error()).‘<br><br>’.htmlspecialcharsbx($sql);
return false;
}
return $rs;
}
function Connect()
{
$this->db_Conn = $this->mysqli ? @mysqli_connect($this->DBHost, $this->DBLogin, $this->DBPassword) : @mysql_connect($this->DBHost, $this->DBLogin, $this->DBPassword);
if (!$this->db_Conn)
{
$this->db_Error = «<font color=#ff0000>MySQL connect error!</font><br>«.($this->mysqli ? mysqli_connect_error() : mysql_error()).‘<br>’;
return false;
}
$this->Query(‘SET FOREIGN_KEY_CHECKS = 0’);
$dbExists = $this->mysqli ? @mysqli_select_db($this->db_Conn, $this->DBName) : @mysql_select_db($this->DBName, $this->db_Conn);
if(!$dbExists)
{
if (@$_REQUEST[«create_db«]==»Y«)
{
if(!@$this->Query(«CREATE DATABASE `«.$this->escapeString($this->DBName).»`«))
{
$this->db_Error = getMsg(«ERR_CREATE_DB«).‘: ‘.($this->mysqli ? mysqli_error($this->db_Conn) : mysql_error());
return false;
}
$dbExists = $this->mysqli ? @mysqli_select_db($this->db_Conn, $this->DBName) : @mysql_select_db($this->DBName, $this->db_Conn);
}
if(!$dbExists)
{
$this->db_Error = «<font color=#ff0000>Error! mysql«.($this->mysqli ? ‘i’ : »).»_select_db(«.htmlspecialcharsbx($this->DBName).»)</font><br>«.($this->mysqli ? mysqli_error($this->db_Conn) : mysql_error()).»<br>«;
return false;
}
}
$this->Query(«SET SQL_MODE=»«);
$this->Query(«SET innodb_strict_mode=0«);
$after_file = str_replace(‘.sql’,‘_after_connect.sql’,$this->DBdump);
if (file_exists($after_file))
{
$arSql = explode(‘;’,file_get_contents($after_file));
foreach($arSql as $sql)
{
$sql = str_replace(‘<DATABASE>’, $this->DBName, $sql);
if (trim($sql))
$this->Query($sql);
}
}
return true;
}
function Fetch($rs)
{
return $this->mysqli ? mysqli_fetch_assoc($rs) : mysql_fetch_assoc($rs);
}
function escapeString($str)
{
return $this->mysqli ? mysqli_real_escape_string($this->db_Conn, $str) : mysql_real_escape_string($str, $this->db_Conn);
}
function readSql()
{
$cache = «»;
while(CTar::substr($cache, —2, 1) != «;«)
{
$line = fgets($this->_dFile);
if (feof($this->_dFile) || $line === false)
{
fclose($this->_dFile);
if (file_exists($next_part = self::getNextName($this->DBdump)))
{
$this->DBdump = $next_part;
if (!$this->_dFile = fopen($this->DBdump, ‘rb’))
{
$this->db_Error = «Can’t open file: «.$this->DBdump;
return false;
}
}
else
{
$this->f_end = true;
break;
}
}
$cache .= $line;
}
if($this->f_end)
return false;
return $cache;
}
function restore()
{
clearstatcache();
while($this->d_pos > ($s = filesize($this->DBdump)) && file_exists($this->DBdump))
{
$this->d_pos -= $s;
$this->DBdump = self::getNextName($this->DBdump);
}
if (!$this->_dFile = fopen($this->DBdump, ‘rb’))
{
$this->db_Error = «Can’t open file: «.$this->DBdump;
return false;
}
if($this->d_pos > 0)
fseek($this->_dFile, $this->d_pos);
while(haveTime() && ($sql = $this->readSql()))
{
if (defined(‘VMBITRIX’))
{
if (preg_match(‘#^CREATE TABLE#i’,$sql))
{
$sql = preg_replace(‘#ENGINE=MyISAM#i’,»,$sql);
$sql = preg_replace(‘#TYPE=MyISAM#i’,»,$sql);
}
}
$rs = $this->Query($sql);
if(!$rs)
{
if ($this->getErrorNum() != 1062) // Duplicate entry
return false;
if (preg_match(‘#^(INSERT[^(]+VALUES[^(]+)((.+));?$#mis’, $sql, $regs))
{
$insert = $regs[1];
foreach(preg_split(‘#),[^(](#’, $regs[2]) as $values)
{
$rs = $this->Query($insert.‘(‘.$values.‘)’);
if (!$rs && $this->getErrorNum() != 1062)
return false;
}
return true;
}
}
}
$this->Query(‘SET FOREIGN_KEY_CHECKS = 1’);
if ($this->LocalCloud && $this->f_end)
{
$i = »;
while(file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/upload/’.($name = ‘clouds’.$i)))
$i++;
if (!file_exists($f = $_SERVER[‘DOCUMENT_ROOT’].‘/upload’))
mkdir($f);
if (rename($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/backup/clouds’, $_SERVER[‘DOCUMENT_ROOT’].‘/upload/’.$name))
{
$arFiles = scandir($_SERVER[‘DOCUMENT_ROOT’].‘/upload/’.$name);
foreach($arFiles as $file)
{
if ($id = intval($file))
$this->Query(‘UPDATE b_file SET SUBDIR = CONCAT(«‘.$name.‘/’.$id.‘/», SUBDIR), HANDLER_ID=NULL WHERE HANDLER_ID =’.$id);
}
}
}
return true;
}
function getError()
{
return $this->db_Error;
}
function getErrorNum()
{
return $this->mysqli ? mysqli_errno($this->db_Conn) : mysql_errno();
}
function getPos()
{
if (is_resource($this->_dFile))
{
$res = ftell($this->_dFile);
$prev = preg_replace(‘#.[0-9]+$#’, », $this->DBdump);
clearstatcache();
while($this->DBdump != $prev)
{
if (!file_exists($prev))
return false;
$res += filesize($prev);
$prev = self::getNextName($prev);
}
return $res;
}
}
function close()
{
unset($this->_dFile);
return true;
}
function is_end()
{
return $this->f_end;
}
public static function getNextName($file)
{
static $CACHE;
$c = &$CACHE[$file];
if (!$c)
{
$l = strrpos($file, ‘.’);
$num = CTar::substr($file,$l+1);
if (is_numeric($num))
$file = CTar::substr($file,0,$l+1).++$num;
else
$file .= ‘.1’;
$c = $file;
}
return $c;
}
}
function getDumpList()
{
$arDump = array();
if (is_dir($back_dir = $_SERVER[«DOCUMENT_ROOT«].»/bitrix/backup«))
{
$handle = opendir($back_dir);
while (false !== ($file = readdir($handle)))
{
if($file == «.» || $file == «..«)
continue;
if(is_dir($back_dir.‘/’.$file))
continue;
if (strpos($file,‘_after_connect.sql’))
continue;
if(substr($file, strlen($file) — 3, 3) == «sql«)
$arDump[] = $file;
}
}
return $arDump;
}
function getMsg($index, $ar = [])
{
global $MESS;
$str = $MESS[$index];
foreach($ar as $k => $v)
$str = str_replace($k, $v, $str);
return $str;
}
function getArcList()
{
$arc = «»;
global $strErrMsg;
$handle = @opendir($_SERVER[«DOCUMENT_ROOT«]);
if (!$handle)
return false;
while (false !== ($file = @readdir($handle)))
{
if($file == «.» || $file == «..«)
continue;
if(is_dir($_SERVER[«DOCUMENT_ROOT«].»/«.$file))
continue;
if(preg_match(‘#.(tar|enc)(.gz)?$#’,$file))
$arc .= «<option value=«$file«> «.$file;
if(substr($file, strlen($file) — 7, 7) == «tar.tar«)
$strErrMsg = getMsg(‘ERR_TAR_TAR’);
}
return $arc;
}
function showMsg($title, $msg, $bottom=»)
{
$ar = array(
‘TITLE’ => $title,
‘TEXT’ => $msg,
‘BOTTOM’ => $bottom
);
html($ar);
}
function html($ar)
{
$isCrm = getenv(‘BITRIX_ENV_TYPE’) == ‘crm’;
?>
<html>
<head>
<title><?=$ar[‘TITLE’]?></title>
</head>
<body>
<style>
html, body {
padding: 0 10px;
margin: 0;
background: #2fc6f7;
position: relative;
}
p {
margin: 0 0 40px;
font-family: «Helvetica Neue», Helvetica, Arial, sans-serif;
font-size:13px;
}
.wrap {
min-height: 100vh;
position: relative;
}
.cloud {
background-size: contain;
position: absolute;
z-index: 1;
background-repeat: no-repeat;
background-position: center;
opacity: .8;
}
.cloud-fill {
background-image: url();
}
.cloud-border {
background-image: url();
}
.cloud-1 {
top: 9%;
left: 50%;
width: 60px;
height: 38px;
}
.cloud-2 {
top: 14%;
left: 12%;
width: 80px;
height: 51px;
}
.cloud-3 {
top: 11%;
right: 14%;
width: 106px;
height: 67px;
}
.cloud-4 {
top: 33%;
right: 13%;
width: 80px;
height: 51px;
}
.cloud-5 {
bottom: 23%;
right: 12%;
width: 80px;
height: 51px;
}
.cloud-6 {
bottom: 23%;
left: 12%;
width: 80px;
height: 51px;
}
.cloud-7 {
top: 13%;
left: 6%;
width: 60px;
height: 31px;
opacity: 1;
}
.cloud-8 {
top: 43%;
right: 6%;
width: 86px;
height: 54px;
opacity: 1;
}
.header {
min-height: 220px;
max-width: 727px;
margin: 0 auto;
box-sizing: border-box;
position: relative;
z-index: 10;
}
.logo-link,
.buslogo-link{
position: absolute;
top: 50%;
margin-top: -23px;
}
.logo {
width: 255px;
display: block;
height: 46px;
background-repeat: no-repeat;
/*background-size:cover;*/
}
.wrap.en .logo,
.wrap.de .logo { background-image: url(); }
.wrap.ru .logo { background-image: url(); }
.wrap.ua .logo { background-image: url(); }
.buslogo {
width: 255px;
display: block;
height: 46px;
background-repeat: no-repeat;
/*background-size:cover;*/
}
.wrap.en .buslogo,
.wrap.de .buslogo { background-image: url(); }
.wrap.ru .buslogo { background-image: url(); }
.wrap.ua .buslogo { background-image: url(); }
.content {
z-index: 10;
position: relative;
margin-bottom: 20px;
}
.content-container {
z-index: 10;
max-width: 727px;
margin: 0 auto;
padding: 28px 25px 25px;
border-radius: 11px;
box-shadow: 0 4px 20px 0 rgba(6, 54, 70, .15);
box-sizing: border-box;
text-align: center;
background-color: #fff;
position: relative;
}
.content-block {
position: relative;
z-index: 10;
}
hr {
margin: 79px 0 45px;
border: none;
height: 1px;
background: #f2f2f2;
}
h1.content-header {
color: #2fc6f7;
font: 500 40px/45px «Helvetica Neue», Helvetica, Arial, sans-serif;
margin-bottom: 13px;
margin-top: 62px;
}
h2.content-header {
color: #2fc6f7;
font: 400 27px/27px «Helvetica Neue», Helvetica, Arial, sans-serif;
margin-bottom: 13px;
margin-top: 31px;
}
h3.content-header {
color: #000;
font: 400 30px/41px «Helvetica Neue», Helvetica, Arial, sans-serif;
margin-bottom: 40px;
margin-top: 46px;
}
.content-logo {
width: 100%;
height: 57px;
background-repeat: no-repeat;
background-position: center 0;
}
.wrap.de .content-logo,
.wrap.en .content-logo{ background-image: url(); }
.wrap.ru .content-logo { background-image: url(); }
.wrap.ua .content-logo { background-image: url(); }
.setup-btn,
.content-table input[type=«submit»],
.content-table input[type=«button»]{
height: 45px;
line-height: 45px;
color: #fff;
background-color: #b5e00f;
padding: 0 45px;
text-transform: uppercase;
text-decoration: none;
font-family: «Open Sans», «Helvetica Neue», Helvetica, Arial, sans-serif;
vertical-align: middle;
border-radius: 25px;
transition: all 250ms ease;
display: inline-block;
font-size: 15px;
border: none;
cursor: pointer;
}
.setup-btn:hover {
background-color: #9bc40e;
}
.lnk {
color: #2fc6f7;
font: 15px/25px «Open Sans», «Helvetica Neue», Helvetica, Arial, sans-serif;
border-bottom: 1px solid;
text-decoration: none;
transition: all 250ms ease;
}
.lnk:hover {
border-bottom-color: transparent;
}
.progressbar-container {
width: 70%;
margin: 55px auto;
}
.progressbar-track {
height: 19px;
background: #edf2f5;
border-radius: 9px;
overflow: hidden;
position: relative;
}
.progressbar-loader {
position: absolute;
left: 0;
top: 0;
bottom: 0;
border-radius: 9px;
background-image: url();
background-position: 0 0;
background-size: auto 19px;
-webkit-animation: progressbar 2s infinite linear;
-moz-animation: progressbar 2s infinite linear;
-ms-animation: progressbar 2s infinite linear;
-o-animation: progressbar 2s infinite linear;
animation: progressbar 2s infinite linear;
}
@-webkit-keyframes progressbar { 0 {background-position: 0 0;} 100% {background-position: 75px 0;} }
@-moz-keyframes progressbar { 0 {background-position: 0 0;} 100% {background-position: 75px 0;} }
@-ms-keyframes progressbar { 0 {background-position: 0 0;} 100% {background-position: 75px 0;} }
@-o-keyframes progressbar { 0 {background-position: 0 0;} 100% {background-position: 75px 0;} }
@keyframes progressbar { 0 {background-position: 0 0;} 100% {background-position: 75px 0;} }
.progressbar-counter {
padding-top: 10px;
color: #000;
font: 300 28px/29px «Helvetica Neue», Helvetica, Arial, sans-serif;
text-align: center;
}
.lang {
vertical-align: middle;
text-align: left;
box-sizing: border-box;
color: #333;
font: 12px/22px «Helvetica Neue», Helvetica, Arial, sans-serif;
display: block;
text-decoration: none;
padding: 5px 5px 5px 35px;
}
.lang:after {
background: no-repeat center url();
content: »;
display: block;
width:16px;
height:12px;
position: absolute;
top: 50%;
left: 12px;
margin-top: -6px;
}
.lang.ru:after{ background-position: 0 0;}
.lang.en:after{ background-position: 0 -13px;}
.lang.ua:after{ background-position: 0 -26px;}
.lang.de:after{ background-position: 0 -39px;}
/**/
.select-container{
border:2px solid #d5dde0;
height: 32px;
position: absolute;
right: 0;
bottom: 0;
width:50px;
border-radius: 2px;
}
.select-container > .select-block,
.selected-lang {
width:100%;
display: block;
height: 32px;
position: relative;
cursor: pointer;
}
.selected-lang:before{
content: »;
border:4px solid #fff;
border-top:4px solid #7e939b;
display: block;
position: absolute;
right: 8px;
top: 15px;
}
.selected-lang:after{
left:11px;
}
.select-popup{
display: none;
position: absolute;
top:100%;
z-index:100;
min-width:100%;
border-radius:2px;
padding:5px 0;
background-color: #fff;
box-shadow: 0 5px 5px rgba(0,0,0,.4);
}
.select-lang-item{
height: 32px;
width:100%;
position: relative;
padding:0 10px;
box-sizing: border-box;
transition: 220ms all ease;
}
.select-lang-item:hover{
background-color: #f3f3f3;
}
.sub-header{
font-size:21px;
font-family: «Helvetica Neue», Helvetica, Arial, sans-serif;
text-align: left;
margin-bottom: 10px;
}
.select-version-container{
padding: 10px 20px 20px;
text-align: left;
font-size:16px;
line-height:25px;
font-family: «Helvetica Neue», Helvetica, Arial, sans-serif;
}
.content-table {
text-align: left;
font-size:16px;
font-family: «Helvetica Neue», Helvetica, Arial, sans-serif;
width: 100%;
}
.input-license-container input,
.content-table input[type=«text»],
.content-table input[type=«password»],
.content-table select{
/*width:100%;*/
box-sizing: border-box;
border: 2px solid #e0e6e9;
border-radius:3px;
font-size:15px;
padding: 10px;
outline: none !important;
}
.input-license-container{
padding-bottom:40px;
}
.div-tool
{
border:1px solid #CCCCCC;
padding:10px;
margin: 10px;
}
.t_div
{
padding:5px;
}
</style>
<script>
var frm;
function reloadPage(val, delay)
{
frm = document.forms.restore;
frm.Step.value = val;
window.setTimeout(function() {frm.submit()}, delay == null ? 0 : delay * 1000);
}
function addFileField()
{
var input = document.createElement(‘input’);
input.type = ‘file’;
input.name = ‘archive[]’;
input.size = 40;
input.onchange = addFileField;
input.multiple = true;
var div = document.getElementById(‘div2’);
div.appendChild(input);
}
function div_show(i)
{
document.getElementById(‘start_button’).disabled = i == 0;
for(j=0;j<=4;j++)
{
if (ob = document.getElementById(‘div’ + j))
ob.style.display = i == j ? ‘block’ : ‘none’;
}
arSources = [ ‘bitrixcloud’,‘download’,‘upload’,‘local’,‘dump’ ];
document.forms.restore.source.value = arSources[i];
if (i == 2)
document.forms.restore.action = ‘/restore.php?lang=<?=LANG?>&Step=2&source=’ + arSources[i];
}
function LoadFileList()
{
xml = new XMLHttpRequest(); // forget IE6
xml.onreadystatechange = function ()
{
if (xml.readyState == 4)
{
str = xml.responseText;
document.getElementById(‘file_list’).innerHTML = str;
document.getElementById(‘start_button’).disabled = !/<select/.test(str);
}
}
xml.open(‘POST’, ‘/restore.php’, true);
query = ‘LoadFileList=Y&lang=<?=LANG?>&bitrixcloud_backup=<?=htmlspecialcharsbx($_REQUEST[‘bitrixcloud_backup’])?>&license_key=’ + document.getElementById(‘license_key’).value;
xml.setRequestHeader(«Content-type», «application/x-www-form-urlencoded»);
xml.send(query);
}
<?
if ($_REQUEST[‘arc_down_url’])
{
?>
window.onload = div_show(1);
<?
}
elseif ($_REQUEST[‘bitrixcloud_backup’])
{
?>
window.onload = function() {
div_show(0);
LoadFileList();
}
<?
}
?>
</script>
<div class=»wrap <?=LANG?>«>
<form name=»restore» name=»restore» action=»restore.php» enctype=»multipart/form-data» method=»POST«>
<input type=»hidden» name=»lang» value=»<?=LANG?>«>
<input type=»hidden» name=»source«>
<input type=»hidden» name=»skip«>
<input type=»hidden» name=»Step» value=»<?=$Step?>«>
<header class=»header«>
<?if ($isCrm):?>
<a href=»» target=»_blank» class=»logo-link«><span class=»logo <?=LANG?>«></span></a>
<?else:?>
<a href=»» target=»_blank» class=»buslogo-link«><span class=»buslogo <?=LANG?>«></span></a>
<?endif?>
</header>
<section class=»content«>
<div class=»content-container«>
<table class=»content-table«>
<tr>
<td valign=»middle«>
<table cellpadding=0 cellspacing=0 border=0 width=100%><tr>
<td align=left >
<h3 class=»content-header» style=»margin: 0;padding: 0;text-align: left;«><?=$ar[‘TITLE’]?></h3>
<hr style=»margin: 15px 0;«>
</td>
<td align=right>
<?
$arLang = array();
foreach(array(‘en’) as $l)
$arLang[] = LANG == $l ? «<span style=’color:grey’>$l</span>» : «<a href=’?lang=$l‘ style=’color:black’>$l</a>«;
# echo implode(‘ | ‘,$arLang);
?>
</td>
</tr></table>
</td>
</tr>
<tr>
<td style=»padding:10px;font-size:10pt» valign=»<?=$ar[‘TEXT_ALIGN’]?$ar[‘TEXT_ALIGN’]:‘top’?>«>
<?=$ar[‘TEXT’]?>
</td>
</tr>
<tr>
<td style=»font-size:10pt» align=»center» valign=»middle» height=»40px«><?=$ar[‘BOTTOM’]?></td>
</tr>
</table>
<div class=»content-block«>
<div class=»select-container» onclick=»document.getElementById(‘lang-popup’).style.display = document.getElementById(‘lang-popup’).style.display == ‘block’ ? ‘none’ : ‘block’«>
<label for=»ss«><span class=»selected-lang lang <?=LANG?>«></span></label>
<div class=»select-popup» id=»lang-popup«>
<?
foreach(array(‘en’,‘de’,‘ru’) as $l)
{
?>
<div class=»select-lang-item«>
<a href=»?lang=<?=$l?>» class=»lang <?=$l?>«><?=$l?></a>
</div>
<?
}
?>
</div>
</div>
</div>
</div>
</section>
<div class=»cloud-layer«>
<div class=»cloud cloud-1 cloud-fill«></div>
<div class=»cloud cloud-2 cloud-border«></div>
<div class=»cloud cloud-3 cloud-border«></div>
<div class=»cloud cloud-4 cloud-border«></div>
<div class=»cloud cloud-5 cloud-border«></div>
<div class=»cloud cloud-6 cloud-border«></div>
</div>
</form>
</div>
</body></html>
<?
}
function SetCurrentProgress($cur,$total=0,$red=true)
{
global $status;
if (!$total)
{
$total=100;
$cur=0;
}
$val = intval($cur/$total*100);
if ($val > 100)
$val = 99;
$status = ‘
<div class=»progressbar-container»>
<div class=»progressbar-track»>
<div class=»progressbar-loader» style=»width:’.$val.‘%»></div>
</div>
<div class=»progressbar-counter»>’.$val.‘%</div>
</div>’;
}
function LoadFile($strRealUrl, $strFilename, $arHeaders = array(), $customHost = »)
{
global $proxyaddr, $proxyport, $strUserAgent, $replycode;
$ssl = preg_match(‘#^https://#i’, $strRealUrl);
$iStartSize = 0;
if (file_exists($strFilename.».tmp«))
$iStartSize = filesize($strFilename.».tmp«);
$parsedurl = parse_url($strRealUrl);
$strOriginalFile = basename($parsedurl[‘path’]);
SetCurrentStatus(getMsg(«LOADER_LOAD_QUERY_DISTR«, [«#DISTR#» => $strOriginalFile]));
$sockethandle = false;
do
{
$lasturl = $strRealUrl;
$redirection = «»;
$parsedurl = parse_url($strRealUrl);
$useproxy = (($proxyaddr != «») && ($proxyport != «»));
if (!$useproxy)
{
$host = $parsedurl[«host«];
$port = $parsedurl[«port«];
$hostname = $host;
}
else
{
$host = $proxyaddr;
$port = $proxyport;
$hostname = $parsedurl[«host«];
}
if ($customHost)
$host = $customHost;
SetCurrentStatus(getMsg(«LOADER_LOAD_CONN2HOST«, [«#HOST#» => $host]));
$port = $port ? $port : ($ssl ? 443 : 80);
$context = stream_context_create(
array(
‘ssl’ => array(
‘verify_peer’ => false,
‘allow_self_signed’ => true,
)
)
);
$sockethandle = stream_socket_client(($ssl ? ‘ssl://’ : »).$host.‘:’.$port, $error_id, $error_msg, 10, STREAM_CLIENT_CONNECT, $context);
if (!$sockethandle)
{
SetCurrentStatus(getMsg(«LOADER_LOAD_NO_CONN2HOST«, [«#HOST#» => $host]).» [«.$error_id.»] «.$error_msg);
return false;
}
else
{
debug(«n======nConnected to «.($ssl ? ‘ssl://’ : »).»$host:$portn»);
if (!$parsedurl[«path«])
$parsedurl[«path«] = «/«;
$request = «»;
if (!$useproxy)
{
$request .= «GET «.$parsedurl[«path«].($parsedurl[«query«] ? ‘?’.$parsedurl[«query«] : »).» HTTP/1.0rn»;
$request .= «Host: $hostnamern»;
}
else
{
$request .= «GET «.$strRealUrl.» HTTP/1.0rn»;
$request .= «Host: $hostnamern»;
}
if ($strUserAgent != «»)
$request .= «User-Agent: $strUserAgentrn»;
if ($iStartSize > 0)
$request .= «Range: bytes=«.$iStartSize.»—rn»;
foreach($arHeaders as $k => $v)
$request .= $k.‘: ‘.$v.»rn»;
$request .= «rn»;
fwrite($sockethandle, $request);
$result = «»;
$replyheader = «»;
while (($result = fgets($sockethandle, 4096)) && $result!=»rn»)
$replyheader .= $result;
debug(«n======n».$request.»nn».$replyheader);
$ar_replyheader = explode(«rn», $replyheader);
$replyproto = «»;
$replyversion = «»;
$replycode = 0;
$replymsg = «»;
if (preg_match(«#([A-Z]{4})/([0-9.]{3}) ([0-9]{3})#«, $ar_replyheader[0], $regs))
{
$replyproto = $regs[1];
$replyversion = $regs[2];
$replycode = IntVal($regs[3]);
$replymsg = substr($ar_replyheader[0], strpos($ar_replyheader[0], $replycode) + strlen($replycode) + 1, strlen($ar_replyheader[0]) — strpos($ar_replyheader[0], $replycode) + 1);
}
if ($replycode!=200 && $replycode!=206 && $replycode!=302 && $replycode!=301)
{
if ($replycode==403)
SetCurrentStatus(getMsg(«LOADER_LOAD_SERVER_ANSWER1«, [«#ANS#» => $replycode.» — «.$replymsg]));
else
SetCurrentStatus(getMsg(«LOADER_LOAD_SERVER_ANSWER«, [«#ANS#» => $replycode.» — «.$replymsg]));
return false;
}
$strContentRange = «»;
$strAcceptRanges = «»;
$strLocationUrl = «»;
$iNewRealSize = 0;
foreach ($ar_replyheader as $i => $headerLine)
{
if (strpos($headerLine, «Content-Range«) !== false)
$strContentRange = trim(substr($headerLine, strpos($headerLine, «:«) + 1, strlen($headerLine) — strpos($headerLine, «:«) + 1));
elseif (strpos($headerLine, «Location«) === 0)
$strLocationUrl = trim(substr($headerLine, strpos($headerLine, «:«) + 1, strlen($headerLine) — strpos($headerLine, «:«) + 1));
elseif (strpos($headerLine, «Content-Length«) !== false)
$iNewRealSize = IntVal(Trim(substr($headerLine, strpos($headerLine, «:«) + 1, strlen($headerLine) — strpos($headerLine, «:«) + 1)));
elseif (strpos($headerLine, «Accept-Ranges«) !== false)
$strAcceptRanges = Trim(substr($headerLine, strpos($headerLine, «:«) + 1, strlen($headerLine) — strpos($headerLine, «:«) + 1));
}
if (strlen($strLocationUrl)>0)
{
$redirection = $strLocationUrl;
$redirected = true;
if (!preg_match(‘#^https?://#’, $redirection))
$strRealUrl = dirname($lasturl).»/«.$redirection;
else
$strRealUrl = $redirection;
$ssl = preg_match(‘#^https://#i’, $strRealUrl);
}
if (!$strLocationUrl) // нет редиректа, загружаем файл
{
if (strpos($strRealUrl, $strOriginalFile) === false)
{
SetCurrentStatus(getMsg(«LOADER_LOAD_CANT_REDIRECT«, [«#URL#» => htmlspecialcharsbx($strRealUrl)]));
return false;
}
SetCurrentStatus(getMsg(«LOADER_LOAD_LOAD_DISTR«, [«#DISTR#» => $strRealUrl]));
$fh = fopen($strFilename.».tmp«, «ab«);
if (!$fh)
{
SetCurrentStatus(getMsg(«ERROR_CANT_WRITE«, [«#FILE#» => $strFilename.».tmp«, ‘#SPACE#’ => freeSpace()]));
return false;
}
$bFinished = True;
$downloadsize = (double) $iStartSize;
SetCurrentStatus(getMsg(«LOADER_LOAD_LOADING«));
while (!feof($sockethandle))
{
if (!haveTime())
{
$bFinished = False;
break;
}
$result = fread($sockethandle, 40960);
$downloadsize += strlen($result);
if ($result==»»)
break;
if (fwrite($fh, $result) === false)
{
SetCurrentStatus(getMsg(«ERROR_CANT_WRITE«, [«#FILE#» => $strFilename.».tmp«, ‘#SPACE#’ => freeSpace()]));
return false;
}
}
SetCurrentProgress($downloadsize,$iNewRealSize);
fclose($fh);
fclose($sockethandle);
if ($bFinished)
{
bx_unlink($strFilename);
if (rename($strFilename.».tmp«, $strFilename))
{
SetCurrentStatus(getMsg(«LOADER_LOAD_FILE_SAVED«, [«#FILE#» => $strFilename, «#SIZE#» => $downloadsize]));
return 1;
}
else
{
SetCurrentStatus(getMsg(«LOADER_LOAD_ERR_RENAME«, [«#FILE1#» => $strFilename.».tmp«, «#FILE2#» => $strFilename]));
return false;
}
}
else
return 2;
}
fclose($sockethandle);
}
}
while (true);
}
function SetCurrentStatus($str)
{
global $strLog;
$strLog .= $str.»n»;
}
class CTar
{
var $gzip;
var $file;
var $err = array();
var $LastErrCode;
var $res;
var $Block = 0;
var $BlockHeader;
var $path;
var $FileCount = 0;
var $DirCount = 0;
var $ReadBlockMax = 2000;
var $ReadBlockCurrent = 0;
var $ReadFileSize = 0;
var $header = null;
var $ArchiveSizeLimit;
const BX_EXTRA = ‘BX0000’;
const BX_SIGNATURE = ‘Bitrix Encrypted File’;
var $BufferSize;
var $Buffer;
var $dataSizeCache = array();
var $EncryptKey;
var $prefix = »;
##############
# READ
# {
function openRead($file)
{
if (!isset($this->gzip) && (self::substr($file,-3)==‘.gz’ || self::substr($file,-4)==‘.tgz’))
$this->gzip = true;
$this->BufferSize = 51200;
if ($this->open($file, ‘r’))
{
if (» !== $str = $this->gzip ? gzread($this->res,512) : fread($this->res,512))
{
$data = unpack(«a100empty/a90signature/a10version/a56tail/a256enc«, $str);
if (trim($data[‘signature’]) != self::BX_SIGNATURE)
{
if (self::strlen($this->EncryptKey))
$this->Error(‘Invalid encryption signature’,‘ENC_SIGN’);
// Probably archive is not encrypted
$this->gzip ? gzseek($this->res, 0) : fseek($this->res, 0);
$this->EncryptKey = null;
return $this->res;
}
$version = trim($data[‘version’]);
if (version_compare($version, ‘1.2’, ‘>’))
return $this->Error(‘Unsupported archive version: ‘.$version, ‘ENC_VER’);
$key = $this->getEncryptKey();
$this->BlockHeader = $this->Block = 1;
if (!$key || self::substr($str, 0, 256) != self::decrypt($data[‘enc’], $key))
return $this->Error(‘Invalid encryption key’, ‘ENC_KEY’);
}
}
return $this->res;
}
function readBlock($bIgnoreOpenNextError = false)
{
if (!$this->Buffer)
{
$str = $this->gzip ? gzread($this->res, $this->BufferSize) : fread($this->res, $this->BufferSize);
if ($str === » && $this->openNext($bIgnoreOpenNextError))
$str = $this->gzip ? gzread($this->res, $this->BufferSize) : fread($this->res, $this->BufferSize);
if ($str !== » && $key = $this->getEncryptKey())
$str = self::decrypt($str, $key);
$this->Buffer = $str;
}
$str = »;
if ($this->Buffer)
{
$str = self::substr($this->Buffer, 0, 512);
$this->Buffer = self::substr($this->Buffer, 512);
$this->Block++;
}
return $str;
}
function SkipFile()
{
if ($this->Skip(ceil(intval($this->header[‘size’])/512)))
{
$this->header = null;
return true;
}
return false;
}
function Skip($Block)
{
if ($Block == 0)
return true;
$this->Block += $Block;
$toSkip = $Block * 512;
if (self::strlen($this->Buffer) > $toSkip)
{
$this->Buffer = self::substr($this->Buffer, $toSkip);
return true;
}
$this->Buffer = »;
$NewPos = $this->Block * 512;
if ($ArchiveSize = $this->getDataSize($file = self::getFirstName($this->file)))
{
while($NewPos > $ArchiveSize)
{
$file = $this->getNextName($file);
$NewPos -= $ArchiveSize;
}
}
if ($file != $this->file)
{
$this->close();
if (!$this->open($file, $this->mode))
return false;
}
if (0 === ($this->gzip ? gzseek($this->res, $NewPos) : fseek($this->res, $NewPos)))
return true;
return $this->Error(‘File seek error (file: ‘.$this->file.‘, position: ‘.$NewPos.‘)’);
}
function SkipTo($Block)
{
return $this->Skip($Block — $this->Block);
}
function readHeader($Long = false)
{
$str = »;
while(trim($str) == »)
{
if (!($l = self::strlen($str = $this->readBlock($bIgnoreOpenNextError = true))))
return 0; // finish
}
if (!$Long)
$this->BlockHeader = $this->Block — 1;
if ($l != 512)
return $this->Error(‘Wrong block size: ‘.self::strlen($str).‘ (block ‘.$this->Block.‘)’);
$data = unpack(«a100filename/a8mode/a8uid/a8gid/a12size/a12mtime/a8checksum/a1type/a100link/a6magic/a2version/a32uname/a32gname/a8devmajor/a8devminor/a155prefix«, $str);
$chk = $data[‘devmajor’].$data[‘devminor’];
if (!is_numeric(trim($data[‘checksum’])) || $chk!=» && $chk!=0)
return $this->Error(‘Archive is corrupted, wrong block: ‘.($this->Block—1).‘, file: ‘.$this->file.‘, md5sum: ‘.md5_file($this->file));
$header[‘filename’] = trim(trim($data[‘prefix’], «x00»).‘/’.trim($data[‘filename’], «x00»),‘/’);
$header[‘mode’] = OctDec($data[‘mode’]);
$header[‘uid’] = OctDec($data[‘uid’]);
$header[‘gid’] = OctDec($data[‘gid’]);
$header[‘size’] = OctDec($data[‘size’]);
$header[‘mtime’] = OctDec($data[‘mtime’]);
$header[‘type’] = trim($data[‘type’], «x00»);
// $header[‘link’] = $data[‘link’];
if (self::strpos($header[‘filename’],‘./’) === 0)
$header[‘filename’] = self::substr($header[‘filename’], 2);
if ($header[‘type’]==‘L’) // Long header
{
$filename = »;
$n = ceil($header[‘size’]/512);
for ($i = 0; $i < $n; $i++)
$filename .= $this->readBlock();
if (!is_array($header = $this->readHeader($Long = true)))
return $this->Error(‘Wrong long header, block: ‘.$this->Block);
$header[‘filename’] = self::substr($filename,0,self::strpos($filename,chr(0)));
}
if (self::strpos($header[‘filename’],‘/’) === 0) // trailing slash
$header[‘type’] = 5; // Directory
if ($header[‘type’]==‘5’)
$header[‘size’] = »;
if ($header[‘filename’]==»)
return $this->Error(‘Filename is empty, wrong block: ‘.($this->Block—1));
if (!$this->checkCRC($str, $data))
return $this->Error(‘Checksum error on file: ‘.$header[‘filename’]);
$this->header = $header;
return $header;
}
function checkCRC($str, $data)
{
$checksum = $this->checksum($str);
$res = octdec($data[‘checksum’]) == $checksum || $data[‘checksum’]===0 && $checksum==256;
return $res;
}
function extractFile()
{
if ($this->header === null)
{
if(($header = $this->readHeader()) === false || $header === 0 || $header === true)
{
if ($header === true && $this->SkipFile() === false)
return false;
return $header;
}
$this->lastPath = $f = $this->path.‘/’.$header[‘filename’];
if ($this->ReadBlockCurrent == 0)
{
if ($header[‘type’]==5) // dir
{
if(!file_exists($f) && !self::xmkdir($f))
return $this->ErrorAndSkip(‘Can’t create folder: ‘.$f);
//chmod($f, $header[‘mode’]);
}
else // file
{
if (!self::xmkdir($dirname = dirname($f)))
return $this->ErrorAndSkip(‘Can’t create folder: ‘.$dirname);
elseif (($rs = fopen($f, ‘wb’))===false)
return $this->ErrorAndSkip(getMsg(‘ERROR_CANT_WRITE’, [‘#FILE#’ => $f, ‘#SPACE#’ => freeSpace()]));
}
}
else
return $this->Skip($this->ReadBlockCurrent);
}
else // файл уже частично распакован, продолжаем на том же хите
{
$header = $this->header;
$this->lastPath = $f = $this->path.‘/’.$header[‘filename’];
}
if ($header[‘type’] != 5) // пишем контент в файл
{
if (!$rs)
{
if (($rs = fopen($f, ‘ab’))===false)
return $this->ErrorAndSkip(getMsg(‘ERROR_CANT_WRITE’, [‘#FILE#’ => $f, ‘#SPACE#’ => freeSpace()]));
}
$i = 0;
$FileBlockCount = ceil($header[‘size’] / 512);
while(++$this->ReadBlockCurrent <= $FileBlockCount && ($contents = $this->readBlock()))
{
if ($this->ReadBlockCurrent == $FileBlockCount && ($chunk = $header[‘size’] % 512))
$contents = self::substr($contents, 0, $chunk);
$ret = fwrite($rs, $contents);
if ($ret === false || $ret === 0)
return $this->Error(getMsg(‘ERROR_CANT_WRITE’, [‘#FILE#’ => $f, ‘#SPACE#’ => freeSpace()]));
if ($this->ReadBlockMax && ++$i >= $this->ReadBlockMax)
{
fclose($rs);
return true; // Break
}
}
fclose($rs);
if (($s = filesize($f)) != $header[‘size’])
return $this->Error(‘File size is wrong: ‘.$header[‘filename’].‘ (real: ‘.$s.‘ expected: ‘.$header[‘size’].‘)’);
//chmod($f, $header[‘mode’]);
}
if ($this->header[‘type’]==5)
$this->DirCount++;
else
$this->FileCount++;
$this->debug_header = $this->header;
$this->BlockHeader = $this->Block;
$this->ReadBlockCurrent = 0;
$this->header = null;
return true;
}
function openNext($bIgnoreOpenNextError)
{
if (file_exists($file = $this->getNextName()))
{
$this->close();
return $this->open($file,$this->mode);
}
elseif (!$bIgnoreOpenNextError)
return $this->Error(«File doesn’t exist: «.$file);
return false;
}
public static function getLastNum($file)
{
$file = self::getFirstName($file);
if (!file_exists($file))
return false;
$f = fopen($file, ‘rb’);
fseek($f, 12);
if (fread($f, 2) == ‘LN’)
$res = end(unpack(‘va’,fread($f, 2)));
else
$res = false;
fclose($f);
return $res;
}
# }
##############
##############
# BASE
# {
function open($file, $mode=‘r’)
{
$this->file = $file;
$this->mode = $mode;
if (is_dir($file))
return $this->Error(‘File is directory: ‘.$file);
if ($this->EncryptKey && !function_exists(‘mcrypt_encrypt’) && !function_exists(‘openssl_encrypt’))
return $this->Error(‘Function mcrypt_encrypt/openssl_encrypt is not available’);
if ($mode == ‘r’ && !file_exists($file))
return $this->Error(‘File does not exist: ‘.$file);
if ($this->gzip)
{
if(!function_exists(‘gzopen’))
return $this->Error(‘Function "gzopen" is not available’);
else
{
if ($mode == ‘a’ && !file_exists($file) && !$this->createEmptyGzipExtra($file))
return false;
$this->res = gzopen($file,$mode.»b«);
}
}
else
$this->res = fopen($file,$mode.»b«);
return $this->res;
}
function close()
{
if ($this->mode == ‘a’)
$this->flushBuffer();
if ($this->gzip)
{
gzclose($this->res);
if ($this->mode == ‘a’)
{
// добавим фактический размер всех несжатых данных в extra поле
$f = fopen($this->file, ‘rb+’);
fseek($f, 18);
fwrite($f, pack(«V«, $this->ArchiveSizeCurrent));
fclose($f);
$this->dataSizeCache[$this->file] = $this->ArchiveSizeCurrent;
// сохраним номер последней части в первый архив для многотомных архивов
if (preg_match(‘#^(.+).([0-9]+)$#’, $this->file, $regs))
{
$f = fopen($regs[1], ‘rb+’);
fseek($f, 12);
fwrite($f, ‘LN’.pack(«v«,$regs[2]));
fclose($f);
}
}
}
else
fclose($this->res);
clearstatcache();
}
public function getNextName($file = »)
{
if (!$file)
$file = $this->file;
static $CACHE;
$c = &$CACHE[$file];
if (!$c)
{
$l = strrpos($file, ‘.’);
$num = self::substr($file,$l+1);
if (is_numeric($num))
$file = self::substr($file,0,$l+1).++$num;
else
$file .= ‘.1’;
$c = $file;
}
return $c;
}
function checksum($s)
{
$chars = count_chars(self::substr($s,0,148).‘ ‘.self::substr($s,156,356));
$sum = 0;
foreach($chars as $ch => $cnt)
$sum += $ch*$cnt;
return $sum;
}
public static function substr($s, $a, $b = null)
{
if (function_exists(‘mb_orig_substr’))
return $b === null ? mb_orig_substr($s, $a) : mb_orig_substr($s, $a, $b);
return $b === null ? substr($s, $a) : substr($s, $a, $b);
}
public static function strlen($s)
{
if (function_exists(‘mb_orig_strlen’))
return mb_orig_strlen($s);
return strlen($s);
}
public static function strpos($s, $a)
{
if (function_exists(‘mb_orig_strpos’))
return mb_orig_strpos($s, $a);
return strpos($s, $a);
}
function getDataSize($file)
{
$size = &$this->dataSizeCache[$file];
if (!$size)
{
if (!file_exists($file))
$size = false;
else
{
if (preg_match(‘#.gz(.[0-9]+)?$#’,$file))
{
$f = fopen($file, «rb«);
fseek($f, 16);
if (fread($f, 2) == ‘BX’)
$size = end(unpack(«V«, fread($f, 4)));
else
{
// $this->Error(‘Wrong GZIP Extra Field’);
$size = false;
}
fclose($f);
}
else
$size = filesize($file);
}
}
return $size;
}
function Error($str = », $code = »)
{
if ($code)
$this->LastErrCode = $code;
$this->err[] = $str;
return false;
}
function ErrorAndSkip($str = », $code = »)
{
$this->Error($str, $code);
$this->SkipFile();
if ($this->readHeader() === 0)
$this->BlockHeader = $this->Block;
return false;
}
public static function xmkdir($dir)
{
if (!file_exists($dir))
{
$upper_dir = dirname($dir);
if (!file_exists($upper_dir) && !self::xmkdir($upper_dir))
return $this->Error(‘Can’t create folder: ‘.$upper_dir);
return mkdir($dir);
}
return is_dir($dir);
}
function getEncryptKey()
{
if (!$this->EncryptKey)
return false;
static $key;
if (!$key)
$key = md5($this->EncryptKey);
return $key;
}
function getFileInfo($f)
{
$f = str_replace(‘’, ‘/’, $f);
$path = self::substr($f,self::strlen($this->path) + 1);
$ar = array();
if (is_dir($f))
{
$ar[‘type’] = 5;
$path .= ‘/’;
}
else
$ar[‘type’] = 0;
if (!$info = stat($f))
return $this->Error(‘Can’t get file info: ‘.$f);
if ($info[‘size’] < 0)
return $this->Error(‘File is too large: ‘.$f);
$ar[‘mode’] = 0777 & $info[‘mode’];
$ar[‘uid’] = $info[‘uid’];
$ar[‘gid’] = $info[‘gid’];
$ar[‘size’] = $ar[‘type’]==5 ? 0 : $info[‘size’];
$ar[‘mtime’] = $info[‘mtime’];
$ar[‘filename’] = $this->prefix.$path;
return $ar;
}
public static function getCheckword($key)
{
return md5(‘BITRIXCLOUDSERVICE’.$key);
}
public static function getFirstName($file)
{
return preg_replace(‘#.[0-9]+$#’,»,$file);
}
public static function encrypt($data, $md5_key)
{
if ($m = self::strlen($data)%8)
$data .= str_repeat(«x00», 8 — $m);
if (function_exists(‘openssl_encrypt’))
return openssl_encrypt($data, ‘BF-ECB’, $md5_key, OPENSSL_RAW_DATA | OPENSSL_NO_PADDING);
else
return mcrypt_encrypt(MCRYPT_BLOWFISH, $md5_key, $data, MCRYPT_MODE_ECB);
}
public static function decrypt($data, $md5_key)
{
if (function_exists(‘openssl_encrypt’))
$val = openssl_decrypt($data, ‘BF-ECB’, $md5_key, OPENSSL_RAW_DATA | OPENSSL_NO_PADDING);
else
$val = mcrypt_decrypt(MCRYPT_BLOWFISH, $md5_key, $data, MCRYPT_MODE_ECB);
return $val;
}
# }
##############
}
class CTarRestore extends CTar
{
function readHeader($Long = false)
{
$header = parent::readHeader($Long);
if (!$Long && is_array($header))
{
$dr = str_replace(array(‘/’,‘’),»,$_SERVER[‘DOCUMENT_ROOT’]);
$f = str_replace(array(‘/’,‘’),»,$this->path.‘/’.$header[‘filename’]);
if ($header[‘type’] != 5 && (self::strpos($f, $dr.‘bitrixmodules’) === 0 || self::strpos($f, $dr.‘bitrixcomponentsbitrix’) === 0))
{
if (!file_exists(RESTORE_FILE_LIST))
{
self::xmkdir($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/tmp’);
file_put_contents(RESTORE_FILE_LIST, ‘<‘.‘?php die(); ?’.»>n»);
}
file_put_contents(RESTORE_FILE_LIST, addslashes(self::substr(str_replace(‘’,‘/’,$header[‘filename’]), 7)).»n», 8); // strlen(bitrix/) = 7
}
if ($f == $dr.‘restore.php’)
return true;
elseif ($f == $dr.‘.htaccess’)
{
$header[‘filename’] .= ‘.restore’;
$this->header[‘filename’] = $header[‘filename’];
}
elseif ($f == $dr.‘bitrixphp_interfacedbconn.php’ && file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/php_interface/dbconn.php’))
$header[‘filename’] = str_replace(‘dbconn.php’,‘dbconn.restore.php’,$header[‘filename’]);
elseif (preg_match(‘#[^x00-x7f]#’, $header[‘filename’])) // non ASCII character detected
{
$this->header[‘filename’] = $header[‘filename’] = $this->DecodeFileName($header[‘filename’]);
if ($this->header[‘filename’] === false)
return false;
}
}
return $header;
}
function DecodeFileName($str)
{
if (!$this->EncCurrent)
{
if (PHP_EOL == «rn») // win
{
if (preg_match(‘#.([0-9]+)$#’, setlocale(LC_CTYPE, 0), $regs))
$this->EncCurrent = ‘cp’.$regs[1];
else
$this->EncCurrent = ‘cp1251’;
}
else
$this->EncCurrent = ‘utf-8’;
if (preg_match(«/[xC0—xDF][x80—xBF]{1}|[xE0—xEF][x80—xBF]{2}|[xF0—xF7][x80—xBF]{3}/«, $str)) // 110xxxxx 10xxxxxx | 1110xxxx 10xxxxxx 10xxxxxx | 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx
$this->EncRemote = ‘utf-8’;
elseif (preg_match(«/[xC0—xFF]/«,$str))
$this->EncRemote = ‘cp1251’;
else
return $this->Error(getMsg(‘ERR_CANT_DETECT_ENC’).‘ /’.$str);
}
if ($this->EncCurrent == $this->EncRemote)
return $str;
if (!function_exists(‘mb_convert_encoding’))
return $this->Error(getMsg(‘ERR_CANT_DECODE’));
return mb_convert_encoding($str, $this->EncCurrent, $this->EncRemote);
}
}
function haveTime()
{
return microtime(true) — START_EXEC_TIME < STEP_TIME;
}
function img($name)
{
if (file_exists($_SERVER[‘DOCUMENT_ROOT’].‘/images/’.$name))
return ‘/images/’.$name;
return ‘https://www.1c-bitrix.ru/images/bitrix_setup/’.$name;
}
function bx_accelerator_reset()
{
if(function_exists(«accelerator_reset«))
accelerator_reset();
elseif(function_exists(«wincache_refresh_if_changed«))
wincache_refresh_if_changed();
}
function DeleteDirRec($path)
{
if (file_exists($path) && $dir = opendir($path))
{
while(($item = readdir($dir)) !== false)
{
if ($item == ‘.’ || $item == ‘..’)
continue;
if (is_file($f = $path.‘/’.$item))
{
if (!bx_unlink($f))
return false;
}
else
{
if (!DeleteDirRec($f))
return false;
}
}
closedir($dir);
if (!rmdir($path))
return false;
}
return true;
return true;
}
function CheckHtaccessAndWarn()
{
$tmp = $_SERVER[‘DOCUMENT_ROOT’].‘/.htaccess’;
$tmp1 = $tmp.‘.restore’;
if (!file_exists($tmp1))
return »;
if (file_exists($tmp))
{
if (trim(file_get_contents($tmp)) == trim(file_get_contents($tmp1)))
{
bx_unlink($tmp1);
return »;
}
else
return ‘<li>’.getMsg(‘HTACCESS_RENAMED_WARN’);
}
else
{
if (file_put_contents($tmp,
‘Options -Indexes
ErrorDocument 404 /404.php
<IfModule mod_php5.c>
php_flag allow_call_time_pass_reference 1
php_flag session.use_trans_sid off
#php_value display_errors 1
#php_value mbstring.internal_encoding UTF-8
</IfModule>
<IfModule mod_rewrite.c>
Options +FollowSymLinks
RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-l
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !/bitrix/urlrewrite.php$
RewriteRule ^(.*)$ /bitrix/urlrewrite.php [L]
RewriteRule .* — [E=REMOTE_USER:%{HTTP:Authorization}]
</IfModule>
<IfModule mod_dir.c>
DirectoryIndex index.php index.html
</IfModule>
<IfModule mod_expires.c>
ExpiresActive on
ExpiresByType image/jpeg «access plus 3 day»
ExpiresByType image/gif «access plus 3 day»
</IfModule>’))
return ‘<li>’.getMsg(‘HTACCESS_WARN’);
else
return ‘<li>’.getMsg(‘HTACCESS_ERR_WARN’);
}
}
function GetHidden($ar)
{
$str = »;
foreach($ar as $k)
{
if (is_array($_REQUEST[$k]))
{
foreach($_REQUEST[$k] as $k0 => $v)
$str .= ‘<input type=hidden name=»‘.$k.‘[‘.htmlspecialcharsbx($k0).‘]» value=»‘.htmlspecialcharsbx($_REQUEST[$k][$k0]).‘»>’;
}
else
$str .= ‘<input type=hidden name=»‘.$k.‘» value=»‘.htmlspecialcharsbx($_REQUEST[$k]).‘»>’;
}
return $str;
}
class CDirScan
{
var $DirCount = 0;
var $FileCount = 0;
var $err= array();
var $bFound = false;
var $nextPath = »;
var $startPath = »;
var $arIncludeDir = false;
function __construct()
{
}
function ProcessDirBefore($f)
{
return true;
}
function ProcessDirAfter($f)
{
return true;
}
function ProcessFile($f)
{
return true;
}
function Skip($f)
{
if ($this->startPath)
{
if (strpos($this->startPath.‘/’, $f.‘/’) === 0)
{
if ($this->startPath == $f)
unset($this->startPath);
return false;
}
else
return true;
}
return false;
}
function Scan($dir)
{
$dir = str_replace(‘’,‘/’,$dir);
if ($this->Skip($dir))
return;
$this->nextPath = $dir;
if (is_dir($dir))
{
#############################
# DIR
#############################
if (!$this->startPath)
{
$r = $this->ProcessDirBefore($dir);
if ($r === false)
return false;
}
if (!($handle = opendir($dir)))
{
$this->err[] = ‘Error opening dir: ‘.$dir;
return false;
}
while (($item = readdir($handle)) !== false)
{
if ($item == ‘.’ || $item == ‘..’ || false !== CTar::strpos($item,‘’))
continue;
$f = $dir.»/«.$item;
$r = $this->Scan($f);
if ($r === false || $r === ‘BREAK’)
{
closedir($handle);
return $r;
}
}
closedir($handle);
if (!$this->startPath)
{
if ($this->ProcessDirAfter($dir) === false)
return false;
$this->DirCount++;
}
}
else
{
#############################
# FILE
#############################
$r = $this->ProcessFile($dir);
if ($r === false)
return false;
elseif ($r === ‘BREAK’)
return $r;
$this->FileCount++;
}
return true;
}
}
class CDirRealScan extends CDirScan
{
function Scan($dir)
{
if (!$this->cut)
$this->cut = CTar::strlen($_SERVER[‘DOCUMENT_ROOT’].‘/bitrix/’);
return parent::Scan($dir);
}
function ProcessFile($f)
{
if (!haveTime())
return ‘BREAK’;
global $a;
if (!$a)
return;
$k = CTar::substr($f, $this->cut);
if (!isset($a[$k]))
{
$to = RESTORE_FILE_DIR.‘/’.$k;
CTar::xmkdir(dirname($to));
rename($f, $to);
}
return true;
}
}
class CMultiGet
{
static $error = »;
static $free_connections = [];
static $connections = [];
static $current;
static $bytes = 0;
static function getConnection($connect_string)
{
$new_connection = null;
foreach(self::$free_connections as $key => $connection)
{
if ($connection[‘connect_string’] == $connect_string)
{
if (!self::isAlive($connection[‘socket’])) // сервер закрыл соединение
{
unset(self::$free_connections[$key]);
continue;
}
$new_connection = self::$free_connections[$key];
unset(self::$free_connections[$key]);
break;
}
}
if (!$new_connection)
{
if ($sock = stream_socket_client($connect_string, $errno, $errstr, 5))
{
stream_set_blocking($sock, false);
}
else
{
self::$error = ‘Can’t connect to ‘.$connect_string.‘ [‘.$errno.‘] ‘.$errstr;
return false;
}
$new_connection = [
‘connect_string’ => $connect_string,
‘socket’ => $sock,
‘code’ => 0,
‘length’ => 0,
‘saved_length’ => 0,
‘microtime’ => microtime(1),
];
}
return $new_connection;
}
static function startLoad($url, $file)
{
$u = parse_url($url);
if (!$u[‘port’])
$u[‘port’] = $u[‘scheme’] == ‘https’ ? 443 : 80;
$connect_string = ($u[‘scheme’] == ‘https’ ? ‘ssl://’ : ‘tcp://’).$u[‘host’].‘:’.$u[‘port’];
if (!$connection = self::getConnection($connect_string))
return false;
$connection[‘state’] = 0;
$connection[‘url’] = $url;
$connection[‘file’] = $file;
$strReq =
‘GET ‘.$u[‘path’].($u[‘query’] ? ‘?’.$u[‘query’] : »).‘ HTTP/1.0’.»rn».
‘Connection: keep-alive’.»rn».
‘Host: ‘.$u[‘host’].»rn»;
if (file_exists($file))
$strReq .= ‘Range: bytes=’.filesize($file).‘-‘.»rn»;
$strReq .= «rn»;
if (!fwrite($connection[‘socket’], $strReq))
{
self::$error = ‘Can’t write to ‘.$connect_string;
return false;
}
self::$connections[] = $connection;
end(self::$connections);
debug(‘Connection #’.key(self::$connections).‘ request’.»n».$strReq);
return true;
}
static function getPart()
{
if (!count(self::$connections))
return true;
$arReadSock = [];
foreach(self::$connections as $key => $connection)
{
if (self::isAlive($connection[‘socket’]))
$arReadSock[] = $connection[‘socket’];
}
$n = stream_select($arReadSock, $w = null, $e = null, 1);
if ($n === 0)
return true;
foreach($arReadSock as $sock)
{
$key = null;
foreach(self::$connections as $key => $connection)
if ($connection[‘socket’] == $sock)
break;
self::$current = $key;
$file = self::$connections[$key][‘file’];
$state =& self::$connections[$key][‘state’];
$header = »;
if ($state == 0) // начало загрузки, читаем заголовки
{
if (false === $line = fgets($sock))
continue;
if (!preg_match(‘#^HTTP/1.. ([0-9]+)#’, trim($line), $regs))
{
self::$error = ‘Invalid reply header: ‘.trim($line).»n».‘File: ‘.$file;
return false;
}
$header .= $line;
$state = $regs[1];
if ($state == 416) // Requested Range Not Satisfiable
{
// файл загружен, ничего не делаем
}
elseif (!preg_match(‘#^2d{2}$#’, $state))
{
self::$error = ‘Wrong reply code: ‘.trim($line).»n».‘File: ‘.$file;
return false;
}
do
{
$line = fgets($sock);
if (preg_match(‘#Content-length: (d+)#i’, $line, $regs))
{
self::$connections[$key][‘length’] = $regs[1];
}
$header .= $line;
} while ($line != «rn»);
debug(‘Connection #’.$key.‘ response’.»n».$header);
}
if (feof($sock) || $state == 416)
{
self::freeConnection($key);
}
else
{
$str = fread($sock, 1024 * 1024);
$dir = dirname($file);
if (!file_exists($dir))
mkdir($dir, 0777, true);
$bytes = file_put_contents($file, $str, 8);
if ($bytes === false)
{
self::$error = getMsg(‘ERROR_CANT_WRITE’, [‘#FILE#’ => $file, ‘#SPACE#’ => freeSpace()]);
return false;
}
self::$bytes += $bytes;
self::$connections[$key][‘saved_length’] += $bytes;
if (self::$connections[$key][‘saved_length’] >= self::$connections[$key][‘length’])
{
self::freeConnection($key);
}
}
}
return true;
}
static function isAlive($sock)
{
return is_resource($sock) && get_resource_type($sock) == ‘stream’;
}
static function freeConnection($key)
{
$sock = self::$connections[$key][‘socket’];
if (self::isAlive($sock))
{
self::$free_connections[] = [
‘connect_string’ => self::$connections[$key][‘connect_string’],
‘microtime’ => self::$connections[$key][‘microtime’],
‘socket’ => $sock,
];
unset(self::$connections[$key]);
}
}
static function dropConnection($key)
{
$sock = self::$connections[$key][‘socket’];
if (self::isAlive($sock))
fclose($sock);
unset(self::$connections[$key]);
}
}
function bx_unlink($file)
{
if (!file_exists($file))
return true;
if (DEBUG)
return rename($file, $file.‘.debug’);
return unlink($file);
}
function debug($str)
{
if (!DEBUG)
return;
if (is_array($str))
$str = print_r($str, 1);
file_put_contents($_SERVER[‘DOCUMENT_ROOT’].‘/log.txt’, date(‘[Y-m-d H:i:s] ‘).$str.»n», 8);
}
function freeSpace()
{
$d = disk_free_space(__FILE__);
if ($d === false)
return ‘N/A’;
return HumanSize($d);
}
function HumanSize($s)
{
$i = 0;
$ar = array(‘b’,‘kb’,‘Mb’,‘Gb’);
while($s > 1024)
{
$s /= 1024;
$i++;
}
return round($s,2).».$ar[$i];
}
Если вы еще не восстанавливали/переносили Битрикс на свеженький (или не очень) сервер с PHP 7, то вы счастливый человек. Нет, сам Битрикс на PHP 7 работает более чем хорошо, я бы даже сказал, что намного лучше, чем на 5.X.
Т.е. все прекрасно, кроме самого процесса переноса. Если вы воспользуетесь официальным инструментом от Битрикса – скриптом restore.php, то столкнетесь с проблемами. Собственно, как только дело дойдет до восстановления базы данных – сервер упадает в 500 ошибку, а в логах появится следующая запись:
|
PHP Fatal error: Uncaught Error:Call toundefined functionmysql_connect()in/var/www/html/restore.php:1328 Stack trace: #0 /var/www/html/restore.php(670): CDBRestore->Connect() #1 {main} thrown in/var/www/html/restore.php on line1328 |
Т.е. в скрипте по прежнему используется старая библиотекой php для работы с MySQL, вместо mysqli – уже как несколько лет обозначенной, как единственно верное и поддерживаемое решение.
А в PHP 7 больше нет поддержки старой библиотеки для mysql, это известно всем, кроме тех людей которые занимаются скрипом восстановления (я уверен, что им уже сказали, но пока они раскачаются…).
Решение тут понятно простое – восстановить базу руками, что, понятно, несложно. Нет один, два раза без проблем, но на десятый уже конечно надоедает.
Собственно решение тут одно, взять и поправить код скрипта, благо правок так немного. Что и было сделано:
- Убран код, который скачивает свежую версию скрипта с Битрикса и подменяет текущий файл;
- Собственно все старые не поддерживаемые функции заменены на аналоги из mysqli
Скачать: bitrix_restore_php7
Как только в Битрикс выпустят свою нормальную версию – ссылку заменю на официальный продукт.
Upd: Вышел официальный restore.php с поддержкой mysqli.
Когда вы разрабатываете веб-сайт, есть один важный шаг: настройка резервного копирования на Битриксе. К счастью, создание надлежащей резервной копии — относительно простая задача. Для этого нужно решить следующие вопросы:
В этой статье компания Alto расскажет:
- Как избежать системного краха сайта.
- Как сделать резервную копию сайта на 1С-Битрикс.
- Как восстановить сайт на 1С-Битрикс.
Как избежать системного краха сайта
Первоначально владельцу нужно разработать политику создания резервных копий:
- сроки или периодичность;
- количество архивов датируемых разной датой.
Например, рекомендуется перед каждым изменением кода сайта сохранять старую версию, чтобы быстро «откатить» к ней в случае выявления серьезных неполадок в работе ресурса. Дамп предыдущего релиза лучше хранить вплоть до последующих изменений, ведь неизвестно, когда проявятся вероятные ошибки в коде.
Соблюдать график резервного копирования сайта на Битрикс легко, достаточно настроить систему копирования на автоматическое создание архивов. Такой подход снижает потребность в персонале, потому что дампы будут сниматься без участия сотрудников. Восстановление также работает «по нажатию одной кнопки».
Полностью рассчитывать на автоматику не стоит, особенно, в периоды обновления контента. Средств защиты от системных сбоев с абсолютной гарантией не существует, поэтому делать резервную копию своего сайта нужно обязательно. То же относится и к другим продуктам 1С типа CRM Битрикс24.
Во втором случае резервное копирование в Битрикс лучше осуществлять вручную и сразу скачивать архивы на локальный накопитель. Например, при подключении к серверу через FTP. Тогда даже при полном крахе сайта получится восстановить работу почти без простоев (типовой ресурс разворачивается за 10-40 минут). Такое особенно интересно, если по тарифу хостинг-провайдера не удается делать более 2-3 архивных копий.
Перед составлением расписания по созданию резервных копий сайта Битрикс нужно дать ответы на ряд вопросов, касающихся эффективности использования системы архивации. Например, будет ли резервироваться весь сайт или речь идет о папке с пользовательским контентом. Или планируется запуск сразу нескольких ресурсов с разным содержимым и по каждому нужно определить политику архивации отдельно от других.
Есть еще один важный вопрос – что «пропадет» на официальном сайте, если его вернуть в прежнее состояние. Здесь поможет ответственный подход к публикации контента, появление новых вкладок и обновление опубликованных лучше отмечать в специальном отчете. Если речь идет об интернет-магазине, могут вернуться некорректные цены. В этом случае после восстановления лучше сразу же произвести синхронизацию с учетной базой для восстановления нужных цифровых значений.
Резервные копии на стороне хостинг-провайдера
.png)
На виртуальном хостинге независимо от провайдера обычно работает система резервирования без вмешательства пользователя. Созданные копии отображаются в личном кабинете, там же имеются кнопки для восстановления, ручного архивирования сайта. На размещение файла «по умолчанию» используется пространство, выделенное под сайт, поэтому на сервере лежит не более 2-3 последних дампов, включающих полную копию ресурса (и парочка только с изменениями).
Иногда система хостинга делает архивы, которые содержат только файлы, измененные с момента осуществления последнего «снимка». Поэтому надо быть внимательным при переносе резервного файла на локальное хранилище. Если есть сомнения, лучше сделать копию Битрикс вручную, и уже ее необходимо скачать как «шаблонный» вариант. Последнее важно, потому что проблемы могут проявиться через 2-3 недели после изменений, за это время прежние копии будут затерты новыми.
Резервное копирование средствами платформы 1С-Битрикс
У провайдеров есть возможность создавать и управлять резервными копиями. Несмотря на это надежнее в месте с этим использовать встроенные средства архивации. Это бывает удобно, когда нужно дать доступ сотруднику только к панели администратора CMS.
Такой подход позволит управлять количеством резервных копий, объемом независимо от лимитов тарифного плана провайдера виртуального хостинга.
.png)
Возможности встроенного инструмента для создания бэкапов в 1С-Битрикс:
Если размер несжатых данных превышает 1 Гбайт, система разобьет дамп резервной копии на две и более частей. Можно сделать резервную копию сайта Битрикс с размещением на локальном или облачном хранилище. Главное, разделить места хранения для повышения сохранности ресурса, но с учетом удобства обратного копирования. При обнаружении неполадок нужно иметь возможность быстро исправить их развертыванием даже «очень старой» копией.
Сохранение архива в «локальную» папку
Существует три способа сделать копию сайта Битрикс:
В первом случае достаточно выбрать пункт меню «Резервное копирование», расположенный по пути – Настройки> Инструменты. Вот здесь впервые пользователь сталкивается с условиями компании 1С-Битрикс по предоставлению дискового пространства.
.png)
Основные моменты для локальных резервных копий сайта:
Так, на «Старте» и «Стандарте» выделяется всего 2 Гбайт, для «Малого бизнеса» лимит увеличен до 4 Гбайт, для «Бизнеса» до 10 Гбайт. На тарифе «Энтерпрайз» пространство под бэкапы ничем не ограничено. Выбрать место расположения архива можно после включения режима «экспертных настроек», он доступен в закладке «Параметры» и активируется галочкой в соответствующем пункте. Там же указывают, какие файлы исключить из резервирования.
.png)
Правила применения масок исключения:
Здесь же можно исключить из БД статистику, поисковый индекс, журнал событий, файлы размером свыше указанного. Чуть ниже выбирают, шифровать данные резервной копии или нет, проверять ли архив после создания. Последняя операция аналогична распаковке при восстановлении только реально файлы не создаются, а осуществляется их сверка с оригиналом. Архивировать ядро нужно обязательно, если лицензия еще не активирована.
.png)
При шифровании применяется алгоритм openssl_encrypt. Его можно отключить только при выборе локального хранилища, в «облаках» архивы хранятся только в зашифрованном виде. Такой подход необходим для поддержания заявленного уровня безопасности, все-таки сторонние хранилища – это определенный риск доступа к информации посторонних. Заданный пользователем пароль нужно хранить в надежном месте, потому что восстановить его при утрате не получится.
Сохранение резервной копии в облачном хранилище
При выборе в качестве места хранения «облака», предоставляемого разработчиком, понадобится установить модуль «Облачные хранилища (Clouds)» или убедиться в его наличии. То же относится к «облакам» сторонних разработчиков. Перед первым резервированием желательно уточнить, есть ли подключение с сервисом. В первом случае также понадобится легитимный коммерческий ключ и установленный модуль «Облако 1С-Битрикс (Bitrixcloud)».
.png)
Особенности:
Последнее зависит от встроенного алгоритма – файл всегда сначала создается локально и только затем переносится в «облако». При временных сбоях дамп можно найти по стандартному пути, в каталоге /bitrix/backup. После восстановления связи, когда система все-таки сможет скопировать сайт Битрикс в хранилище, тот будет автоматически удален. Физически хранилище «1С-Битрикс» расположено на сторонних серверах Amazon S3, поэтому приходится учитывать риски сбоев.
Особенности резервирования при многосайтовости
Сохранять в архиве можно любое количество сайтов, достаточно отметить нужные в перечне. Но со стороны администратора понадобится определенный перечень действий, чтобы сэкономить место на локальном накопителе или в «облаке». Например, нужно учитывать, что база данных копируется полностью, включая содержимое сайтов, которые не отмечены. Поэтому нет смысла создавать раздельные копии, они только займут «лишнее место».
Есть еще особенности:
Только после этого остальные сайты будут работоспособны. Подобные нюансы лучше учитывать еще на этапе развертывания, чтобы, по возможности, настроить систему на максимальное быстрое восстановление. Так, многосайтовость требует определенных настроек веб-сервера Apache, чтобы файловая система обязательно была не FAT32, т.к. она не поддерживает символьные ссылки и т.д. (это тема для отдельной статьи, поэтому здесь приводим лишь краткую информацию).
Восстановление сайта на 1С-Битрикс
.png)
Заходим в админку Битрикс и нажимаем на настройки.
Далее переходим в инструменты, в резервное копирование. В выпадающем списке можно создать резервное копирование вручную, просмотреть уже существующие копии и журнал изменений. При оплаченном аккаунте резервную копию можно поместить в облаке «1-С Битрикс». Она будет храниться не на вашем хостинге, а на серверах Битрикс.
При лицензии сервер даёт 2 ГБ, то есть, можно сохранить всего 2-3 раза. Чтобы сделать несколько копий, лучше использовать сохранение на компьютере.
Нажимаем «Создать резервную копию». Как раз здесь можно выбрать, где мы будем сохранять данные сайта: в облаке «1-С Битрикс» или в папке сайта.
При необходимости перейдите в «экспертные настройки», в «параметры». Там можно отрегулировать сохранение данных с сайта.
Например, если нужно исключить из архива скачивания базу данных, и оставить только архивацию файлов. В этих же настройках задаётся максимальный размер несжатых данных в одной части архива и снижение нагрузки на процессор.
Ждём, пока закончится загрузка резервной копии. Она может быть всего 20 минут, если проект небольшой и весит несколько ГБ. Все зависит от размера и количества данных.
Для переноса резервной копии сайта на другой хостинг нужно поместить в папку нового сайта скрипт для восстановления. Скачиваем файл >/restore.php.
В корневую папку сайта поместим скачанный скрипт для восстановления. Для этого перейдем в архив сайта на компьютере.
Локальная копия готова, время создания — несколько минут, дальше мы можем перейти в список. Перемещаемся по папкам: bxsite — bitrix — backup. Копируем все файлы, кроме index.php. Вставляем в папке bxrestored.
Затем в строке браузера проверяем скрипт для восстановления. Набираем bxrestored.loc/. В выпадающем списке нажимаем на restore.php и галочкой отмечаем «Архив загружен в корневую папку сайта».
Файлы для восстановления базы данных готовы!
Скачивание архива через FTP
Выше вариант скачивания именно на базе системы Битрикс. Резервное копирование также можно сделать через FTP-клиента. Для этого:
-
Через FTP-клиента подключаемся к сайту. Так как через Битрикс или сам хостинг были созданы резервные копии, переходим к папке backup;
-
Здесь есть уже созданные архивы резервных копий. Если созданных резервных копий много, то можно посмотреть по дате самую свежую из них;
-
Выбираем нужную базу данных и просто скачиваем эти файлы себе на компьютер. Но перед скачиванием обязательно проверьте свободное место на своём хостинге под новый сайт.
-
Расшифровка резервных копий может быть только по лицензионному ключу и паролю. Ни в коем случае не теряйте их. Даже сотрудники Битрикса не смогут помочь, если нет пароля.
Где найти созданные копии в системе Битрикс?
После успешного создания резервная копия сайта будет расположена в списке резервных копий. Для этого нужно нажать на «Настройки», и в меню инструментов выбрать «Список резервных копий».
В этом списке также доступно восстановление любых версий сайта, которые были загружены. В том числе, можно скачать архив последнего бэкапа. Перед скачиванием проверьте версию копии, так как при восстановлении баз данных и файлы сайта будут перезаписаны тем, что указано в бэкапе.
Нюансы при переносе данных в Битрикс
Для снижения ошибок при резервном копировании, нужно учитывать особенности работы с системой:
-
Перед восстановлением данных стоит оценить нагрузку на сайт. Возможно придется закрыть публичную часть для снижения этой нагрузки.
-
Чтобы не занимать свободное место, из выгрузки нужно удалить папки /bitrix/backup/ и /bitrix/cache/.
-
После запуска процесса восстановления нужно дождаться завершения процесса. При параллельном восстановлении часть файлов может не отобразиться и занять больше времени при переносе.
-
Если сервер слабый, то ждите ошибок 502 и 504. В этом случае нужно рассмотреть другие варианты хостингов.
-
Формат *.tar.gz предназначен для распаковки файлом restore.php. Никакой другой архиватор не подойдёт, так как есть риск потерять часть данных с этого резервной копии. Модификации с rar, zip, 7z также повредят файлы.
-
При загрузке данных с Database Host: [localhost:31006] — допустимы только имена localhost:31006 или localhost:3 6448. Это базовые настройки и изменить их нельзя.
Восстановление сайта из резервной копии на новом хостинге
Чтобы с самого начала не сталкиваться с ошибками при переносе данных, нужно протестировать новый сервер на соответствие с настройками Битрикс. Делать это нужно через скрипт, который расположен на сервер-тесте. Этот скрипт постоянно обновляется, учитываются новые ошибки и обновления системы.
-
Скачиваем файл restore.php и переходим по ссылке вашего домена + restore.php.
-
Высветится стартовый экран для переноса данных, там поставьте галочку на «Скачать резервную копию с дальнего сайта».
-
После скачивания система запросит данные для восстановления базы и файлов.
-
Начнется выгрузка сайта. После этого можно удалить архив с копией и скриптами. Восстановление завершено.
Ручной перенос данных через систему rsync
Если сайт, который нужно перенести на новый сервер, имеет размер несколько сотен ГБ и много мелких файлов, то автоматическое копирование на сервер нецелесообразно. В этом случае нужно перейти на автоматический перенос и минимизировать ошибки, лучше перейти на ручную проверку. Это можно сделать через систему rsync, так как:
-
можно вручную найти мелкие ошибки и изменения в переносе данных, чтобы сразу их исправлять;
-
при таком переносе не нужно будет закрывать публичную часть сайта, а значит, вся работа будет проходить бесперебойно;
-
снижается нагрузка на сервер, в отличие от автоматического копирования на Битрикс. В этом случае можно работать с тяжелыми и большими сайтами.
Битрикс restore.php (Или как развернуть бекап Битрикс)
Чтобы восстановить сайт Битрикс из бекапа на новый хостинг или на локальный веб сервер достаточно скачать файл restore.php и запустить его из строки браузера.
Файл можно скачать с сайта Битрикс: www.1c-bitrix.ru/download/scripts/restore.php
После его нужно скопировать в корневую папку сайта и запустить.
А запустить введя такую строку в браузер (в случае локально установленного веб сервера): //localhost/restore.php
Количество показов: 21620
Возврат к списку
