diff --git a/phpBB/config/default/container/services_db.yml b/phpBB/config/default/container/services_db.yml index d538177603..c368a6f464 100644 --- a/phpBB/config/default/container/services_db.yml +++ b/phpBB/config/default/container/services_db.yml @@ -34,7 +34,6 @@ services: class: phpbb\db\extractor\mssql_extractor shared: false arguments: - - '%core.root_path%' - '@request' - '@dbal.conn.driver' @@ -42,7 +41,6 @@ services: class: phpbb\db\extractor\mysql_extractor shared: false arguments: - - '%core.root_path%' - '@request' - '@dbal.conn.driver' @@ -50,7 +48,6 @@ services: class: phpbb\db\extractor\oracle_extractor shared: false arguments: - - '%core.root_path%' - '@request' - '@dbal.conn.driver' @@ -58,7 +55,6 @@ services: class: phpbb\db\extractor\postgres_extractor shared: false arguments: - - '%core.root_path%' - '@request' - '@dbal.conn.driver' @@ -66,6 +62,5 @@ services: class: phpbb\db\extractor\sqlite3_extractor shared: false arguments: - - '%core.root_path%' - '@request' - '@dbal.conn.driver' diff --git a/phpBB/config/default/container/services_storage.yml b/phpBB/config/default/container/services_storage.yml index f114dc1dfe..7216b4b5c4 100644 --- a/phpBB/config/default/container/services_storage.yml +++ b/phpBB/config/default/container/services_storage.yml @@ -17,6 +17,14 @@ services: tags: - { name: storage } + storage.backup: + class: phpbb\storage\storage + arguments: + - '@storage.adapter.factory' + - 'backup' + tags: + - { name: storage } + # Factory storage.adapter.factory: class: phpbb\storage\adapter_factory diff --git a/phpBB/config/default/container/tables.yml b/phpBB/config/default/container/tables.yml index 4aed35710b..6adacca0d0 100644 --- a/phpBB/config/default/container/tables.yml +++ b/phpBB/config/default/container/tables.yml @@ -8,6 +8,7 @@ parameters: tables.auth_provider_oauth_token_storage: '%core.table_prefix%oauth_tokens' tables.auth_provider_oauth_states: '%core.table_prefix%oauth_states' tables.auth_provider_oauth_account_assoc: '%core.table_prefix%oauth_accounts' + tables.backups: '%core.table_prefix%backups' tables.banlist: '%core.table_prefix%banlist' tables.bbcodes: '%core.table_prefix%bbcodes' tables.bookmarks: '%core.table_prefix%bookmarks' diff --git a/phpBB/includes/acp/acp_database.php b/phpBB/includes/acp/acp_database.php index b98756a34b..ea1656d297 100644 --- a/phpBB/includes/acp/acp_database.php +++ b/phpBB/includes/acp/acp_database.php @@ -27,9 +27,10 @@ class acp_database function main($id, $mode) { global $cache, $db, $user, $template, $table_prefix, $request; - global $phpbb_root_path, $phpbb_container, $phpbb_log; + global $phpbb_root_path, $phpbb_container, $phpbb_log, $table_prefix; $this->db_tools = $phpbb_container->get('dbal.tools'); + $storage = $phpbb_container->get('storage.backup'); $user->add_lang('acp/database'); @@ -98,50 +99,103 @@ class acp_database $filename = 'backup_' . $time . '_' . unique_id(); - $extractor = $phpbb_container->get('dbal.extractor'); - $extractor->init_extractor($format, $filename, $time, $download, $store); - - $extractor->write_start($table_prefix); - - foreach ($table as $table_name) + try { - // Get the table structure - if ($structure) + $extractor = $phpbb_container->get('dbal.extractor'); + $extractor->init_extractor($format, $filename, $time, $download, $store); + + $extractor->write_start($table_prefix); + + foreach ($table as $table_name) { - $extractor->write_table($table_name); - } - else - { - // We might wanna empty out all that junk :D - switch ($db->get_sql_layer()) + // Get the table structure + if ($structure) { - case 'sqlite3': - $extractor->flush('DELETE FROM ' . $table_name . ";\n"); - break; + $extractor->write_table($table_name); + } + else + { + // We might wanna empty out all that junk :D + switch ($db->get_sql_layer()) + { + case 'sqlite3': + $extractor->flush('DELETE FROM ' . $table_name . ";\n"); + break; - case 'mssql_odbc': - case 'mssqlnative': - $extractor->flush('TRUNCATE TABLE ' . $table_name . "GO\n"); - break; + case 'mssql_odbc': + case 'mssqlnative': + $extractor->flush('TRUNCATE TABLE ' . $table_name . "GO\n"); + break; - case 'oracle': - $extractor->flush('TRUNCATE TABLE ' . $table_name . "/\n"); - break; + case 'oracle': + $extractor->flush('TRUNCATE TABLE ' . $table_name . "/\n"); + break; - default: - $extractor->flush('TRUNCATE TABLE ' . $table_name . ";\n"); - break; + default: + $extractor->flush('TRUNCATE TABLE ' . $table_name . ";\n"); + break; + } + } + + // Data + if ($schema_data) + { + $extractor->write_data($table_name); } } - // Data - if ($schema_data) - { - $extractor->write_data($table_name); - } + $extractor->write_end(); + } + catch (\phpbb\exception\runtime_exception $e) + { + trigger_error($e->getMessage(), E_USER_ERROR); } - $extractor->write_end(); + try + { + if ($store) + { + // Get file name + switch ($format) + { + case 'text': + $ext = '.sql'; + break; + case 'bzip2': + $ext = '.sql.gz2'; + break; + case 'gzip': + $ext = '.sql.gz'; + break; + } + + $file = $filename . $ext; + + // Copy to storage using streams + $fp = fopen(sys_get_temp_dir() . '/' . $file, 'rb'); + + if ($fp === false) + { + throw new \phpbb\exception\runtime_exception('CANNOT_OPEN_FILE'); + } + + $storage->write_stream($file, $fp); + + fclose($fp); + + // Remove file from tmp + @unlink(sys_get_temp_dir() . '/' . $file); + + // Save to database + $sql = "INSERT INTO " . $table_prefix . "backups (filename) + VALUES ('$file');"; + $db->sql_query($sql); + } + } + catch (\phpbb\exception\runtime_exception $e) + { + trigger_error($e->getMessage(), E_USER_ERROR); + } $phpbb_log->add('admin', $user->data['user_id'], $user->ip, 'LOG_DB_BACKUP'); @@ -208,9 +262,9 @@ class acp_database trigger_error($user->lang['BACKUP_INVALID'] . adm_back_link($this->u_action), E_USER_WARNING); } - $file_name = $phpbb_root_path . 'store/' . $matches[0]; + $file_name = $matches[0]; - if (!file_exists($file_name) || !is_readable($file_name)) + if (!$storage->exists($file_name)) { trigger_error($user->lang['BACKUP_INVALID'] . adm_back_link($this->u_action), E_USER_WARNING); } @@ -219,8 +273,24 @@ class acp_database { if (confirm_box(true)) { - unlink($file_name); - $phpbb_log->add('admin', $user->data['user_id'], $user->ip, 'LOG_DB_DELETE'); + try + { + // Delete from storage + $storage->delete($file_name); + + // Add log entry + $phpbb_log->add('admin', $user->data['user_id'], $user->ip, 'LOG_DB_DELETE'); + + // Remove from database + $sql = "DELETE FROM " . $table_prefix . "backups + WHERE filename = '" . $file_name . "';"; + $db->sql_query($sql); + } + catch (\Exception $e) + { + trigger_error($user->lang['BACKUP_ERROR'] . adm_back_link($this->u_action), E_USER_WARNING); + } + trigger_error($user->lang['BACKUP_DELETE'] . adm_back_link($this->u_action)); } else @@ -253,25 +323,51 @@ class acp_database @set_time_limit(0); - $fp = @fopen($file_name, 'rb'); - - if ($fp !== false) + try { + $fp = $storage->read_stream($file_name); + while (!feof($fp)) { echo fread($fp, 8192); } fclose($fp); } + catch (\phpbb\storage\exception\exception $e) + { + + } flush(); exit; } + // Copy file to temp folder to decompress it + $temp_file_name = sys_get_temp_dir() . '/' . $file_name; + + try + { + $stream = $storage->read_stream($file_name); + $fp = fopen($temp_file_name, 'w+b'); + + while (!feof($stream)) + { + fwrite($fp, fread($stream, 8192)); + } + + fclose($fp); + fclose($stream); + } + catch (\phpbb\storage\exception\exception $e) + { + // Error while copying file from storage to temp + exit; + } + switch ($matches[1]) { case 'sql': - $fp = fopen($file_name, 'rb'); + $fp = fopen($temp_file_name, 'rb'); $read = 'fread'; $seek = 'fseek'; $eof = 'feof'; @@ -280,7 +376,7 @@ class acp_database break; case 'sql.bz2': - $fp = bzopen($file_name, 'r'); + $fp = bzopen($temp_file_name, 'r'); $read = 'bzread'; $seek = ''; $eof = 'feof'; @@ -289,7 +385,7 @@ class acp_database break; case 'sql.gz': - $fp = gzopen($file_name, 'rb'); + $fp = gzopen($temp_file_name, 'rb'); $read = 'gzread'; $seek = 'gzseek'; $eof = 'gzeof'; @@ -368,6 +464,8 @@ class acp_database $close($fp); + @unlink($temp_file_name); + // Purge the cache due to updated data $cache->purge(); @@ -393,26 +491,25 @@ class acp_database $methods[] = $type; } - $dir = $phpbb_root_path . 'store/'; - $dh = @opendir($dir); + $sql = 'SELECT filename + FROM ' . BACKUPS_TABLE; + $result = $db->sql_query($sql); $backup_files = array(); - if ($dh) + while ($row = $db->sql_fetchrow($result)) { - while (($file = readdir($dh)) !== false) + if (preg_match('#^backup_(\d{10,})_[a-z\d]{16}\.(sql(?:\.(?:gz|bz2))?)$#', $row['filename'], $matches)) { - if (preg_match('#^backup_(\d{10,})_[a-z\d]{16}\.(sql(?:\.(?:gz|bz2))?)$#', $file, $matches)) + if (in_array($matches[2], $methods)) { - if (in_array($matches[2], $methods)) - { - $backup_files[(int) $matches[1]] = $file; - } + $backup_files[(int) $matches[1]] = $row['filename']; } } - closedir($dh); } + $db->sql_freeresult($result); + if (!empty($backup_files)) { krsort($backup_files); diff --git a/phpBB/includes/constants.php b/phpBB/includes/constants.php index 2fa11e4d86..ee8b642b9b 100644 --- a/phpBB/includes/constants.php +++ b/phpBB/includes/constants.php @@ -239,6 +239,7 @@ define('ACL_ROLES_DATA_TABLE', $table_prefix . 'acl_roles_data'); define('ACL_ROLES_TABLE', $table_prefix . 'acl_roles'); define('ACL_USERS_TABLE', $table_prefix . 'acl_users'); define('ATTACHMENTS_TABLE', $table_prefix . 'attachments'); +define('BACKUPS_TABLE', $table_prefix . 'backups'); define('BANLIST_TABLE', $table_prefix . 'banlist'); define('BBCODES_TABLE', $table_prefix . 'bbcodes'); define('BOOKMARKS_TABLE', $table_prefix . 'bookmarks'); diff --git a/phpBB/install/schemas/schema_data.sql b/phpBB/install/schemas/schema_data.sql index c6a60af67b..14c63fab72 100644 --- a/phpBB/install/schemas/schema_data.sql +++ b/phpBB/install/schemas/schema_data.sql @@ -291,6 +291,8 @@ INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\attachment INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\attachment\config\path', 'files'); INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\avatar\provider', 'phpbb\storage\provider\local'); INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\avatar\config\path', 'images/avatars/upload'); +INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\backup\provider', 'phpbb\storage\provider\local'); +INSERT INTO phpbb_config (config_name, config_value) VALUES ('storage\backup\config\path', 'store'); INSERT INTO phpbb_config (config_name, config_value, is_dynamic) VALUES ('cache_last_gc', '0', 1); INSERT INTO phpbb_config (config_name, config_value, is_dynamic) VALUES ('cron_lock', '0', 1); diff --git a/phpBB/language/en/acp/database.php b/phpBB/language/en/acp/database.php index ab85701eaa..ac5b19d5d9 100644 --- a/phpBB/language/en/acp/database.php +++ b/phpBB/language/en/acp/database.php @@ -43,6 +43,7 @@ $lang = array_merge($lang, array( 'BACKUP_DELETE' => 'The backup file has been deleted successfully.', 'BACKUP_INVALID' => 'The selected file to backup is invalid.', + 'BACKUP_ERROR' => 'Error when saving file.', 'BACKUP_OPTIONS' => 'Backup options', 'BACKUP_SUCCESS' => 'The backup file has been created successfully.', 'BACKUP_TYPE' => 'Backup type', diff --git a/phpBB/phpbb/db/extractor/base_extractor.php b/phpBB/phpbb/db/extractor/base_extractor.php index 547c85f066..2803b89357 100644 --- a/phpBB/phpbb/db/extractor/base_extractor.php +++ b/phpBB/phpbb/db/extractor/base_extractor.php @@ -21,10 +21,6 @@ use phpbb\db\extractor\exception\extractor_not_initialized_exception; */ abstract class base_extractor implements extractor_interface { - /** - * @var string phpBB root path - */ - protected $phpbb_root_path; /** * @var \phpbb\request\request_interface @@ -84,13 +80,11 @@ abstract class base_extractor implements extractor_interface /** * Constructor * - * @param string $phpbb_root_path * @param \phpbb\request\request_interface $request * @param \phpbb\db\driver\driver_interface $db */ - public function __construct($phpbb_root_path, \phpbb\request\request_interface $request, \phpbb\db\driver\driver_interface $db) + public function __construct(\phpbb\request\request_interface $request, \phpbb\db\driver\driver_interface $db) { - $this->phpbb_root_path = $phpbb_root_path; $this->request = $request; $this->db = $db; $this->fp = null; @@ -164,13 +158,13 @@ abstract class base_extractor implements extractor_interface if ($store === true) { - $file = $this->phpbb_root_path . 'store/' . $filename . $ext; + $file = sys_get_temp_dir() . '/' . $filename . $ext; $this->fp = $open($file, 'w'); if (!$this->fp) { - trigger_error('FILE_WRITE_FAIL', E_USER_ERROR); + throw new \phpbb\exception\runtime_exception('FILE_WRITE_FAIL'); } } diff --git a/phpBB/phpbb/db/migration/data/v330/storage_backup.php b/phpBB/phpbb/db/migration/data/v330/storage_backup.php new file mode 100644 index 0000000000..accdfb0575 --- /dev/null +++ b/phpBB/phpbb/db/migration/data/v330/storage_backup.php @@ -0,0 +1,40 @@ + +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\db\migration\data\v330; + +class storage_backup extends \phpbb\db\migration\migration +{ + public function update_schema() + { + return array( + 'add_tables' => array( + $this->table_prefix . 'backups' => array( + 'COLUMNS' => array( + 'backup_id' => array('UINT', NULL, 'auto_increment'), + 'filename' => array('VCHAR', ''), + ), + 'PRIMARY_KEY' => 'backup_id', + ), + ), + ); + } + + public function update_data() + { + return array( + array('config.add', array('storage\\backup\\provider', \phpbb\storage\provider\local::class)), + array('config.add', array('storage\\backup\\config\\path', 'store')), + ); + } +}