Completed
Push — add/gui-testing ( 5d6617...131071 )
by
unknown
82:01 queued 73:42
created

class.jetpack-idc.php (2 issues)

Upgrade to new PHP Analysis Engine

These results are based on our legacy PHP analysis, consider migrating to our new PHP analysis engine instead. Learn more

1
<?php
2
3
/**
4
 * This class will handle everything involved with fixing an Identity Crisis.
5
 *
6
 * @since 4.4.0
7
 */
8
class Jetpack_IDC {
9
10
	/**
11
	 * @var Jetpack_IDC
12
	 **/
13
	private static $instance = null;
14
15
	/**
16
	 * The wpcom value of the home URL
17
	 * @var string
18
	 */
19
	static $wpcom_home_url;
20
21
	/**
22
	 * Has safe mode been confirmed?
23
	 * @var bool
24
	 */
25
	static $is_safe_mode_confirmed;
0 ignored issues
show
The visibility should be declared for property $is_safe_mode_confirmed.

The PSR-2 coding standard requires that all properties in a class have their visibility explicitly declared. If you declare a property using

class A {
    var $property;
}

the property is implicitly global.

To learn more about the PSR-2, please see the PHP-FIG site on the PSR-2.

Loading history...
26
27
	/**
28
	 * The current screen, which is set if the current user is a non-admin and this is an admin page.
29
	 * @var WP_Screen
30
	 */
31
	static $current_screen;
0 ignored issues
show
The visibility should be declared for property $current_screen.

The PSR-2 coding standard requires that all properties in a class have their visibility explicitly declared. If you declare a property using

class A {
    var $property;
}

the property is implicitly global.

To learn more about the PSR-2, please see the PHP-FIG site on the PSR-2.

Loading history...
32
33
	/**
34
	 * The link to the support document used to explain Safe Mode to users
35
	 * @var string
36
	 */
37
	const SAFE_MODE_DOC_LINK = 'https://jetpack.com/support/safe-mode';
38
39
	static function init() {
40
		if ( is_null( self::$instance ) ) {
41
			self::$instance = new Jetpack_IDC;
42
		}
43
44
		return self::$instance;
45
	}
46
47
	private function __construct() {
48
		add_action( 'jetpack_sync_processed_actions', array( $this, 'maybe_clear_migrate_option' ) );
49
50
		if ( false === $urls_in_crisis = Jetpack::check_identity_crisis() ) {
51
			return;
52
		}
53
54
		self::$wpcom_home_url = $urls_in_crisis['wpcom_home'];
55
		add_action( 'init', array( $this, 'wordpress_init' ) );
56
	}
57
58
	/**
59
	 * This method loops through the array of processed items from sync and checks if one of the items was the
60
	 * home_url or site_url callable. If so, then we delete the jetpack_migrate_for_idc option.
61
	 *
62
	 * @param $processed_items array Array of processed items that were synced to WordPress.com
63
	 */
64
	function maybe_clear_migrate_option( $processed_items ) {
65
		foreach ( (array) $processed_items as $item ) {
66
67
			// First, is this item a jetpack_sync_callable action? If so, then proceed.
68
			$callable_args = ( is_array( $item ) && isset( $item[0], $item[1] ) && 'jetpack_sync_callable' === $item[0] )
69
				? $item[1]
70
				: null;
71
72
			// Second, if $callable_args is set, check if the callable was home_url or site_url. If so,
73
			// clear the migrate option.
74
			if (
75
				isset( $callable_args, $callable_args[0] )
76
				&& ( 'home_url' === $callable_args[0] || 'site_url' === $callable_args[1] )
77
			) {
78
				Jetpack_Options::delete_option( 'migrate_for_idc' );
79
				break;
80
			}
81
		}
82
	}
83
84
	function wordpress_init() {
85
		if ( ! current_user_can( 'jetpack_disconnect' ) && is_admin() ) {
86
			add_action( 'admin_notices', array( $this, 'display_non_admin_idc_notice' ) );
87
			add_action( 'admin_enqueue_scripts', array( $this,'enqueue_idc_notice_files' ) );
88
			add_action( 'current_screen', array( $this, 'non_admins_current_screen_check' ) );
89
			return;
90
		}
91
92
		if (
93
			isset( $_GET['jetpack_idc_clear_confirmation'], $_GET['_wpnonce'] ) &&
94
			wp_verify_nonce( $_GET['_wpnonce'], 'jetpack_idc_clear_confirmation' )
95
		) {
96
			Jetpack_Options::delete_option( 'safe_mode_confirmed' );
97
			self::$is_safe_mode_confirmed = false;
98
		} else {
99
			self::$is_safe_mode_confirmed = (bool) Jetpack_Options::get_option( 'safe_mode_confirmed' );
100
		}
101
102
		// 121 Priority so that it's the most inner Jetpack item in the admin bar.
103
		add_action( 'admin_bar_menu', array( $this, 'display_admin_bar_button' ), 121 );
104
		add_action( 'admin_enqueue_scripts', array( $this, 'enqueue_admin_bar_css' ) );
105
106
		if ( is_admin() && ! self::$is_safe_mode_confirmed ) {
107
			add_action( 'admin_notices', array( $this, 'display_idc_notice' ) );
108
			add_action( 'admin_enqueue_scripts', array( $this,'enqueue_idc_notice_files' ) );
109
		}
110
	}
111
112
	function non_admins_current_screen_check( $current_screen ) {
113
		self::$current_screen = $current_screen;
114
		if ( isset( $current_screen->id ) && 'toplevel_page_jetpack' == $current_screen->id ) {
115
			return null;
116
		}
117
118
		// If the user has dismissed the notice, and we're not currently on a Jetpack page,
119
		// then do not show the non-admin notice.
120
		if ( isset( $_COOKIE, $_COOKIE['jetpack_idc_dismiss_notice'] ) ) {
121
			remove_action( 'admin_notices', array( $this, 'display_non_admin_idc_notice' ) );
122
			remove_action( 'admin_enqueue_scripts', array( $this,'enqueue_idc_notice_files' ) );
123
		}
124
	}
125
126
	function display_admin_bar_button() {
127
		global $wp_admin_bar;
128
129
		$href = is_admin()
130
			? add_query_arg( 'jetpack_idc_clear_confirmation', '1' )
131
			: add_query_arg( 'jetpack_idc_clear_confirmation', '1', admin_url() );
132
133
		$href = wp_nonce_url( $href, 'jetpack_idc_clear_confirmation' );
134
135
		$title = sprintf(
136
			'<span class="jp-idc-admin-bar">%s %s</span>',
137
			'<span class="dashicons dashicons-warning"></span>',
138
			esc_html__( 'Jetpack Safe Mode', 'jetpack' )
139
		);
140
141
		$menu = array(
142
			'id'     => 'jetpack-idc',
143
			'title'  => $title,
144
			'href'   => esc_url( $href ),
145
			'parent' => 'top-secondary',
146
		);
147
148
		if ( ! self::$is_safe_mode_confirmed ) {
149
			$menu['meta'] = array(
150
				'class' => 'hide'
151
			);
152
		}
153
154
		$wp_admin_bar->add_node( $menu );
155
	}
156
157
	static function prepare_url_for_display( $url ) {
158
		return untrailingslashit( Jetpack::normalize_url_protocol_agnostic( $url ) );
159
	}
160
161
	function display_non_admin_idc_notice() {
162
		$classes = 'jp-idc-notice is-non-admin notice notice-warning';
163
		if ( isset( self::$current_screen ) && 'toplevel_page_jetpack' != self::$current_screen->id ) {
164
			$classes .= ' is-dismissible';
165
		}
166
		?>
167
168
		<div class="<?php echo $classes; ?>">
169
			<?php $this->render_notice_header(); ?>
170
			<div class="jp-idc-notice__content-header">
171
				<h3 class="jp-idc-notice__content-header__lead">
172
					<?php echo $this->get_non_admin_notice_text(); ?>
173
				</h3>
174
175
				<p class="jp-idc-notice__content-header__explanation">
176
					<?php echo $this->get_non_admin_contact_admin_text(); ?>
177
				</p>
178
			</div>
179
		</div>
180
	<?php }
181
182
	/**
183
	 * First "step" of the IDC mitigation. Will provide some messaging and two options/buttons.
184
	 * "Confirm Staging" - Dismiss the notice and continue on with our lives in staging mode.
185
	 * "Fix Jetpack Connection" - Will disconnect the site and start the mitigation...
186
	 */
187
	function display_idc_notice() { ?>
188
		<div class="jp-idc-notice notice notice-warning">
189
			<?php $this->render_notice_header(); ?>
190
			<?php $this->render_notice_first_step(); ?>
191
			<?php $this->render_notice_second_step(); ?>
192
		</div>
193
	<?php }
194
195
	function enqueue_admin_bar_css() {
196
		wp_enqueue_style(
197
			'jetpack-idc-admin-bar-css',
198
			plugins_url( 'css/jetpack-idc-admin-bar.css', JETPACK__PLUGIN_FILE ),
199
			array( 'dashicons' ),
200
			JETPACK__VERSION
201
		);
202
	}
203
204
	/**
205
	 * Enqueue scripts for the notice
206
	 */
207
	function enqueue_idc_notice_files() {
208
209
		wp_enqueue_script(
210
			'jetpack-idc-js',
211
			plugins_url( '_inc/idc-notice.js', JETPACK__PLUGIN_FILE ),
212
			array( 'jquery' ),
213
			JETPACK__VERSION,
214
			true
215
		);
216
217
		wp_localize_script(
218
			'jetpack-idc-js',
219
			'idcL10n',
220
			array(
221
				'apiRoot' => esc_url_raw( rest_url() ),
222
				'nonce' => wp_create_nonce( 'wp_rest' ),
223
				'tracksUserData' => Jetpack_Tracks_Client::get_connected_user_tracks_identity(),
224
				'currentUrl' => remove_query_arg( '_wpnonce', remove_query_arg( 'jetpack_idc_clear_confirmation' ) ),
225
				'tracksEventData' => array(
226
					'isAdmin' => current_user_can( 'jetpack_disconnect' ),
227
					'currentScreen' => self::$current_screen ? self::$current_screen->id : false,
228
				),
229
			)
230
		);
231
232
		wp_register_style(
233
			'jetpack-dops-style',
234
			plugins_url( '_inc/build/admin.dops-style.css', JETPACK__PLUGIN_FILE ),
235
			array(),
236
			JETPACK__VERSION
237
		);
238
239
		wp_enqueue_style(
240
			'jetpack-idc-css',
241
			plugins_url( 'css/jetpack-idc.css', JETPACK__PLUGIN_FILE ),
242
			array( 'jetpack-dops-style' ),
243
			JETPACK__VERSION
244
		);
245
246
		// Required for Tracks
247
		wp_enqueue_script(
248
			'jp-tracks',
249
			'//stats.wp.com/w.js',
250
			array(),
251
			gmdate( 'YW' ),
252
			true
253
		);
254
255
		wp_enqueue_script(
256
			'jp-tracks-functions',
257
			plugins_url( '_inc/lib/tracks/tracks-callables.js', JETPACK__PLUGIN_FILE ),
258
			array(),
259
			JETPACK__VERSION,
260
			false
261
		);
262
	}
263
264
	function render_notice_header() { ?>
265
		<div class="jp-idc-notice__header">
266
			<div class="jp-idc-notice__header__emblem">
267
				<?php echo Jetpack::get_jp_emblem(); ?>
268
			</div>
269
			<p class="jp-idc-notice__header__text">
270
				<?php esc_html_e( 'Jetpack Safe Mode', 'jetpack' ); ?>
271
			</p>
272
		</div>
273
274
		<div class="jp-idc-notice__separator"></div>
275
	<?php }
276
277
	function render_notice_first_step() { ?>
278
		<div class="jp-idc-notice__first-step">
279
			<div class="jp-idc-notice__content-header">
280
				<h3 class="jp-idc-notice__content-header__lead">
281
					<?php echo $this->get_first_step_header_lead(); ?>
282
				</h3>
283
284
				<p class="jp-idc-notice__content-header__explanation">
285
					<?php echo $this->get_first_step_header_explanation(); ?>
286
				</p>
287
			</div>
288
289
			<div class="jp-idc-notice__actions">
290
				<div class="jp-idc-notice__action">
291
					<p class="jp-idc-notice__action__explanation">
292
						<?php echo $this->get_confirm_safe_mode_action_explanation(); ?>
293
					</p>
294
					<button id="jp-idc-confirm-safe-mode-action" class="dops-button">
295
						<?php echo $this->get_confirm_safe_mode_button_text(); ?>
296
					</button>
297
				</div>
298
299
				<div class="jp-idc-notice__action">
300
					<p class="jp-idc-notice__action__explanation">
301
						<?php echo $this->get_first_step_fix_connection_action_explanation(); ?>
302
					</p>
303
					<button id="jp-idc-fix-connection-action" class="dops-button">
304
						<?php echo $this->get_first_step_fix_connection_button_text(); ?>
305
					</button>
306
				</div>
307
			</div>
308
		</div>
309
	<?php }
310
311
	function render_notice_second_step() { ?>
312
		<div class="jp-idc-notice__second-step">
313
			<div class="jp-idc-notice__content-header">
314
				<h3 class="jp-idc-notice__content-header__lead">
315
					<?php echo $this->get_second_step_header_lead(); ?>
316
				</h3>
317
			</div>
318
319
			<div class="jp-idc-notice__actions">
320
				<div class="jp-idc-notice__action">
321
					<p class="jp-idc-notice__action__explanation">
322
						<?php echo $this->get_migrate_site_action_explanation(); ?>
323
					</p>
324
					<button id="jp-idc-migrate-action" class="dops-button">
325
						<?php echo $this->get_migrate_site_button_text(); ?>
326
					</button>
327
				</div>
328
329
				<div class="jp-idc-notice__action">
330
					<p class="jp-idc-notice__action__explanation">
331
						<?php echo $this->get_start_fresh_action_explanation(); ?>
332
					</p>
333
					<button id="jp-idc-reconnect-site-action" class="dops-button">
334
						<?php echo $this->get_start_fresh_button_text(); ?>
335
					</button>
336
				</div>
337
338
			</div>
339
340
			<p class="jp-idc-notice__unsure-prompt">
341
				<?php echo $this->get_unsure_prompt(); ?>
342
			</p>
343
		</div>
344
	<?php }
345
346 View Code Duplication
	function get_first_step_header_lead() {
347
		$html = wp_kses(
348
			sprintf(
349
				__(
350
					'Jetpack has been placed into <a href="%1$s">Safe mode</a> because we noticed this is an exact copy of <a href="%2$s">%3$s</a>.',
351
					'jetpack'
352
				),
353
				esc_url( self::SAFE_MODE_DOC_LINK ),
354
				esc_url( self::$wpcom_home_url ),
355
				self::prepare_url_for_display( esc_url_raw( self::$wpcom_home_url ) )
356
			),
357
			array( 'a' => array( 'href' => array() ) )
358
		);
359
360
		/**
361
		 * Allows overriding of the default header text in the first step of the Safe Mode notice.
362
		 *
363
		 * @since 4.4.0
364
		 *
365
		 * @param string $html The HTML to be displayed
366
		 */
367
		return apply_filters( 'jetpack_idc_first_step_header_lead', $html );
368
	}
369
370 View Code Duplication
	function get_first_step_header_explanation() {
371
		$html = wp_kses(
372
			sprintf(
373
				__(
374
					'Please confirm Safe Mode or fix the Jetpack connection. Select one of the options below or <a href="%1$s">learn 
375
					more about Safe Mode</a>.',
376
					'jetpack'
377
				),
378
				esc_url( self::SAFE_MODE_DOC_LINK )
379
			),
380
			array( 'a' => array( 'href' => array() ) )
381
		);
382
383
		/**
384
		 * Allows overriding of the default header explanation text in the first step of the Safe Mode notice.
385
		 *
386
		 * @since 4.4.0
387
		 *
388
		 * @param string $html The HTML to be displayed
389
		 */
390
		return apply_filters( 'jetpack_idc_first_step_header_explanation', $html );
391
	}
392
393 View Code Duplication
	function get_confirm_safe_mode_action_explanation() {
394
		$html = wp_kses(
395
			sprintf(
396
				__(
397
					'Is this website a temporary duplicate of <a href="%1$s">%2$s</a> for the purposes 
398
					of testing, staging or development? If so, we recommend keeping it in Safe Mode.',
399
					'jetpack'
400
				),
401
				esc_url( untrailingslashit( self::$wpcom_home_url ) ),
402
				self::prepare_url_for_display( esc_url( self::$wpcom_home_url ) )
403
			),
404
			array( 'a' => array( 'href' => array() ) )
405
		);
406
407
		/**
408
		 * Allows overriding of the default text used to explain the confirm safe mode action.
409
		 *
410
		 * @since 4.4.0
411
		 *
412
		 * @param string $html The HTML to be displayed
413
		 */
414
		return apply_filters( 'jetpack_idc_confirm_safe_mode_explanation', $html );
415
	}
416
417
	function get_confirm_safe_mode_button_text() {
418
		$string =  esc_html__( 'Confirm Safe Mode', 'jetpack' );
419
420
		/**
421
		 * Allows overriding of the default text used for the confirm safe mode action button.
422
		 *
423
		 * @since 4.4.0
424
		 *
425
		 * @param string $string The string to be displayed
426
		 */
427
		return apply_filters( 'jetpack_idc_confirm_safe_mode_button_text', $string );
428
	}
429
430 View Code Duplication
	function get_first_step_fix_connection_action_explanation() {
431
		$html = wp_kses(
432
			sprintf(
433
				__(
434
					'If this is a separate and new website, or the new home of <a href="%1$s">%2$s</a>, 
435
					we recommend turning Safe Mode off, and re-establishing your connection to WordPress.com.',
436
					'jetpack'
437
				),
438
				esc_url( untrailingslashit( self::$wpcom_home_url ) ),
439
				self::prepare_url_for_display( esc_url( self::$wpcom_home_url ) )
440
			),
441
			array( 'a' => array( 'href' => array() ) )
442
		);
443
444
		/**
445
		 * Allows overriding of the default text used to explain the fix Jetpack connection action.
446
		 *
447
		 * @since 4.4.0
448
		 *
449
		 * @param string $html The HTML to be displayed
450
		 */
451
		return apply_filters( 'jetpack_idc_first_fix_connection_explanation', $html );
452
	}
453
454
	function get_first_step_fix_connection_button_text() {
455
		$string = esc_html__( "Fix Jetpack's Connection", 'jetpack' );
456
457
		/**
458
		 * Allows overriding of the default text used for the fix Jetpack connection action button.
459
		 *
460
		 * @since 4.4.0
461
		 *
462
		 * @param string $string The string to be displayed
463
		 */
464
		return apply_filters( 'jetpack_idc_first_step_fix_connection_button_text', $string );
465
	}
466
467
	function get_second_step_header_lead() {
468
		$string = sprintf(
469
			esc_html__(
470
				'Is %1$s the new home of %2$s?',
471
				'jetpack'
472
			),
473
			untrailingslashit( Jetpack::normalize_url_protocol_agnostic( get_home_url() ) ),
474
			untrailingslashit( Jetpack::normalize_url_protocol_agnostic( esc_url_raw( self::$wpcom_home_url ) ) )
475
		);
476
477
		/**
478
		 * Allows overriding of the default header text in the second step of the Safe Mode notice.
479
		 *
480
		 * @since 4.4.0
481
		 *
482
		 * @param string $html The HTML to be displayed
483
		 */
484
		return apply_filters( 'jetpack_idc_second_step_header_lead', $string );
485
	}
486
487 View Code Duplication
	function get_migrate_site_action_explanation() {
488
		$html = wp_kses(
489
			sprintf(
490
				__(
491
					'Yes. <a href="%1$s">%2$s</a> is replacing <a href="%3$s">%4$s</a>. I would like to
492
					migrate my stats and subscribers from <a href="%3$s">%4$s</a> to <a href="%1$s">%2$s</a>.',
493
					'jetpack'
494
				),
495
				esc_url( get_home_url() ),
496
				self::prepare_url_for_display( get_home_url() ),
497
				esc_url( self::$wpcom_home_url ),
498
				untrailingslashit( Jetpack::normalize_url_protocol_agnostic( esc_url_raw( self::$wpcom_home_url ) ) )
499
			),
500
			array( 'a' => array( 'href' => array() ) )
501
		);
502
503
		/**
504
		 * Allows overriding of the default text for explaining the migrate site action.
505
		 *
506
		 * @since 4.4.0
507
		 *
508
		 * @param string $html The HTML to be displayed
509
		 */
510
		return apply_filters( 'jetpack_idc_migrate_site_explanation', $html );
511
	}
512
513
	function get_migrate_site_button_text() {
514
		$string = esc_html__( 'Migrate stats &amp; and Subscribers', 'jetpack' );
515
516
		/**
517
		 * Allows overriding of the default text used for the migrate site action button.
518
		 *
519
		 * @since 4.4.0
520
		 *
521
		 * @param string $string The string to be displayed
522
		 */
523
		return apply_filters( 'jetpack_idc_migrate_site_button_text', $string );
524
	}
525
526 View Code Duplication
	function get_start_fresh_action_explanation() {
527
		$html = wp_kses(
528
			sprintf(
529
				__(
530
					'No. <a href="%1$s">%2$s</a> is a new and different website that\'s separate from 
531
					<a href="%3$s">%4$s</a>. It requires  a new connection to WordPress.com for new stats and subscribers.',
532
					'jetpack'
533
				),
534
				esc_url( get_home_url() ),
535
				self::prepare_url_for_display( get_home_url() ),
536
				esc_url( self::$wpcom_home_url ),
537
				untrailingslashit( Jetpack::normalize_url_protocol_agnostic( esc_url_raw( self::$wpcom_home_url ) ) )
538
			),
539
			array( 'a' => array( 'href' => array() ) )
540
		);
541
542
		/**
543
		 * Allows overriding of the default text for explaining the start fresh action.
544
		 *
545
		 * @since 4.4.0
546
		 *
547
		 * @param string $html The HTML to be displayed
548
		 */
549
		return apply_filters( 'jetpack_idc_start_fresh_explanation', $html );
550
	}
551
552
	function get_start_fresh_button_text() {
553
		$string = esc_html__( 'Start fresh &amp; create new connection', 'jetpack' );
554
555
		/**
556
		 * Allows overriding of the default text used for the start fresh action button.
557
		 *
558
		 * @since 4.4.0
559
		 *
560
		 * @param string $string The string to be displayed
561
		 */
562
		return apply_filters( 'jetpack_idc_start_fresh_button_text', $string );
563
	}
564
565 View Code Duplication
	function get_unsure_prompt() {
566
		$html = wp_kses(
567
			sprintf(
568
				__(
569
					'Unsure what to do? <a href="%1$s">Read more about Jetpack Safe Mode</a>',
570
					'jetpack'
571
				),
572
				esc_url( self::SAFE_MODE_DOC_LINK )
573
			),
574
			array( 'a' => array( 'href' => array() ) )
575
		);
576
577
		/**
578
		 * Allows overriding of the default text using in the "Unsure what to do?" prompt.
579
		 *
580
		 * @since 4.4.0
581
		 *
582
		 * @param string $html The HTML to be displayed
583
		 */
584
		return apply_filters( 'jetpack_idc_unsure_prompt', $html );
585
	}
586
587 View Code Duplication
	function get_non_admin_notice_text() {
588
		$html = wp_kses(
589
			sprintf(
590
				__(
591
					'Jetpack has been placed into Safe Mode. Learn more about <a href="%1$s">Safe Mode</a>.',
592
					'jetpack'
593
				),
594
				esc_url( self::SAFE_MODE_DOC_LINK )
595
			),
596
			array( 'a' => array( 'href' => array() ) )
597
		);
598
599
		/**
600
		 * Allows overriding of the default text that is displayed to non-admin on the Jetpack admin page.
601
		 *
602
		 * @since 4.4.0
603
		 *
604
		 * @param string $html The HTML to be displayed
605
		 */
606
		return apply_filters( 'jetpack_idc_non_admin_notice_text', $html );
607
	}
608
609
	function get_non_admin_contact_admin_text() {
610
		$string = esc_html__( 'An administrator of this site can take Jetpack out of Safe Mode.', 'jetpack' );
611
612
		/**
613
		 * Allows overriding of the default text that is displayed to non-admins prompting them to contact an admin.
614
		 *
615
		 * @since 4.4.0
616
		 *
617
		 * @param string $string The string to be displayed
618
		 */
619
		return apply_filters( 'jetpack_idc_non_admin_contact_admin_text', $string );
620
	}
621
}
622
623
Jetpack_IDC::init();
624