| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401 |
- <?php
- /**
- * Jetpack Debug Data for the Site Health sections.
- *
- * @package automattic/jetpack
- */
- use Automattic\Jetpack\Connection\Tokens;
- use Automattic\Jetpack\Connection\Urls;
- use Automattic\Jetpack\Constants;
- use Automattic\Jetpack\Identity_Crisis;
- use Automattic\Jetpack\Redirect;
- use Automattic\Jetpack\Sync\Modules;
- use Automattic\Jetpack\Sync\Sender;
- /**
- * Class Jetpack_Debug_Data
- *
- * Collect and return debug data for Jetpack.
- *
- * @since 7.3.0
- */
- class Jetpack_Debug_Data {
- /**
- * Determine the active plan and normalize it for the debugger results.
- *
- * @since 7.3.0
- *
- * @return string The plan slug.
- */
- public static function what_jetpack_plan() {
- $plan = Jetpack_Plan::get();
- return ! empty( $plan['class'] ) ? $plan['class'] : 'undefined';
- }
- /**
- * Convert seconds to human readable time.
- *
- * A dedication function instead of using Core functionality to allow for output in seconds.
- *
- * @since 7.3.0
- *
- * @param int $seconds Number of seconds to convert to human time.
- *
- * @return string Human readable time.
- */
- public static function seconds_to_time( $seconds ) {
- $seconds = (int) $seconds;
- $units = array(
- 'week' => WEEK_IN_SECONDS,
- 'day' => DAY_IN_SECONDS,
- 'hour' => HOUR_IN_SECONDS,
- 'minute' => MINUTE_IN_SECONDS,
- 'second' => 1,
- );
- // specifically handle zero.
- if ( 0 === $seconds ) {
- return '0 seconds';
- }
- $human_readable = '';
- foreach ( $units as $name => $divisor ) {
- $quot = (int) ( $seconds / $divisor );
- if ( $quot ) {
- $human_readable .= "$quot $name";
- $human_readable .= ( abs( $quot ) > 1 ? 's' : '' ) . ', ';
- $seconds -= $quot * $divisor;
- }
- }
- return substr( $human_readable, 0, -2 );
- }
- /**
- * Return debug data in the format expected by Core's Site Health Info tab.
- *
- * @since 7.3.0
- *
- * @param array $debug {
- * The debug information already compiled by Core.
- *
- * @type string $label The title for this section of the debug output.
- * @type string $description Optional. A description for your information section which may contain basic HTML
- * markup: `em`, `strong` and `a` for linking to documentation or putting emphasis.
- * @type boolean $show_count Optional. If set to `true` the amount of fields will be included in the title for
- * this section.
- * @type boolean $private Optional. If set to `true` the section and all associated fields will be excluded
- * from the copy-paste text area.
- * @type array $fields {
- * An associative array containing the data to be displayed.
- *
- * @type string $label The label for this piece of information.
- * @type string $value The output that is of interest for this field.
- * @type boolean $private Optional. If set to `true` the field will not be included in the copy-paste text area
- * on top of the page, allowing you to show, for example, API keys here.
- * }
- * }
- *
- * @return array $args Debug information in the same format as the initial argument.
- */
- public static function core_debug_data( $debug ) {
- $support_url = Jetpack::is_development_version()
- ? Redirect::get_url( 'jetpack-contact-support-beta-group' )
- : Redirect::get_url( 'jetpack-contact-support' );
- $jetpack = array(
- 'jetpack' => array(
- 'label' => __( 'Jetpack', 'jetpack' ),
- 'description' => sprintf(
- /* translators: %1$s is URL to jetpack.com's contact support page. %2$s accessibility text */
- __(
- 'Diagnostic information helpful to <a href="%1$s" target="_blank" rel="noopener noreferrer">your Jetpack Happiness team<span class="screen-reader-text">%2$s</span></a>',
- 'jetpack'
- ),
- esc_url( $support_url ),
- __( '(opens in a new tab)', 'jetpack' )
- ),
- 'fields' => self::debug_data(),
- ),
- );
- $debug = array_merge( $debug, $jetpack );
- return $debug;
- }
- /**
- * Compile and return array of debug information.
- *
- * @since 7.3.0
- *
- * @return array $args {
- * Associated array of arrays with the following.
- * @type string $label The label for this piece of information.
- * @type string $value The output that is of interest for this field.
- * @type boolean $private Optional. Set to true if data is sensitive (API keys, etc).
- * }
- */
- public static function debug_data() {
- $debug_info = array();
- /* Add various important Jetpack options */
- $debug_info['site_id'] = array(
- 'label' => 'Jetpack Site ID',
- 'value' => Jetpack_Options::get_option( 'id' ),
- 'private' => false,
- );
- $debug_info['ssl_cert'] = array(
- 'label' => 'Jetpack SSL Verfication Bypass',
- 'value' => ( Jetpack_Options::get_option( 'fallback_no_verify_ssl_certs' ) ) ? 'Yes' : 'No',
- 'private' => false,
- );
- $debug_info['time_diff'] = array(
- 'label' => "Offset between Jetpack server's time and this server's time.",
- 'value' => Jetpack_Options::get_option( 'time_diff' ),
- 'private' => false,
- );
- $debug_info['version_option'] = array(
- 'label' => 'Current Jetpack Version Option',
- 'value' => Jetpack_Options::get_option( 'version' ),
- 'private' => false,
- );
- $debug_info['old_version'] = array(
- 'label' => 'Previous Jetpack Version',
- 'value' => Jetpack_Options::get_option( 'old_version' ),
- 'private' => false,
- );
- $debug_info['public'] = array(
- 'label' => 'Jetpack Site Public',
- 'value' => ( Jetpack_Options::get_option( 'public' ) ) ? 'Public' : 'Private',
- 'private' => false,
- );
- $debug_info['master_user'] = array(
- 'label' => 'Jetpack Master User',
- 'value' => self::human_readable_master_user(), // Only ID number and user name.
- 'private' => false,
- );
- /**
- * Token information is private, but awareness if there one is set is helpful.
- *
- * To balance out information vs privacy, we only display and include the "key",
- * which is a segment of the token prior to a period within the token and is
- * technically not private.
- *
- * If a token does not contain a period, then it is malformed and we report it as such.
- */
- $user_id = get_current_user_id();
- $blog_token = ( new Tokens() )->get_access_token();
- $user_token = ( new Tokens() )->get_access_token( $user_id );
- $tokenset = '';
- if ( $blog_token ) {
- $tokenset = 'Blog ';
- $blog_key = substr( $blog_token->secret, 0, strpos( $blog_token->secret, '.' ) );
- // Intentionally not translated since this is helpful when sent to Happiness.
- $blog_key = ( $blog_key ) ? $blog_key : 'Potentially Malformed Token.';
- }
- if ( $user_token ) {
- $tokenset .= 'User';
- $user_key = substr( $user_token->secret, 0, strpos( $user_token->secret, '.' ) );
- // Intentionally not translated since this is helpful when sent to Happiness.
- $user_key = ( $user_key ) ? $user_key : 'Potentially Malformed Token.';
- }
- if ( ! $tokenset ) {
- $tokenset = 'None';
- }
- $debug_info['current_user'] = array(
- 'label' => 'Current User',
- 'value' => self::human_readable_user( $user_id ),
- 'private' => false,
- );
- $debug_info['tokens_set'] = array(
- 'label' => 'Tokens defined',
- 'value' => $tokenset,
- 'private' => false,
- );
- $debug_info['blog_token'] = array(
- 'label' => 'Blog Public Key',
- 'value' => ( $blog_token ) ? $blog_key : 'Not set.',
- 'private' => false,
- );
- $debug_info['user_token'] = array(
- 'label' => 'User Public Key',
- 'value' => ( $user_token ) ? $user_key : 'Not set.',
- 'private' => false,
- );
- /** Jetpack Environmental Information */
- $debug_info['version'] = array(
- 'label' => 'Jetpack Version',
- 'value' => JETPACK__VERSION,
- 'private' => false,
- );
- $debug_info['jp_plugin_dir'] = array(
- 'label' => 'Jetpack Directory',
- 'value' => JETPACK__PLUGIN_DIR,
- 'private' => false,
- );
- $debug_info['plan'] = array(
- 'label' => 'Plan Type',
- 'value' => self::what_jetpack_plan(),
- 'private' => false,
- );
- foreach ( array(
- 'HTTP_HOST',
- 'SERVER_PORT',
- 'HTTPS',
- 'GD_PHP_HANDLER',
- 'HTTP_AKAMAI_ORIGIN_HOP',
- 'HTTP_CF_CONNECTING_IP',
- 'HTTP_CLIENT_IP',
- 'HTTP_FASTLY_CLIENT_IP',
- 'HTTP_FORWARDED',
- 'HTTP_FORWARDED_FOR',
- 'HTTP_INCAP_CLIENT_IP',
- 'HTTP_TRUE_CLIENT_IP',
- 'HTTP_X_CLIENTIP',
- 'HTTP_X_CLUSTER_CLIENT_IP',
- 'HTTP_X_FORWARDED',
- 'HTTP_X_FORWARDED_FOR',
- 'HTTP_X_IP_TRAIL',
- 'HTTP_X_REAL_IP',
- 'HTTP_X_VARNISH',
- 'REMOTE_ADDR',
- ) as $header ) {
- if ( isset( $_SERVER[ $header ] ) ) {
- $debug_info[ $header ] = array(
- 'label' => 'Server Variable ' . $header,
- 'value' => ( $_SERVER[ $header ] ) ? $_SERVER[ $header ] : 'false',
- 'private' => true, // This isn't really 'private' information, but we don't want folks to easily paste these into public forums.
- );
- }
- }
- $debug_info['protect_header'] = array(
- 'label' => 'Trusted IP',
- 'value' => wp_json_encode( get_site_option( 'trusted_ip_header' ) ),
- 'private' => false,
- );
- /** Sync Debug Information */
- $sync_module = Modules::get_module( 'full-sync' );
- if ( $sync_module ) {
- $sync_statuses = $sync_module->get_status();
- $human_readable_sync_status = array();
- foreach ( $sync_statuses as $sync_status => $sync_status_value ) {
- $human_readable_sync_status[ $sync_status ] =
- in_array( $sync_status, array( 'started', 'queue_finished', 'send_started', 'finished' ), true )
- ? gmdate( 'r', $sync_status_value ) : $sync_status_value;
- }
- $debug_info['full_sync'] = array(
- 'label' => 'Full Sync Status',
- 'value' => wp_json_encode( $human_readable_sync_status ),
- 'private' => false,
- );
- }
- $queue = Sender::get_instance()->get_sync_queue();
- $debug_info['sync_size'] = array(
- 'label' => 'Sync Queue Size',
- 'value' => $queue->size(),
- 'private' => false,
- );
- $debug_info['sync_lag'] = array(
- 'label' => 'Sync Queue Lag',
- 'value' => self::seconds_to_time( $queue->lag() ),
- 'private' => false,
- );
- $full_sync_queue = Sender::get_instance()->get_full_sync_queue();
- $debug_info['full_sync_size'] = array(
- 'label' => 'Full Sync Queue Size',
- 'value' => $full_sync_queue->size(),
- 'private' => false,
- );
- $debug_info['full_sync_lag'] = array(
- 'label' => 'Full Sync Queue Lag',
- 'value' => self::seconds_to_time( $full_sync_queue->lag() ),
- 'private' => false,
- );
- /**
- * IDC Information
- *
- * Must follow sync debug since it depends on sync functionality.
- */
- $idc_urls = array(
- 'home' => Urls::home_url(),
- 'siteurl' => Urls::site_url(),
- 'WP_HOME' => Constants::is_defined( 'WP_HOME' ) ? Constants::get_constant( 'WP_HOME' ) : '',
- 'WP_SITEURL' => Constants::is_defined( 'WP_SITEURL' ) ? Constants::get_constant( 'WP_SITEURL' ) : '',
- );
- $debug_info['idc_urls'] = array(
- 'label' => 'IDC URLs',
- 'value' => wp_json_encode( $idc_urls ),
- 'private' => false,
- );
- $debug_info['idc_error_option'] = array(
- 'label' => 'IDC Error Option',
- 'value' => wp_json_encode( Jetpack_Options::get_option( 'sync_error_idc' ) ),
- 'private' => false,
- );
- $debug_info['idc_optin'] = array(
- 'label' => 'IDC Opt-in',
- 'value' => Identity_Crisis::should_handle_idc(),
- 'private' => false,
- );
- // @todo -- Add testing results?
- $cxn_tests = new Jetpack_Cxn_Tests();
- $debug_info['cxn_tests'] = array(
- 'label' => 'Connection Tests',
- 'value' => '',
- 'private' => false,
- );
- if ( $cxn_tests->pass() ) {
- $debug_info['cxn_tests']['value'] = 'All Pass.';
- } else {
- $debug_info['cxn_tests']['value'] = wp_json_encode( $cxn_tests->list_fails() );
- }
- return $debug_info;
- }
- /**
- * Returns a human readable string for which user is the master user.
- *
- * @return string
- */
- private static function human_readable_master_user() {
- $master_user = Jetpack_Options::get_option( 'master_user' );
- if ( ! $master_user ) {
- return __( 'No master user set.', 'jetpack' );
- }
- $user = new WP_User( $master_user );
- if ( ! $user ) {
- return __( 'Master user no longer exists. Please disconnect and reconnect Jetpack.', 'jetpack' );
- }
- return self::human_readable_user( $user );
- }
- /**
- * Return human readable string for a given user object.
- *
- * @param WP_User|int $user Object or ID.
- *
- * @return string
- */
- private static function human_readable_user( $user ) {
- $user = new WP_User( $user );
- return sprintf( '#%1$d %2$s', $user->ID, $user->user_login ); // Format: "#1 username".
- }
- }
|