File: /var/www/html/wp-content/plugins/redirection/redirection-admin.php
<?php
include dirname( __FILE__ ) . '/models/group.php';
include dirname( __FILE__ ) . '/models/monitor.php';
include dirname( __FILE__ ) . '/models/file-io.php';
define( 'RED_DEFAULT_PER_PAGE', 25 );
define( 'RED_MAX_PER_PAGE', 200 );
class Redirection_Admin {
private static $instance = null;
private $monitor;
static function init() {
if ( is_null( self::$instance ) ) {
self::$instance = new Redirection_Admin();
}
return self::$instance;
}
function __construct() {
add_action( 'admin_menu', array( $this, 'admin_menu' ) );
add_action( 'plugin_action_links_' . basename( dirname( REDIRECTION_FILE ) ) . '/' . basename( REDIRECTION_FILE ), array( $this, 'plugin_settings' ), 10, 4 );
add_filter( 'redirection_save_options', array( $this, 'flush_schedule' ) );
add_filter( 'set-screen-option', array( $this, 'set_per_page' ), 10, 3 );
add_action( 'redirection_redirect_updated', array( $this, 'set_default_group' ), 10, 2 );
add_action( 'wp_ajax_red_proxy', array( $this, 'red_proxy' ) );
if ( defined( 'REDIRECTION_FLYING_SOLO' ) && REDIRECTION_FLYING_SOLO ) {
add_filter( 'script_loader_src', array( $this, 'flying_solo' ), 10, 2 );
}
register_deactivation_hook( REDIRECTION_FILE, array( 'Redirection_Admin', 'plugin_deactivated' ) );
register_uninstall_hook( REDIRECTION_FILE, array( 'Redirection_Admin', 'plugin_uninstall' ) );
$this->monitor = new Red_Monitor( red_get_options() );
$this->run_hacks();
}
// These are only called on the single standard site, or in the network admin of the multisite - they run across all available sites
public static function plugin_activated() {
if ( is_network_admin() ) {
foreach ( get_sites() as $site ) {
switch_to_blog( $site->blog_id );
Redirection_Admin::update();
Red_Flusher::schedule();
red_set_options();
restore_current_blog();
}
} else {
Redirection_Admin::update();
Red_Flusher::schedule();
red_set_options();
}
}
// These are only called on the single standard site, or in the network admin of the multisite - they run across all available sites
public static function plugin_deactivated() {
if ( is_network_admin() ) {
foreach ( get_sites() as $site ) {
switch_to_blog( $site->blog_id );
Red_Flusher::clear();
restore_current_blog();
}
} else {
Red_Flusher::clear();
}
}
// These are only called on the single standard site, or in the network admin of the multisite - they run across all available sites
public static function plugin_uninstall() {
include_once dirname( REDIRECTION_FILE ) . '/models/database.php';
$database = new RE_Database();
if ( is_network_admin() ) {
foreach ( get_sites() as $site ) {
switch_to_blog( $site->blog_id );
$database->remove( REDIRECTION_FILE );
restore_current_blog();
}
} else {
$database->remove( REDIRECTION_FILE );
}
}
private static function update() {
$version = get_option( 'redirection_version' );
Red_Flusher::schedule();
if ( $version !== REDIRECTION_DB_VERSION || ( defined( 'REDIRECTION_FORCE_UPDATE' ) && REDIRECTION_FORCE_UPDATE ) ) {
include_once dirname( REDIRECTION_FILE ) . '/models/database.php';
$database = new RE_Database();
if ( $version === false ) {
$database->install();
}
$database->upgrade( $version, REDIRECTION_DB_VERSION );
}
}
// So it finally came to this... some plugins include their JS in all pages, whether they are needed or not. If there is an error
// then this can prevent Redirection running and it's a little sensitive about that. We use the nuclear option here to disable
// all other JS while viewing Redirection
public function flying_solo( $src, $handle ) {
if ( isset( $_SERVER['REQUEST_URI'] ) && strpos( $_SERVER['REQUEST_URI'], 'page=redirection.php' ) !== false ) {
if ( substr( $src, 0, 4 ) === 'http' && $handle !== 'redirection' && strpos( $src, 'plugins' ) !== false ) {
if ( $this->ignore_this_plugin( $src ) ) {
return false;
}
}
}
return $src;
}
private function ignore_this_plugin( $src ) {
$ignore = array(
'mootools',
'wp-seo-',
'authenticate',
'wordpress-seo',
'yikes',
);
foreach ( $ignore as $text ) {
if ( strpos( $src, $text ) !== false ) {
return true;
}
}
return false;
}
public function flush_schedule( $options ) {
Red_Flusher::schedule();
return $options;
}
function set_per_page( $status, $option, $value ) {
if ( $option === 'redirection_log_per_page' ) {
return max( 1, min( intval( $value, 10 ), RED_MAX_PER_PAGE ) );
}
return $status;
}
function plugin_settings( $links ) {
$settings_link = '<a href="tools.php?page=' . basename( REDIRECTION_FILE ) . '&sub=options">' . __( 'Settings', 'redirection' ) . '</a>';
array_unshift( $links, $settings_link );
return $links;
}
function redirection_head() {
global $wp_version;
$this->check_rest_api();
if ( isset( $_REQUEST['action'] ) && isset( $_REQUEST['_wpnonce'] ) && wp_verify_nonce( $_REQUEST['_wpnonce'], 'wp_rest' ) ) {
if ( $_REQUEST['action'] === 'fixit' ) {
$this->run_fixit();
} elseif ( $_REQUEST['action'] === 'rest_api' ) {
$this->set_rest_api( intval( $_REQUEST['rest_api'], 10 ) );
} elseif ( $_REQUEST['action'] === 'red_proxy' ) {
// Hack to get around clash with WP page param
if ( isset( $_GET['page'] ) && $_GET['page'] === 'redirection.php' ) {
unset( $_GET['page'] );
}
if ( isset( $_GET['ppage'] ) ) {
$_GET['page'] = $_GET['ppage'];
}
$this->red_proxy();
}
}
$build = REDIRECTION_VERSION . '-' . REDIRECTION_BUILD;
$preload = $this->get_preload_data();
$options = red_get_options();
$versions = array(
'Plugin: ' . REDIRECTION_VERSION,
'WordPress: ' . $wp_version . ' (' . ( is_multisite() ? 'multi' : 'single' ) . ')',
'PHP: ' . phpversion(),
'Browser: ' . Redirection_Request::get_user_agent(),
'JavaScript: ' . plugin_dir_url( REDIRECTION_FILE ) . 'redirection.js',
'REST API: ' . red_get_rest_api(),
);
$this->inject();
if ( ! isset( $_GET['sub'] ) || ( isset( $_GET['sub'] ) && ( in_array( $_GET['sub'], array( 'log', '404s', 'groups' ) ) ) ) ) {
add_screen_option( 'per_page', array(
/* translators: maximum number of log entries */
'label' => sprintf( __( 'Log entries (%d max)', 'redirection' ), RED_MAX_PER_PAGE ),
'default' => RED_DEFAULT_PER_PAGE,
'option' => 'redirection_log_per_page',
) );
}
if ( defined( 'REDIRECTION_DEV_MODE' ) && REDIRECTION_DEV_MODE ) {
wp_enqueue_script( 'redirection', 'http://localhost:3312/redirection.js', array(), $build, true );
} else {
wp_enqueue_script( 'redirection', plugin_dir_url( REDIRECTION_FILE ) . 'redirection.js', array(), $build, true );
}
wp_enqueue_style( 'redirection', plugin_dir_url( REDIRECTION_FILE ) . 'redirection.css', array(), $build );
wp_localize_script( 'redirection', 'Redirectioni10n', array(
'WP_API_root' => esc_url_raw( red_get_rest_api() ),
'WP_API_nonce' => wp_create_nonce( 'wp_rest' ),
'pluginBaseUrl' => plugins_url( '', REDIRECTION_FILE ),
'pluginRoot' => admin_url( 'tools.php?page=redirection.php' ),
'per_page' => $this->get_per_page(),
'locale' => $this->get_i18n_data(),
'localeSlug' => get_locale(),
'token' => $options['token'],
'autoGenerate' => $options['auto_target'],
'preload' => $preload,
'versions' => implode( "\n", $versions ),
'version' => REDIRECTION_VERSION,
'api_setting' => $options['rest_api'],
) );
$this->add_help_tab();
}
// Some plugins misbehave, so this attempts to 'fix' them so Redirection can get on with it's work
private function run_hacks() {
add_filter( 'ip-geo-block-admin', array( $this, 'ip_geo_block' ) );
}
/**
* This works around the IP Geo Block plugin being very aggressive and breaking Redirection
*/
public function ip_geo_block( $validate ) {
$url = Redirection_Request::get_request_url();
$override = array(
'tools.php?page=redirection.php',
'action=red_proxy&rest_path=redirection',
);
foreach ( $override as $path ) {
if ( strpos( $url, $path ) !== false ) {
return array(
'result' => 'passed',
'auth' => false,
'asn' => false,
'code' => false,
'ip' => false,
);
}
}
return $validate;
}
public function check_rest_api() {
$options = red_get_options();
if ( $options['version'] !== REDIRECTION_VERSION || $options['rest_api'] === false || ( defined( 'REDIRECTION_FORCE_UPDATE' ) && REDIRECTION_FORCE_UPDATE ) ) {
include_once dirname( REDIRECTION_FILE ) . '/models/fixer.php';
$fixer = new Red_Fixer();
$status = $fixer->get_rest_status();
if ( $status['status'] === 'problem' ) {
$fixer->fix_rest();
} elseif ( $options['rest_api'] === false ) {
red_set_options( array( 'rest_api' => 0 ) );
}
red_set_options( array( 'version' => REDIRECTION_VERSION ) );
}
}
private function run_fixit() {
if ( current_user_can( apply_filters( 'redirection_role', 'manage_options' ) ) ) {
include_once dirname( REDIRECTION_FILE ) . '/models/fixer.php';
$fixer = new Red_Fixer();
$fixer->fix( $fixer->get_status() );
}
}
private function set_rest_api( $api ) {
if ( $api >= 0 && $api <= REDIRECTION_API_POST ) {
red_set_options( array( 'rest_api' => intval( $api, 10 ) ) );
}
}
private function get_preload_data() {
$page = '';
if ( isset( $_GET['sub'] ) && in_array( $_GET['sub'], array( 'group', '404s', 'log', 'io', 'options', 'support' ) ) ) {
$page = $_GET['sub'];
}
if ( $page === 'support' ) {
$api = new Redirection_Api_Plugin( REDIRECTION_API_NAMESPACE );
return array(
'pluginStatus' => $api->route_status( new WP_REST_Request() ),
);
}
return array();
}
private function add_help_tab() {
/* translators: URL */
$content = sprintf( __( 'You can find full documentation about using Redirection on the <a href="%s" target="_blank">redirection.me</a> support site.', 'redirection' ), 'https://redirection.me/support/?utm_source=redirection&utm_medium=plugin&utm_campaign=context-help' );
$title = __( 'Redirection Support', 'redirection' );
$current_screen = get_current_screen();
$current_screen->add_help_tab( array(
'id' => 'redirection',
'title' => 'Redirection',
'content' => "<h2>$title</h2><p>$content</p>",
) );
}
private function get_per_page() {
$per_page = intval( get_user_meta( get_current_user_id(), 'redirection_log_per_page', true ), 10 );
return $per_page > 0 ? max( 5, min( $per_page, RED_MAX_PER_PAGE ) ) : RED_DEFAULT_PER_PAGE;
}
private function get_i18n_data() {
$locale = get_locale();
// WP 4.7
if ( function_exists( 'get_user_locale' ) ) {
$locale = get_user_locale();
}
$i18n_json = dirname( REDIRECTION_FILE ) . '/locale/json/redirection-' . $locale . '.json';
if ( is_file( $i18n_json ) && is_readable( $i18n_json ) ) {
$locale_data = @file_get_contents( $i18n_json );
if ( $locale_data ) {
return json_decode( $locale_data );
}
}
// Return empty if we have nothing to return so it doesn't fail when parsed in JS
return array();
}
function admin_menu() {
$hook = add_management_page( 'Redirection', 'Redirection', apply_filters( 'redirection_role', 'manage_options' ), basename( REDIRECTION_FILE ), array( &$this, 'admin_screen' ) );
add_action( 'load-' . $hook, array( $this, 'redirection_head' ) );
}
private function check_minimum_wp() {
$wp_version = get_bloginfo( 'version' );
if ( version_compare( $wp_version, REDIRECTION_MIN_WP, '<' ) ) {
/* translators: 1: Expected WordPress version, 2: Actual WordPress version */
$wp_requirement = sprintf( __( 'Redirection requires WordPress v%1$1s, you are using v%2$2s - please update your WordPress', 'redirection' ), REDIRECTION_MIN_WP, $wp_version );
?>
<div class="react-error">
<h1><?php esc_html_e( 'Unable to load Redirection', 'redirection' ); ?></h1>
<p style="text-align: left"><?php echo esc_html( $wp_requirement ); ?></p>
</div>
<?php
return false;
}
return true;
}
private function check_minimum_php() {
if ( version_compare( PHP_VERSION, '5.4' ) < 0 ) {
/* translators: 1: Expected PHP version, 2: Actual PHP version */
$php_version = sprintf( __( 'Redirection requires PHP v%1$1s, you are using v%2$2s. This plugin will stop working from the next version.', 'redirection' ), '5.4', PHP_VERSION );
?>
<div class="error">
<h1><?php esc_html_e( 'Unsupported PHP', 'redirection' ); ?></h1>
<p style="text-align: left"><?php echo esc_html( $php_version ); ?></p>
</div>
<?php
}
}
private function check_tables_exist() {
include_once dirname( REDIRECTION_FILE ) . '/models/database.php';
$database = new RE_Database();
$status = $database->get_status();
if ( $status['status'] !== 'good' ) {
/* translators: URL */
$reason = printf( __( 'Problems were detected with your database tables. Please visit the <a href="%s">support page</a> for more details.', 'redirection' ), 'tools.php?page=redirection.php&sub=support' );
?>
<div class="error">
<h3><?php esc_html_e( 'Redirection not installed properly', 'redirection' ); ?></h3>
<p style="text-align: left"><?php echo esc_html( $reason ); ?></p>
</div>
<?php
return false;
}
return true;
}
public function set_default_group( $id, $redirect ) {
red_set_options( array( 'last_group_id' => $redirect->get_group_id() ) );
}
function admin_screen() {
$version = red_get_plugin_data( REDIRECTION_FILE );
$version = $version['Version'];
Redirection_Admin::update();
if ( $this->check_minimum_php() === false ) {
return;
}
if ( $this->check_minimum_wp() === false ) {
return;
}
if ( $this->check_tables_exist() === false && ( ! isset( $_GET['sub'] ) || $_GET['sub'] !== 'support' ) ) {
return false;
}
?>
<div id="react-modal"></div>
<div id="react-ui">
<div class="react-loading">
<h1><?php esc_html_e( 'Loading, please wait...', 'redirection' ); ?></h1>
<span class="react-loading-spinner"></span>
</div>
<noscript>Please enable JavaScript</noscript>
<div class="react-error" style="display: none">
<h1><?php esc_html_e( 'Unable to load Redirection ☹️', 'redirection' ); ?> v<?php echo esc_html( $version ); ?></h1>
<p><?php esc_html_e( "This may be caused by another plugin - look at your browser's error console for more details.", 'redirection' ); ?></p>
<p><?php esc_html_e( 'If you are using a page caching plugin or service (CloudFlare, OVH, etc) then you can also try clearing that cache.', 'redirection' ); ?></p>
<p><?php esc_html_e( 'Also check if your browser is able to load <code>redirection.js</code>:', 'redirection' ); ?></p>
<p><code><?php echo esc_html( plugin_dir_url( REDIRECTION_FILE ) . 'redirection.js?ver=' . urlencode( REDIRECTION_VERSION ) . '-' . urlencode( REDIRECTION_BUILD ) ); ?></code></p>
<p><?php esc_html_e( 'Please note that Redirection requires the WordPress REST API to be enabled. If you have disabled this then you won\'t be able to use Redirection', 'redirection' ); ?></p>
<p><?php esc_html_e( 'Please see the <a href="https://redirection.me/support/problems/">list of common problems</a>.', 'redirection' ); ?></p>
<p><?php esc_html_e( 'If you think Redirection is at fault then create an issue.', 'redirection' ); ?></p>
<p class="versions"><?php esc_html_e( '<code>Redirectioni10n</code> is not defined. This usually means another plugin is blocking Redirection from loading. Please disable all plugins and try again.', 'redirection' ); ?></p>
<p>
<a class="button-primary" target="_blank" href="https://github.com/johngodley/redirection/issues/new?title=Problem%20starting%20Redirection%20<?php echo esc_attr( $version ); ?>">
<?php esc_html_e( 'Create Issue', 'redirection' ); ?>
</a>
</p>
</div>
</div>
<script>
var prevError = window.onerror;
var errors = [];
var timeout = 0;
var timer = setInterval( function() {
if ( isRedirectionLoaded() ) {
resetAll();
} else if ( errors.length > 0 || timeout++ === 5 ) {
showError();
}
}, 5000 );
function isRedirectionLoaded() {
return typeof redirection !== 'undefined';
}
function showError() {
var errorText = "";
if ( errors.length > 0 ) {
errorText = "```\n" + errors.join( ',' ) + "\n```\n\n";
}
resetAll();
document.querySelector( '.react-loading' ).style.display = 'none';
document.querySelector( '.react-error' ).style.display = 'block';
if ( typeof Redirectioni10n !== 'undefined' ) {
document.querySelector( '.versions' ).innerHTML = Redirectioni10n.versions.replace( /\n/g, '<br />' );
document.querySelector( '.react-error .button-primary' ).href += '&body=' + encodeURIComponent( errorText ) + encodeURIComponent( Redirectioni10n.versions );
}
}
function resetAll() {
clearInterval( timer );
window.onerror = prevError;
}
window.onerror = function( error, url, line ) {
console.error( error );
errors.push( error + ' ' + url + ' ' + line );
};
</script>
<?php
}
/**
* Really wish I didnt have to do this...
* NOTE: nonce is checked by serve_request
*/
public function red_proxy() {
if ( $this->user_has_access() && isset( $_GET['rest_path'] ) && substr( $_GET['rest_path'], 0, 15 ) === 'redirection/v1/' ) {
$_SERVER['HTTP_CONTENT_TYPE'] = 'application/json; charset=utf-8';
$_SERVER['CONTENT_TYPE'] = $_SERVER['HTTP_CONTENT_TYPE'];
$server = rest_get_server();
$server->serve_request( rtrim( '/' . $_GET['rest_path'], '/' ) );
die();
}
}
private function user_has_access() {
return current_user_can( apply_filters( 'redirection_role', 'manage_options' ) );
}
function inject() {
if ( isset( $_GET['page'] ) && isset( $_GET['sub'] ) && $_GET['page'] === 'redirection.php' ) {
$this->try_export_logs();
$this->try_export_redirects();
$this->try_export_rss();
}
}
function try_export_rss() {
if ( isset( $_GET['token'] ) && $_GET['sub'] === 'rss' ) {
$options = red_get_options();
if ( $_GET['token'] === $options['token'] && ! empty( $options['token'] ) ) {
$items = Red_Item::get_all_for_module( intval( $_GET['module'] ) );
$exporter = Red_FileIO::create( 'rss' );
$exporter->force_download();
echo $exporter->get_data( $items, array() );
die();
}
}
}
private function try_export_logs() {
if ( $this->user_has_access() && isset( $_POST['export-csv'] ) && check_admin_referer( 'wp_rest' ) ) {
if ( isset( $_GET['sub'] ) && $_GET['sub'] === 'log' ) {
RE_Log::export_to_csv();
} else {
RE_404::export_to_csv();
}
die();
}
}
private function try_export_redirects() {
if ( $this->user_has_access() && $_GET['sub'] === 'io' && isset( $_GET['exporter'] ) && isset( $_GET['export'] ) ) {
$export = Red_FileIO::export( $_GET['export'], $_GET['exporter'] );
if ( $export !== false ) {
$export['exporter']->force_download();
echo $export['data'];
die();
}
}
}
}
register_activation_hook( REDIRECTION_FILE, array( 'Redirection_Admin', 'plugin_activated' ) );
add_action( 'init', array( 'Redirection_Admin', 'init' ) );