'jobs', 'singular' => 'job', 'ajax' => TRUE ) ); } /** * @return bool|void */ public function ajax_user_can() { return current_user_can( 'backwpup' ); } /** * */ public function prepare_items() { $this->items = BackWPup_Option::get_job_ids(); $this->job_object = BackWPup_Job::get_working_data(); $this->job_types = BackWPup::get_job_types(); $this->destinations = BackWPup::get_registered_destinations(); if ( ! isset( $_GET[ 'order' ] ) || ! isset( $_GET[ 'orderby' ] ) ) { return; } if ( strtolower( $_GET[ 'order' ] ) === 'asc' ) { $order = SORT_ASC; } else { $order = SORT_DESC; } if ( empty( $_GET[ 'orderby' ] ) || ! in_array( strtolower( $_GET[ 'orderby' ] ), array( 'jobname', 'type', 'dest', 'next', 'last' ), true ) ) { $orderby = 'jobname'; } else { $orderby = strtolower( $_GET[ 'orderby' ] ); } //sorting $job_configs = array(); $i = 0; foreach( $this->items as $item ) { $job_configs[ $i ][ 'jobid' ] = $item; $job_configs[ $i ][ 'jobname' ] = BackWPup_Option::get( $item, 'name' ); $job_configs[ $i ][ 'type' ] = BackWPup_Option::get( $item, 'type' ); $job_configs[ $i ][ 'dest' ] = BackWPup_Option::get( $item, 'destinations' ); if ( $order === SORT_ASC ) { sort( $job_configs[ $i ][ 'type' ] ); sort( $job_configs[ $i ][ 'dest' ] ); } else { rsort( $job_configs[ $i ][ 'type' ] ); rsort( $job_configs[ $i ][ 'dest' ] ); } $job_configs[ $i ][ 'type' ] = array_shift( $job_configs[ $i ][ 'type' ] ); $job_configs[ $i ][ 'dest' ] = array_shift( $job_configs[ $i ][ 'dest' ] ); $job_configs[ $i ][ 'next' ] = (int) wp_next_scheduled( 'backwpup_cron', array( 'id' => $item ) ); $job_configs[ $i ][ 'last' ] = BackWPup_Option::get( $item, 'lastrun' ); $i++; } $tmp = array(); foreach ( $job_configs as &$ma ) { $tmp[] = &$ma[ $orderby ]; } array_multisort( $tmp, $order, $job_configs ); $this->items = array(); foreach( $job_configs as $item ) { $this->items[] = $item[ 'jobid' ]; } } /** * */ public function no_items() { _e( 'No Jobs.', 'backwpup' ); } /** * @return array */ public function get_bulk_actions() { if ( ! $this->has_items() ) { return array(); } $actions = array(); $actions[ 'delete' ] = __( 'Delete', 'backwpup' ); return apply_filters( 'backwpup_page_jobs_get_bulk_actions', $actions ); } /** * @return array */ public function get_columns() { $jobs_columns = array(); $jobs_columns[ 'cb' ] = ''; $jobs_columns[ 'jobname' ] = __( 'Job Name', 'backwpup' ); $jobs_columns[ 'type' ] = __( 'Type', 'backwpup' ); $jobs_columns[ 'dest' ] = __( 'Destinations', 'backwpup' ); $jobs_columns[ 'next' ] = __( 'Next Run', 'backwpup' ); $jobs_columns[ 'last' ] = __( 'Last Run', 'backwpup' ); return $jobs_columns; } /** * @return array */ public function get_sortable_columns() { return array( 'jobname' => 'jobname', 'type' => 'type', 'dest' => 'dest', 'next' => 'next', 'last' => 'last', ); } /** * The cb Column * * @param $item * @return string */ public function column_cb( $item ) { return ''; } /** * The jobname Column * * @param $item * @return string */ public function column_jobname( $item ) { $job_normal_hide =''; if ( is_object( $this->job_object ) ) { $job_normal_hide = ' style="display:none;"'; } $r = '' . esc_html( BackWPup_Option::get( $item, 'name' ) ) . ''; $actions = array(); if ( current_user_can( 'backwpup_jobs_edit' ) ) { $actions[ 'edit' ] = "" . esc_html__( 'Edit', 'backwpup' ) . ""; $actions[ 'copy' ] = "" . esc_html__( 'Copy', 'backwpup' ) . ""; $actions[ 'delete' ] = "" . esc_html__( 'Delete', 'backwpup' ) . ""; } if ( current_user_can( 'backwpup_jobs_start' ) ) { $url = BackWPup_Job::get_jobrun_url( 'runnowlink', $item ); $actions[ 'runnow' ] = "" . esc_html__( 'Run now', 'backwpup' ) . ""; } if ( current_user_can( 'backwpup_logs' ) && BackWPup_Option::get( $item, 'logfile' ) ) { $logfile = basename( BackWPup_Option::get( $item, 'logfile' ) ); if ( is_object( $this->job_object ) && $this->job_object->job[ 'jobid' ] == $item ) { $logfile = basename( $this->job_object->logfile ); } $log_name = str_replace( array( '.html', '.gz' ), '', basename( $logfile ) ); $actions[ 'lastlog' ] = '' . __( 'Last log', 'backwpup' ) . ''; } $actions = apply_filters( 'backwpup_page_jobs_actions', $actions, $item, FALSE ); $r .= '
' . $this->row_actions( $actions ) . '
'; if ( is_object( $this->job_object ) ) { $actionsrun = array(); $actionsrun = apply_filters( 'backwpup_page_jobs_actions', $actionsrun, $item, TRUE ); $r .= '
' . $this->row_actions( $actionsrun ) . '
'; } return $r; } /** * The type Column * * @param $item * @return string */ public function column_type( $item ) { $r = ''; if ( $types = BackWPup_Option::get( $item, 'type' ) ) { foreach ( $types as $type ) { if ( isset( $this->job_types[ $type ] ) ) { $r .= $this->job_types[ $type ]->info[ 'name' ] . '
'; } else { $r .= $type . '
'; } } } return $r; } /** * The destination Column * * @param $item * @return string */ public function column_dest( $item ) { $r = ''; $backup_to = FALSE; foreach ( BackWPup_Option::get( $item, 'type' ) as $typeid ) { if ( isset( $this->job_types[ $typeid ] ) && $this->job_types[ $typeid ]->creates_file() ) { $backup_to = TRUE; break; } } if ( $backup_to ) { foreach ( BackWPup_Option::get( $item, 'destinations' ) as $destid ) { if ( isset( $this->destinations[ $destid ][ 'info' ][ 'name' ] ) ) { $r .= $this->destinations[ $destid ][ 'info' ][ 'name' ] . '
'; } else { $r .= $destid . '
'; } } } else { $r .= '' . __( 'Not needed or set', 'backwpup' ) . '
'; } return $r; } /** * The next Column * * @param $item * @return string */ public function column_next( $item ) { $r = ''; $job_normal_hide =''; if ( is_object( $this->job_object ) ) { $job_normal_hide = ' style="display:none;"'; } if ( is_object( $this->job_object ) && $this->job_object->job[ 'jobid' ] == $item ) { $runtime = current_time( 'timestamp' ) - $this->job_object->start_time; $r .= '
' . sprintf( esc_html__( 'Running for: %s seconds', 'backwpup' ), '' . $runtime . '' ) .'
'; } if ( is_object( $this->job_object ) && $this->job_object->job[ 'jobid' ] == $item ) { $r .='
'; } if ( BackWPup_Option::get( $item, 'activetype' ) == 'wpcron' ) { if ( $nextrun = wp_next_scheduled( 'backwpup_cron', array( 'id' => $item ) ) + ( get_option( 'gmt_offset' ) * 3600 ) ) { $r .= '' . sprintf( __( '%1$s at %2$s by WP-Cron', 'backwpup' ) , date_i18n( get_option( 'date_format' ), $nextrun, TRUE ) , date_i18n( get_option( 'time_format' ), $nextrun, TRUE ) ) . '
'; } else { $r .= __( 'Not scheduled!', 'backwpup' ) . '
'; } } elseif ( BackWPup_Option::get( $item, 'activetype' ) == 'easycron' ) { $easycron_status = BackWPup_EasyCron::status( $item ); if ( !empty( $easycron_status ) ) { $nextrun = BackWPup_Cron::cron_next( $easycron_status[ 'cron_expression' ] ) + ( get_option( 'gmt_offset' ) * 3600 ); $r .= '' . sprintf( __( '%1$s at %2$s by EasyCron', 'backwpup' ) , date_i18n( get_option( 'date_format' ), $nextrun, TRUE ) , date_i18n( get_option( 'time_format' ), $nextrun, TRUE ) ) . '
'; } else { $r .= __( 'Not scheduled!', 'backwpup' ) . '
'; } } elseif ( BackWPup_Option::get( $item, 'activetype' ) == 'link' ) { $r .= __( 'External link', 'backwpup' ) . '
'; } else { $r .= __( 'Inactive', 'backwpup' ); } if ( is_object( $this->job_object ) && $this->job_object->job[ 'jobid' ] == $item ) { $r .= '
'; } return $r; } /** * The last Column * * @param $item * @return string */ public function column_last( $item ) { $r = ''; if ( BackWPup_Option::get( $item, 'lastrun' ) ) { $lastrun = BackWPup_Option::get( $item, 'lastrun' ); $r .= sprintf( __( '%1$s at %2$s', 'backwpup' ), date_i18n( get_option( 'date_format' ), $lastrun, TRUE ), date_i18n( get_option( 'time_format' ), $lastrun, TRUE ) ); if ( BackWPup_Option::get( $item, 'lastruntime' ) ) { $r .= '
' . sprintf( __( 'Runtime: %d seconds', 'backwpup' ), BackWPup_Option::get( $item, 'lastruntime' ) ); } } else { $r .= __( 'not yet', 'backwpup' ); } $r .= "
"; if ( current_user_can( 'backwpup_backups_download' ) ) { $download_url = BackWPup_Option::get( $item, 'lastbackupdownloadurl' ); if ( ! empty( $download_url ) ) { $r .= "" . esc_html__( 'Download', 'backwpup' ) . " | "; } } if ( current_user_can( 'backwpup_logs' ) && BackWPup_Option::get( $item, 'logfile' ) ) { $logfile = basename( BackWPup_Option::get( $item, 'logfile' ) ); if ( is_object( $this->job_object ) && $this->job_object->job[ 'jobid' ] == $item ) { $logfile = basename( $this->job_object->logfile ); } $log_name = str_replace( array( '.html', '.gz' ), '', basename( $logfile ) ); $r .= '' . esc_html__( 'Log', 'backwpup' ) . ''; } $r .= ""; return $r; } /** * */ public static function load() { //Create Table self::$listtable = new self; switch ( self::$listtable->current_action() ) { case 'delete': //Delete Job if ( ! current_user_can( 'backwpup_jobs_edit' ) ) { break; } if ( is_array( $_GET[ 'jobs' ] ) ) { check_admin_referer( 'bulk-jobs' ); foreach ( $_GET[ 'jobs' ] as $jobid ) { wp_clear_scheduled_hook( 'backwpup_cron', array( 'id' => absint( $jobid ) ) ); BackWPup_Option::delete_job( absint( $jobid ) ); } } break; case 'copy': //Copy Job if ( ! current_user_can( 'backwpup_jobs_edit' ) ) { break; } $old_job_id = absint( $_GET[ 'jobid' ] ); check_admin_referer( 'copy-job_' . $old_job_id ); //create new $newjobid = BackWPup_Option::get_job_ids(); sort( $newjobid ); $newjobid = end( $newjobid ) + 1; $old_options = BackWPup_Option::get_job( $old_job_id ); foreach ( $old_options as $key => $option ) { if ( $key === "jobid" ) $option = $newjobid; if ( $key === "name" ) $option = __( 'Copy of', 'backwpup' ) . ' ' . $option; if ( $key === "activetype" ) $option = ''; if ( $key === "archivename" ) $option = str_replace( $old_job_id, $newjobid, $option ); if ( $key === "logfile" || $key === "lastbackupdownloadurl" || $key === "lastruntime" || $key === "lastrun" ) continue; BackWPup_Option::update( $newjobid, $key, $option ); } break; case 'runnow': $jobid = absint( $_GET[ 'jobid' ] ); if ( $jobid ) { if ( ! current_user_can( 'backwpup_jobs_start' ) ) { wp_die( __( 'Sorry, you don\'t have permissions to do that.', 'backwpup') ); } check_admin_referer( 'backwpup_job_run-runnowlink' ); //check temp folder $temp_folder_message = BackWPup_File::check_folder( BackWPup::get_plugin_data( 'TEMP' ), TRUE ); BackWPup_Admin::message( $temp_folder_message, TRUE ); //check log folder $log_folder = get_site_option( 'backwpup_cfg_logfolder' ); $log_folder = BackWPup_File::get_absolute_path( $log_folder ); $log_folder_message = BackWPup_File::check_folder( $log_folder ); BackWPup_Admin::message( $log_folder_message, TRUE ); //check backup destinations $job_types = BackWPup::get_job_types(); $job_conf_types = BackWPup_Option::get( $jobid, 'type' ); $creates_file = FALSE; foreach ( $job_types as $id => $job_type_class ) { if ( in_array( $id, $job_conf_types, true ) && $job_type_class->creates_file( ) ) { $creates_file = TRUE; break; } } if ( $creates_file ) { $job_conf_dests = BackWPup_Option::get( $jobid, 'destinations' ); $destinations = 0; /* @var BackWPup_Destinations $dest_class */ foreach ( BackWPup::get_registered_destinations() as $id => $dest ) { if ( ! in_array( $id, $job_conf_dests, true ) || empty( $dest[ 'class' ] ) ) { continue; } $dest_class = BackWPup::get_destination( $id ); $job_settings = BackWPup_Option::get_job( $jobid ); if ( ! $dest_class->can_run( $job_settings ) ) { BackWPup_Admin::message( sprintf( __( 'The job "%s" destination "%s" is not configured properly','backwpup' ), esc_attr( BackWPup_Option::get( $jobid, 'name' ) ), $id ), TRUE ); } $destinations++; } if ( $destinations < 1 ) { BackWPup_Admin::message( sprintf( __( 'The job "%s" needs properly configured destinations to run!','backwpup' ), esc_attr( BackWPup_Option::get( $jobid, 'name' ) ) ), TRUE ); } } //only start job if messages empty $log_messages = BackWPup_Admin::get_messages(); if ( empty ( $log_messages ) ) { $old_log_file = BackWPup_Option::get( $jobid, 'logfile' ); BackWPup_Job::get_jobrun_url( 'runnow', $jobid ); usleep( 250000 ); //wait a quarter second $new_log_file = BackWPup_Option::get( $jobid, 'logfile', null, false ); //sleep as long as job not started $i=0; while ( $old_log_file === $new_log_file ) { usleep( 250000 ); //wait a quarter second for next try $new_log_file = BackWPup_Option::get( $jobid, 'logfile', null, false ); //wait maximal 10 sec. if ( $i >= 40 ) { BackWPup_Admin::message( sprintf( __( 'Job "%s" has started, but not responded for 10 seconds. Please check information.', 'backwpup' ), esc_attr( BackWPup_Option::get( $jobid, 'name' ) ), network_admin_url( 'admin.php' ) . '?page=backwpupsettings#backwpup-tab-information' ), true ); break 2; } $i++; } BackWPup_Admin::message( sprintf( __( 'Job "%s" started.', 'backwpup' ), esc_attr( BackWPup_Option::get( $jobid, 'name' ) ) ) ); } } break; case 'abort': //Abort Job if ( ! current_user_can( 'backwpup_jobs_start' ) ) break; check_admin_referer( 'abort-job' ); if ( ! file_exists( BackWPup::get_plugin_data( 'running_file' ) ) ) break; //abort BackWPup_Job::user_abort(); BackWPup_Admin::message( __( 'Job will be terminated.', 'backwpup' ) ) ; break; default: do_action( 'backwpup_page_jobs_load', self::$listtable->current_action() ); break; } self::$listtable->prepare_items(); } /** * */ public static function admin_print_styles() { ?> '; echo '

' . esc_html( sprintf( __( '%s › Jobs', 'backwpup' ), BackWPup::get_plugin_data( 'name' ) ) ) . ' ' . esc_html__( 'Add new', 'backwpup' ) . '

'; BackWPup_Admin::display_messages(); $job_object = BackWPup_Job::get_working_data(); if ( current_user_can( 'backwpup_jobs_start' ) && is_object( $job_object ) ) { //read existing logfile $logfiledata = file_get_contents( $job_object->logfile ); preg_match( '/]*>/si', $logfiledata, $match ); if ( ! empty( $match[ 0 ] ) ) $startpos = strpos( $logfiledata, $match[ 0 ] ) + strlen( $match[ 0 ] ); else $startpos = 0; $endpos = stripos( $logfiledata, '' ); if ( empty( $endpos ) ) $endpos = strlen( $logfiledata ); $length = strlen( $logfiledata ) - ( strlen( $logfiledata ) - $endpos ) - $startpos; ?>

