settings = ITSEC_Modules::get_settings( 'backup' );
add_action( 'itsec_execute_backup_cron', array( $this, 'do_backup' ) );
add_filter( 'itsec_notifications', array( $this, 'register_notification' ) );
add_filter( 'itsec_backup_notification_strings', array( $this, 'notification_strings' ) );
if ( class_exists( 'pb_backupbuddy' ) ) {
// Don't run when BackupBuddy is active.
return;
}
ITSEC_Core::get_scheduler()->register_custom_schedule( 'backup', DAY_IN_SECONDS * $this->settings['interval'] );
add_action( 'itsec_scheduler_register_events', array( $this, 'register_events' ) );
if ( ! $this->settings['enabled'] || $this->settings['interval'] <= 0 ) {
// Don't run when scheduled backups aren't enabled or the interval is zero or less.
return;
}
add_action( 'itsec_scheduled_backup', array( $this, 'scheduled_callback' ) );
}
/**
* Called when it is time for the backup to run.
*/
public function scheduled_callback() {
$this->do_backup();
}
/**
* Public function to get lock and call backup.
*
* Attempts to get a lock to prevent concurrant backups and calls the backup function itself.
*
* @since 4.0.0
*
* @param boolean $one_time whether this is a one time backup
*
* @return mixed false on error or nothing
*/
public function do_backup( $one_time = false ) {
if ( ! ITSEC_Lib::get_lock( 'backup', 180 ) ) {
return new WP_Error( 'itsec-backup-do-backup-already-running', __( 'Unable to create a backup at this time since a backup is currently being created. If you wish to create an additional backup, please wait a few minutes before trying again.', 'better-wp-security' ) );
}
ITSEC_Lib::set_minimum_memory_limit( '256M' );
$this->execute_backup( $one_time );
ITSEC_Lib::release_lock( 'backup' );
switch ( $this->settings['method'] ) {
case 0:
return __( 'Backup complete. The backup was sent to the selected email recipients and was saved locally.', 'better-wp-security' );
case 1:
return __( 'Backup complete. The backup was sent to the selected email recipients.', 'better-wp-security' );
default:
return __( 'Backup complete. The backup was saved locally.', 'better-wp-security' );
}
}
/**
* Executes backup function.
*
* Handles the execution of database backups.
*
* @since 4.0.0
*
* @param bool $one_time whether this is a one-time backup
*
* @return void
*/
private function execute_backup( $one_time = false ) {
global $wpdb;
require_once( ITSEC_Core::get_core_dir() . 'lib/class-itsec-lib-directory.php' );
$dir = $this->settings['location'];
$result = ITSEC_Lib_Directory::create( $dir );
if ( is_wp_error( $result ) ) {
return $result;
} else if ( ! $result ) {
return new WP_Error( 'itsec-backup-failed-to-create-backup-dir', esc_html__( 'Unable to create the backup directory due to an unknown error.', 'better-wp-security' ) );
}
$file = "$dir/backup-" . substr( sanitize_title( get_bloginfo( 'name' ) ), 0, 20 ) . '-' . current_time( 'Ymd-His' ) . '-' . wp_generate_password( 30, false ) . '.sql';
if ( false === ( $fh = @fopen( $file, 'w' ) ) ) {
return new WP_Error( 'itsec-backup-failed-to-write-backup-file', esc_html__( 'Unable to write the backup file. This may be due to a permissions or disk space issue.', 'better-wp-security' ) );
}
if ( false === $one_time ) {
ITSEC_Modules::set_setting( 'backup', 'last_run', ITSEC_Core::get_current_time_gmt() );
}
if ( $this->settings['all_sites'] ) {
$tables = $wpdb->get_col( 'SHOW TABLES' );
} else {
$tables = $wpdb->get_col( $wpdb->prepare( 'SHOW TABLES LIKE %s', $wpdb->base_prefix . '%' ) );
}
$max_rows_per_query = 1000;
foreach ( $tables as $table ) {
$create_table = $wpdb->get_var( "SHOW CREATE TABLE `$table`;", 1 ) . ';' . PHP_EOL . PHP_EOL;
$create_table = preg_replace( '/^CREATE TABLE /', 'CREATE TABLE IF NOT EXISTS ', $create_table );
@fwrite( $fh, $create_table );
if ( in_array( substr( $table, strlen( $wpdb->prefix ) ), $this->settings['exclude'] ) ) {
// User selected to exclude the data from this table.
fwrite( $fh, PHP_EOL . PHP_EOL );
continue;
}
$num_fields = count( $wpdb->get_results( "DESCRIBE `$table`;" ) );
$offset = 0;
$has_more_rows = true;
while ( $has_more_rows ) {
$rows = $wpdb->get_results( $wpdb->prepare( "SELECT * FROM `$table` LIMIT %d, %d", $offset, $max_rows_per_query ), ARRAY_N );
foreach ( $rows as $row ) {
$sql = "INSERT INTO `$table` VALUES (";
for ( $j = 0; $j < $num_fields; $j ++ ) {
if ( isset( $row[$j] ) ) {
$row[$j] = addslashes( $row[$j] );
if ( PHP_EOL !== "\n" ) {
$row[$j] = preg_replace( '#' . PHP_EOL . '#', "\n", $row[$j] );
}
$sql .= '"' . $row[$j] . '"';
} else {
$sql .= '""';
}
if ( $j < ( $num_fields - 1 ) ) {
$sql .= ',';
}
}
$sql .= ");" . PHP_EOL;
@fwrite( $fh, $sql );
}
if ( count( $rows ) < $max_rows_per_query ) {
$has_more_rows = false;
} else {
$offset += $max_rows_per_query;
}
}
@fwrite( $fh, PHP_EOL . PHP_EOL );
}
@fwrite( $fh, PHP_EOL . PHP_EOL );
@fclose( $fh );
$backup_file = $file;
if ( $this->settings['zip'] ) {
if ( ! class_exists( 'PclZip' ) ) {
require( ABSPATH . 'wp-admin/includes/class-pclzip.php' );
}
$zip_file = substr( $file, 0, -4 ) . '.zip';
$pclzip = new PclZip( $zip_file );
if ( 0 != $pclzip->create( $file, PCLZIP_OPT_REMOVE_PATH, $dir ) ) {
@unlink( $file );
$file = $zip_file;
}
}
if ( 2 !== $this->settings['method'] || true === $one_time ) {
$mail_success = $this->send_mail( $file );
} else {
$mail_success = null;
}
if ( 1 === $this->settings['method'] ) {
@unlink( $file );
} else if ( $this->settings['retain'] > 0 ) {
$files = scandir( $dir, 1 );
if ( is_array( $files ) && count( $files ) > 0 ) {
$count = 0;
foreach ( $files as $file ) {
if ( ! strstr( $file, 'backup' ) ) {
continue;
}
if ( $count >= $this->settings['retain'] ) {
@unlink( trailingslashit( $dir ) . $file );
}
$count++;
}
}
}
$log_data = array(
'settings' => $this->settings,
'mail_success' => $mail_success,
'file' => $backup_file,
);
if ( 0 === $this->settings['method'] ) {
if ( false === $mail_success ) {
ITSEC_Log::add_warning( 'backup', 'email-failed-file-stored', $log_data );
} else {
ITSEC_Log::add_notice( 'backup', 'email-succeeded-file-stored', $log_data );
}
} else if ( 1 === $this->settings['method'] ) {
if ( false === $mail_success ) {
ITSEC_Log::add_error( 'backup', 'email-failed', $log_data );
} else {
ITSEC_Log::add_notice( 'backup', 'email-succeeded', $log_data );
}
} else {
ITSEC_Log::add_notice( 'backup', 'file-stored', $log_data );
}
}
private function send_mail( $file ) {
$nc = ITSEC_Core::get_notification_center();
$mail = $nc->mail();
$mail->add_header( esc_html__( 'Database Backup', 'better-wp-security' ), sprintf( esc_html__( 'Site Database Backup for %s', 'better-wp-security' ), '' . date_i18n( get_option( 'date_format' ) ) . '' ) );
$mail->add_info_box( esc_html__( 'Attached is the database backup file for your site.', 'better-wp-security' ), 'attachment' );
$mail->add_section_heading( esc_html__( 'Website', 'better-wp-security' ) );
$mail->add_text( $mail->get_display_url() );
$mail->add_section_heading( esc_html__( 'Date', 'better-wp-security' ) );
$mail->add_text( esc_html( date_i18n( get_option( 'date_format' ) ) ) );
$mail->add_footer();
$mail->set_recipients( $nc->get_recipients( 'backup' ) );
$subject = $mail->prepend_site_url_to_subject( $nc->get_subject( 'backup' ) );
$subject = apply_filters( 'itsec_backup_email_subject', $subject );
$mail->set_subject( $subject, false );
$mail->add_attachment( $file );
return $nc->send( 'backup', $mail );
}
/**
* Register the events.
*
* @param ITSEC_Scheduler $scheduler
*/
public function register_events( $scheduler ) {
$settings = ITSEC_Modules::get_settings( 'backup' );
if ( $settings['enabled'] && $settings['interval'] > 0 ) {
$scheduler->schedule( 'backup', 'backup' );
}
}
/**
* Register the Backup notification email.
*
* @param array $notifications
*
* @return array
*/
public function register_notification( $notifications ) {
$method = ITSEC_Modules::get_setting( 'backup', 'method' );
if ( 0 === $method || 1 === $method ) {
$notifications['backup'] = array(
'subject_editable' => true,
'recipient' => ITSEC_Notification_Center::R_EMAIL_LIST,
'schedule' => ITSEC_Notification_Center::S_NONE,
'module' => 'backup',
);
}
return $notifications;
}
/**
* Register the strings for the Backup email.
*
* @return array
*/
public function notification_strings() {
return array(
'label' => esc_html__( 'Database Backup', 'better-wp-security' ),
'description' => sprintf( esc_html__( 'The %1$sDatabase Backup%2$s module will send a copy of any backups to the email addresses listed below.', 'better-wp-security' ), '', '' ),
'subject' => esc_html__( 'Database Backup', 'better-wp-security' ),
);
}
}