job[ 'name' ] ) ); ?>

warnings; ?> errors; ?>
step_percent); ?>%
steps_data[ $job_object->step_working ][ 'NAME' ]); ?>
substep_percent); ?>%
lastmsg); ?>
display(); ?>
logfile ) ) { ?> warnings; $errors = $job_object->errors; $step_percent = $job_object->step_percent; $substep_percent = $job_object->substep_percent; $runtime = current_time( 'timestamp' ) - $job_object->start_time; $onstep = $job_object->steps_data[ $job_object->step_working ][ 'NAME' ]; $lastmsg = $job_object->lastmsg; $lasterrormsg = $job_object->lasterrormsg; } else { $logheader = BackWPup_Job::read_logheader( $logfile ); $warnings = $logheader[ 'warnings' ]; $runtime = $logheader[ 'runtime' ]; $errors = $logheader[ 'errors' ]; $step_percent = 100; $substep_percent = 100; $onstep = '

' . esc_html__( 'Job completed' , 'backwpup' ) . '

'; if ( $errors > 0 ) $lastmsg = '

' . esc_html__( 'ERROR:', 'backwpup' ) . ' ' . sprintf( esc_html__( 'Job has ended with errors in %s seconds. You must resolve the errors for correct execution.', 'backwpup' ), $logheader[ 'runtime' ] ) . '

'; elseif ( $warnings > 0 ) $lastmsg = '

' . esc_html__( 'WARNING:', 'backwpup' ) . ' ' . sprintf( esc_html__( 'Job has done with warnings in %s seconds. Please resolve them for correct execution.', 'backwpup' ), $logheader[ 'runtime' ] ) . '

'; else $lastmsg = '

' . sprintf( esc_html__( 'Job done in %s seconds.', 'backwpup' ), $logheader[ 'runtime' ] ) . '

'; $lasterrormsg = ''; $done = 1; } if ( '.gz' == substr( $logfile, -3 ) ) $logfiledata = file_get_contents( 'compress.zlib://' . $logfile, FALSE, NULL, $logpos ); else $logfiledata = file_get_contents( $logfile, FALSE, NULL, $logpos ); preg_match( '/]*>/si', $logfiledata, $match ); if ( ! empty( $match[ 0 ] ) ) $startpos = strpos( $logfiledata, $match[ 0 ] ) + strlen( $match[ 0 ] ); else $startpos = 0; $endpos = stripos( $logfiledata, '' ); if ( FALSE === $endpos ) $endpos = strlen( $logfiledata ); $length = strlen( $logfiledata ) - ( strlen( $logfiledata ) - $endpos ) - $startpos; //check if restart must done on ALTERNATE_WP_CRON if ( is_object( $job_object ) && defined( 'ALTERNATE_WP_CRON' ) && ALTERNATE_WP_CRON ) { $restart = BackWPup_Job::get_jobrun_url( 'restartalt' ); if ( $job_object->pid === 0 && $job_object->uniqid === '' ) { $restart_url = $restart[ 'url' ]; } $last_update = microtime( TRUE ) - $job_object->timestamp_last_update; if ( empty( $job_object->pid ) && $last_update > 10 ) { $restart_url = $restart[ 'url' ]; } } wp_send_json( array( 'log_pos' => strlen( $logfiledata ) + $logpos, 'log_text' => substr( $logfiledata, $startpos, $length ), 'warning_count' => $warnings, 'error_count' => $errors, 'running_time' => $runtime, 'step_percent' => $step_percent, 'on_step' => $onstep, 'last_msg' => $lastmsg, 'last_error_msg' => $lasterrormsg, 'sub_step_percent'=> $substep_percent, 'restart_url' => $restart_url, 'job_done' => $done ) ); } }