blocked-login-page.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611
  1. <?php
  2. /**
  3. * Class Jetpack_Protect_Blocked_Login_Page
  4. *
  5. * Instanciated on the wp-login page when Jetpack modules are loaded and $pagenow
  6. * is available, or during the login_head hook.
  7. *
  8. * Class will only be instanciated if Protect has detected a hard blocked IP address.
  9. *
  10. *
  11. */
  12. class Jetpack_Protect_Blocked_Login_Page {
  13. private static $__instance = null;
  14. public $can_send_recovery_emails;
  15. public $ip_address;
  16. public $valid_blocked_user_id;
  17. public $email_address;
  18. const HELP_URL = 'https://jetpack.com/support/security-features/#unblock';
  19. const HTTP_STATUS_CODE_TOO_MANY_REQUESTS = 429;
  20. /**
  21. * Singleton implementation
  22. *
  23. * @return object
  24. */
  25. public static function instance( $ip_address ) {
  26. if ( ! is_a( self::$__instance, 'Jetpack_Protect_Blocked_Login_Page' ) ) {
  27. self::$__instance = new Jetpack_Protect_Blocked_Login_Page( $ip_address );
  28. }
  29. return self::$__instance;
  30. }
  31. function __construct( $ip_address ) {
  32. /**
  33. * Filter controls if an email recovery form is shown to blocked IPs.
  34. *
  35. * A recovery form allows folks to re-gain access to the login form
  36. * via an email link if their IP was mistakenly blocked.
  37. *
  38. * @module protect
  39. *
  40. * @since 5.6
  41. *
  42. * @param bool $can_send_recovery_emails Defaults to true.
  43. */
  44. $this->can_send_recovery_emails = apply_filters( 'jetpack_protect_can_send_recovery_emails', true );
  45. $this->ip_address = $ip_address;
  46. add_filter( 'wp_authenticate_user', array( $this, 'check_valid_blocked_user' ), 10, 1 );
  47. add_filter( 'site_url', array( $this, 'add_args_to_login_post_url' ), 10, 3 );
  48. add_filter( 'network_site_url', array( $this, 'add_args_to_login_post_url' ), 10, 3 );
  49. add_filter( 'lostpassword_url', array( $this, 'add_args_to_lostpassword_url' ), 10, 2 );
  50. add_filter( 'login_url', array( $this, 'add_args_to_login_url' ), 10, 3 );
  51. add_filter( 'lostpassword_redirect', array( $this, 'add_args_to_lostpassword_redirect_url' ), 10, 1 );
  52. }
  53. public function add_args_to_lostpassword_redirect_url( $url ) {
  54. if ( $this->valid_blocked_user_id ) {
  55. $url = empty( $url ) ? wp_login_url() : $url;
  56. $url = add_query_arg(
  57. array(
  58. 'validate_jetpack_protect_recovery' => $_GET['validate_jetpack_protect_recovery'],
  59. 'user_id' => $_GET['user_id'],
  60. 'checkemail' => 'confirm',
  61. ),
  62. $url
  63. );
  64. }
  65. return $url;
  66. }
  67. public function add_args_to_lostpassword_url( $url, $redirect ) {
  68. if ( $this->valid_blocked_user_id ) {
  69. $args = array(
  70. 'validate_jetpack_protect_recovery' => $_GET['validate_jetpack_protect_recovery'],
  71. 'user_id' => $_GET['user_id'],
  72. 'action' => 'lostpassword',
  73. );
  74. if ( ! empty( $redirect ) ) {
  75. $args['redirect_to'] = $redirect;
  76. }
  77. $url = add_query_arg( $args, $url );
  78. }
  79. return $url;
  80. }
  81. public function add_args_to_login_post_url( $url, $path, $scheme ) {
  82. if ( $this->valid_blocked_user_id && ( 'login_post' === $scheme || 'login' === $scheme ) ) {
  83. $url = add_query_arg(
  84. array(
  85. 'validate_jetpack_protect_recovery' => $_GET['validate_jetpack_protect_recovery'],
  86. 'user_id' => $_GET['user_id'],
  87. ),
  88. $url
  89. );
  90. }
  91. return $url;
  92. }
  93. public function add_args_to_login_url( $url, $redirect, $force_reauth ) {
  94. if ( $this->valid_blocked_user_id ) {
  95. $args = array(
  96. 'validate_jetpack_protect_recovery' => $_GET['validate_jetpack_protect_recovery'],
  97. 'user_id' => $_GET['user_id'],
  98. );
  99. if ( ! empty( $redirect ) ) {
  100. $args['redirect_to'] = $redirect;
  101. }
  102. if ( ! empty( $force_reauth ) ) {
  103. $args['reauth'] = '1';
  104. }
  105. $url = add_query_arg( $args, $url );
  106. }
  107. return $url;
  108. }
  109. public function check_valid_blocked_user( $user ) {
  110. if ( $this->valid_blocked_user_id && $this->valid_blocked_user_id != $user->ID ) {
  111. return new WP_Error( 'invalid_recovery_token', __( 'The recovery token is not valid for this user.', 'jetpack' ) );
  112. }
  113. return $user;
  114. }
  115. public function is_blocked_user_valid() {
  116. if ( ! $this->can_send_recovery_emails ) {
  117. return false;
  118. }
  119. if ( $this->valid_blocked_user_id ) {
  120. return true;
  121. }
  122. if ( ! isset( $_GET['validate_jetpack_protect_recovery'], $_GET['user_id'] ) ) {
  123. return false;
  124. }
  125. if ( ! $this->is_valid_protect_recovery_key( $_GET['validate_jetpack_protect_recovery'], $_GET['user_id'] ) ) {
  126. return false;
  127. }
  128. $this->valid_blocked_user_id = (int) $_GET['user_id'];
  129. return true;
  130. }
  131. public function is_valid_protect_recovery_key( $key, $user_id ) {
  132. $path = sprintf( '/sites/%d/protect/recovery/confirm', Jetpack::get_option( 'id' ) );
  133. $response = Jetpack_Client::wpcom_json_api_request_as_blog(
  134. $path,
  135. '1.1',
  136. array(
  137. 'method' => 'post'
  138. ),
  139. array(
  140. 'token' => $key,
  141. 'user_id' => $user_id,
  142. 'ip' => $this->ip_address,
  143. )
  144. );
  145. $result = json_decode( wp_remote_retrieve_body( $response ) );
  146. if ( is_wp_error( $result ) || empty( $result ) || isset( $result->error ) ) {
  147. return false;
  148. }
  149. return true;
  150. }
  151. public function render_and_die() {
  152. if ( ! $this->can_send_recovery_emails ) {
  153. $this->render_blocked_login_message();
  154. return;
  155. }
  156. if ( isset( $_GET['validate_jetpack_protect_recovery'] ) && $_GET['user_id'] ) {
  157. $error = new WP_Error( 'invalid_token', __( "Oops, we couldn't validate the recovery token.", 'jetpack' ) );
  158. $this->protect_die( $error );
  159. return;
  160. }
  161. if (
  162. isset( $_GET['jetpack-protect-recovery'] ) &&
  163. isset( $_POST['_wpnonce'] ) &&
  164. wp_verify_nonce( $_POST['_wpnonce'], 'bypass-protect' )
  165. ) {
  166. $this->process_recovery_email();
  167. return;
  168. }
  169. if ( isset( $_GET['loggedout'] ) && 'true' === $_GET['loggedout'] ) {
  170. $this->protect_die( __( 'You successfully logged out.', 'jetpack' ) );
  171. }
  172. $this->render_recovery_form();
  173. }
  174. public function render_blocked_login_message() {
  175. $this->protect_die( $this->get_html_blocked_login_message() );
  176. }
  177. function process_recovery_email() {
  178. $sent = $this->send_recovery_email();
  179. $show_recovery_form = true;
  180. if ( is_wp_error( $sent ) ) {
  181. if ( 'email_already_sent' === $sent->get_error_code() ) {
  182. $show_recovery_form = false;
  183. }
  184. $this->protect_die( $sent,null,true, $show_recovery_form );
  185. } else {
  186. $this->render_recovery_success();
  187. }
  188. }
  189. function send_recovery_email() {
  190. $email = isset( $_POST['email'] ) ? $_POST['email'] : '';
  191. if ( sanitize_email( $email ) !== $email || ! is_email( $email ) ) {
  192. return new WP_Error( 'invalid_email', __( "Oops, looks like that's not the right email address. Please try again!", 'jetpack' ) );
  193. }
  194. $user = get_user_by( 'email', trim( $email ) );
  195. if ( ! $user ) {
  196. return new WP_Error( 'invalid_user', __( "Oops, we couldn't find a user with that email. Please try again!", 'jetpack' ) );
  197. }
  198. $this->email_address = $email;
  199. $path = sprintf( '/sites/%d/protect/recovery/request', Jetpack::get_option( 'id' ) );
  200. $response = Jetpack_Client::wpcom_json_api_request_as_blog(
  201. $path,
  202. '1.1',
  203. array(
  204. 'method' => 'post'
  205. ),
  206. array(
  207. 'user_id' => $user->ID,
  208. 'ip' => $this->ip_address
  209. )
  210. );
  211. $code = wp_remote_retrieve_response_code( $response );
  212. $result = json_decode( wp_remote_retrieve_body( $response ) );
  213. if ( self::HTTP_STATUS_CODE_TOO_MANY_REQUESTS === $code ) {
  214. return new WP_Error( 'email_already_sent', sprintf( __( 'Recovery instructions were sent to %s. Check your inbox!', 'jetpack' ), $this->email_address ) );
  215. } else if ( is_wp_error( $result ) || empty( $result ) || isset( $result->error ) ) {
  216. return new WP_Error( 'email_send_error', __( 'Oops, we were unable to send a recovery email. Try again.', 'jetpack' ) );
  217. }
  218. return true;
  219. }
  220. function protect_die( $content, $title = null, $back_link = false, $recovery_form = false ) {
  221. if ( empty( $title ) ) {
  222. $title = __( 'Jetpack has locked your site\'s login page.', 'jetpack' );
  223. }
  224. if ( is_wp_error( $content ) ) {
  225. $svg = '<svg class="gridicon gridicons-notice" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M12 2C6.477 2 2 6.477 2 12s4.477 10 10 10 10-4.477 10-10S17.523 2 12 2zm1 15h-2v-2h2v2zm0-4h-2l-.5-6h3l-.5 6z"/></g></svg>';
  226. $content = '<span class="error"> '. $svg . $content->get_error_message() . '</span>';
  227. }
  228. $content = '<p>'. $content .'</p>';
  229. // If for some reason the login pop up box show up in the wp-admin.
  230. if ( isset( $_GET['interim-login'] ) ) {
  231. $content = "<style>html{ background-color: #fff; } #error-message { margin:0 auto; padding: 1em; box-shadow: none; } </style>" . $content;
  232. }
  233. $this->display_page( $title, $content, $back_link, $recovery_form );
  234. }
  235. function render_recovery_form() {
  236. $content = $this->get_html_blocked_login_message();
  237. $this->protect_die( $content, null, null, true );
  238. }
  239. function render_recovery_success() {
  240. $this->protect_die( sprintf( __( 'Recovery instructions were sent to %s. Check your inbox!', 'jetpack' ), $this->email_address ) );
  241. }
  242. function get_html_blocked_login_message() {
  243. $icon = '<svg class="gridicon gridicons-spam" style="fill:#d94f4f" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M17 2H7L2 7v10l5 5h10l5-5V7l-5-5zm-4 15h-2v-2h2v2zm0-4h-2l-.5-6h3l-.5 6z"/></g></svg>';
  244. $ip = str_replace( 'http://', '', esc_url( 'http://' . $this->ip_address ) );
  245. return sprintf(
  246. __( '<p>Your IP address <code>%2$s</code> has been flagged for potential security violations. You can unlock your login by sending yourself a special link via email. <a href="%3$s">Learn More</a></p>', 'jetpack' ),
  247. $icon,
  248. $ip,
  249. esc_url( self::HELP_URL )
  250. );
  251. }
  252. function get_html_recovery_form() {
  253. ob_start(); ?>
  254. <div>
  255. <form method="post" action="?jetpack-protect-recovery=true">
  256. <?php echo wp_nonce_field( 'bypass-protect' ); ?>
  257. <p><label for="email"><?php esc_html_e( 'Your email', 'jetpack' ); ?><br/></label>
  258. <input type="email" name="email" class="text-input"/>
  259. <input type="submit" class="button"
  260. value="<?php esc_attr_e( 'Send email', 'jetpack' ); ?>"/>
  261. </p>
  262. </form>
  263. </div>
  264. <?php
  265. $contents = ob_get_contents();
  266. ob_end_clean();
  267. return $contents;
  268. }
  269. function display_page( $title, $message, $back_button = false, $recovery_form = false ) {
  270. if ( ! headers_sent() ) {
  271. nocache_headers();
  272. header( 'Content-Type: text/html; charset=utf-8' );
  273. }
  274. $text_direction = 'ltr';
  275. if ( is_rtl() ) {
  276. $text_direction = 'rtl';
  277. }
  278. ?>
  279. <!DOCTYPE html>
  280. <html xmlns="http://www.w3.org/1999/xhtml" <?php if ( function_exists( 'language_attributes' ) && function_exists( 'is_rtl' ) ) {
  281. language_attributes();
  282. } else {
  283. echo "dir='$text_direction'";
  284. } ?>>
  285. <head>
  286. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  287. <meta name="viewport" content="width=device-width">
  288. <?php
  289. if ( function_exists( 'wp_no_robots' ) ) {
  290. wp_no_robots();
  291. }
  292. ?>
  293. <title><?php echo $title ?></title>
  294. <style type="text/css">
  295. html {
  296. background: #f3f6f8;
  297. }
  298. body {
  299. color: #2e4453;
  300. font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif;
  301. margin: 2em auto;
  302. padding: 1em 2em;
  303. max-width: 460px;
  304. text-align: left;
  305. }
  306. body.is-rtl {
  307. text-align: right;
  308. }
  309. h1 {
  310. clear: both;
  311. color: #3d596d;
  312. font-size: 24px;
  313. margin:0 0 24px 0;
  314. padding: 0;
  315. font-weight: 400;
  316. }
  317. #error-message {
  318. box-sizing: border-box;
  319. background: white;
  320. box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3;
  321. padding: 24px;
  322. }
  323. #error-message img {
  324. margin: 0 auto;
  325. display: block;
  326. }
  327. #error-page {
  328. margin-top: 50px;
  329. }
  330. #error-page p {
  331. font-size: 14px;
  332. line-height: 1.5;
  333. margin: 24px 0 0;
  334. }
  335. #error-page code {
  336. font-family: Consolas, Monaco, monospace;
  337. }
  338. ul li {
  339. margin-bottom: 10px;
  340. font-size: 14px;
  341. }
  342. a {
  343. color: #00aadc;
  344. }
  345. label {
  346. font-weight: bold;
  347. font-size:16px;
  348. }
  349. a:hover,
  350. a:active {
  351. color: #0085be;
  352. }
  353. a:focus {
  354. color: #124964;
  355. -webkit-box-shadow: 0 0 0 1px #5b9dd9,
  356. 0 0 2px 1px rgba(30, 140, 190, .8);
  357. box-shadow: 0 0 0 1px #5b9dd9,
  358. 0 0 2px 1px rgba(30, 140, 190, .8);
  359. outline: none;
  360. }
  361. .button {
  362. background: #00aadc;
  363. color: white;
  364. border-color: #008ab3;
  365. border-style: solid;
  366. border-width: 1px 1px 2px;
  367. cursor: pointer;
  368. display: inline-block;
  369. margin: 0;
  370. margin-right: 0px;
  371. outline: 0;
  372. overflow: hidden;
  373. font-weight: 500;
  374. text-overflow: ellipsis;
  375. text-decoration: none;
  376. vertical-align: top;
  377. box-sizing: border-box;
  378. font-size: 14px;
  379. line-height: 21px;
  380. border-radius: 4px;
  381. padding: 7px 14px 9px;
  382. -webkit-appearance: none;
  383. -moz-appearance: none;
  384. appearance: none;
  385. font-size: 14px;
  386. width: 100%;
  387. }
  388. .button:hover,
  389. .button:focus {
  390. border-color: #005082;
  391. outline: none;
  392. }
  393. .button:focus {
  394. border-color: #005082;
  395. -webkit-box-shadow: 0 0 3px rgba(0, 115, 170, .8);
  396. box-shadow: 0 0 3px rgba(0, 115, 170, .8);
  397. outline: none;
  398. }
  399. .button::-moz-focus-inner {
  400. border: 0;
  401. }
  402. .button:active {
  403. border-width: 2px 1px 1px;
  404. }
  405. .gridicon {
  406. fill: currentColor;
  407. vertical-align: middle;
  408. }
  409. #error-footer {
  410. padding: 16px;
  411. }
  412. #error-footer a {
  413. text-decoration: none;
  414. line-height:20px;
  415. font-size: 14px;
  416. color: #4f748e;
  417. }
  418. #error-footer a:hover {
  419. color: #2e4453;
  420. }
  421. #error-footer .gridicon{
  422. width: 16px;
  423. }
  424. #error-footer .gridicons-help {
  425. width: 24px;
  426. margin-right:8px;
  427. }
  428. .is-rtl #error-footer .gridicons-help {
  429. margin-left:8px;
  430. }
  431. .error {
  432. background: #d94f4f;
  433. color:#FFF;
  434. display: block;
  435. border-radius: 3px;
  436. line-height: 1.5;
  437. padding: 16px;
  438. padding-left: 42px;
  439. }
  440. .is-rtl .error {
  441. padding-right: 42px;
  442. }
  443. .error .gridicon {
  444. float: left;
  445. margin-left: -32px;
  446. }
  447. .is-rtl .error .gridicon {
  448. float: right;
  449. margin-right: -32px;
  450. }
  451. .text-input {
  452. margin: 0;
  453. padding: 7px 14px;
  454. width: 100%;
  455. color: #2e4453;
  456. font-size: 16px;
  457. line-height: 1.5;
  458. border: 1px solid #c8d7e1;
  459. background-color: white;
  460. transition: all .15s ease-in-out;
  461. box-sizing: border-box;
  462. margin: 8px 0 16px;
  463. }
  464. #image {
  465. display: block;
  466. width: 200px;
  467. margin: 0 auto;
  468. }
  469. <?php
  470. $rtl_class = '';
  471. if ( 'rtl' == $text_direction ) {
  472. $rtl_class = 'class="is-rtl"';
  473. echo 'body { font-family: Tahoma, Arial; }';
  474. }
  475. ?>
  476. </style>
  477. </head>
  478. <body id="error-page" <?php echo $rtl_class; ?>>
  479. <h1 id="error-title"><?php echo esc_html( $title ); ?></h1>
  480. <div id="error-message">
  481. <svg id="image" xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" viewBox="0 0 250 134">
  482. <path fill="#E9EFF4" d="M205.2,129.8c3.7-0.7,7.4-0.9,11.1-1.1l5.5-0.1l5.5,0c3.7,0,7.4,0.1,11.1,0.2c3.7,0.1,7.4,0.3,11.1,0.8 c0.3,0,0.5,0.3,0.5,0.6c0,0.2-0.2,0.4-0.5,0.5c-3.7,0.5-7.4,0.6-11.1,0.8c-3.7,0.1-7.4,0.2-11.1,0.2l-5.5,0l-5.5-0.1 c-3.7-0.1-7.4-0.4-11.1-1.1c-0.1,0-0.2-0.2-0.2-0.3C205,129.9,205.1,129.8,205.2,129.8"/>
  483. <path fill="#E9EFF4" d="M0.2,130.9c3-0.7,5.9-0.9,8.9-1.1l4.4-0.1l4.4,0c3,0,5.9,0.1,8.9,0.2c3,0.1,5.9,0.3,8.9,0.8 c0.3,0,0.5,0.3,0.4,0.6c0,0.2-0.2,0.4-0.4,0.4c-3,0.5-5.9,0.6-8.9,0.8c-3,0.1-5.9,0.2-8.9,0.2l-4.4,0l-4.4-0.1 c-3-0.1-5.9-0.4-8.9-1.1c-0.1,0-0.2-0.2-0.2-0.3C0,131,0.1,130.9,0.2,130.9"/>
  484. <path fill="#C8D7E2" d="M101.6,130.1H70.1V52.5c0-8.5,6.9-15.3,15.3-15.3h16.1V130.1z"/>
  485. <path fill="#0DA9DD" d="M191.5,130.1h-73.8v-5.4c0-8.9,7.2-16.1,16.1-16.1h57.7V130.1z"/>
  486. <path fill="#C7E9F5" d="M55.2,25.6l-0.1,9.8L55,57l-0.1,21.6c0,0.2,0.2,0.4,0.4,0.4c0.2,0,0.4-0.2,0.4-0.4L56.6,57l0.8-21.6 c0.1-3.3,0.2-6.5,0.3-9.8H55.2z"/>
  487. <path fill="#C7E9F5" d="M203.1,25.6l0.1,18.1c0.2,28.8,0.4,57.6,1.2,86.3c0,0.4,0.4,0.8,0.8,0.8c0.4,0,0.8-0.3,0.8-0.8 c0.8-28.8,1-57.6,1.2-86.3l0.1-18.1H203.1z"/>
  488. <path fill="#7FD3F2" d="M55.3,25.6v-8.2v-6.8c0-5.9,4-10.7,9-10.7h134c5,0,9,4.8,9,10.7v14.9H55.3z"/>
  489. <path fill="#005083" d="M210.7,25.6c-13.3,1.1-26.7,1-40,1l-40,0.2l-40-0.2c-13.3-0.1-26.7,0-40-1V25c13.3-1.1,26.7-1,40-1l40-0.2 l40,0.2c13.3,0.1,26.7,0,40,1V25.6z"/>
  490. <polygon fill="#C7E9F5" points="168.7,95.6 117.7,95.6 117.7,44.6 "/>
  491. <path fill="#C8D7E2" d="M191.5,56.5c0,11-8.9,19.9-19.9,19.9c-11,0-19.9-8.9-19.9-19.9c0-11,8.9-19.9,19.9-19.9 C182.6,36.6,191.5,45.5,191.5,56.5"/>
  492. <path fill="#FFFFFF" d="M213.2,95.5c-3.3-5.1-3.2-16.7-3.2-28.4h-32.3c0,0-5.2,25.5,4.6,33c7.5-0.1,29.9-0.6,29.9-0.6"/>
  493. <path fill="#C8D7E2" d="M213.5,95.3l-0.1-0.1l-0.3-0.5c-0.2-0.4-0.3-0.7-0.5-1.1c-0.3-0.8-0.5-1.6-0.7-2.4c-0.1-0.5-0.2-1.1-0.3-1.6 c-0.4,0-0.8,0-1.2,0c0.5,2.1,1.1,4.3,2.4,6.1l0.2,0.2c0.2,0,0.4-0.1,0.5-0.3C213.6,95.5,213.6,95.4,213.5,95.3L213.5,95.3z"/>
  494. <path fill="#C8D7E2" d="M212.5,98.6c-0.1,0-0.2,0-0.3,0l-0.1,0H212l-0.3,0l-0.6,0l-1.3,0l-2.5,0l-5,0l-19.5,0.2 c-1.9-1.7-3.1-4.1-3.8-6.5c-0.8-2.6-1.1-5.4-1.2-8.2c-0.2-5.2,0.3-10.4,1.1-15.6l5.7-0.1c0-0.9,0-1.8,0-2.6l-4.4,0l-2.5,0 c-0.4,0-0.8,0.2-1,0.5c-0.1,0.2-0.2,0.3-0.3,0.5l-0.1,0.3l-0.2,1.2c-0.3,1.7-0.5,3.3-0.7,5c-0.3,3.3-0.5,6.7-0.4,10.1 c0.1,3.4,0.5,6.7,1.5,10c0.5,1.6,1.2,3.2,2.2,4.7c0.5,0.7,1,1.4,1.7,2c0.3,0.3,0.6,0.6,1,0.9l0.1,0.1c0.1,0,0.2,0.1,0.3,0.2 c0.2,0.1,0.5,0.1,0.6,0.1l0.6,0l20-0.6l5-0.2l2.5-0.1l1.2,0l0.3,0l0.2,0c0,0,0.3,0,0.4-0.1c0.3-0.2,0.5-0.5,0.5-0.9 C213.1,99.1,212.9,98.7,212.5,98.6z"/>
  495. <path fill="#FFFFFF" d="M223.1,84.8c-3.3-5.1-4.8-16.7-4.8-28.4h-32.3c0,0-3.5,25.5,6.3,33c7.5-0.1,29.9-0.6,29.9-0.6"/>
  496. <path fill="#C8D7E2" d="M222.9,84.9c-1.3-2.1-2.2-4.4-2.8-6.7c-0.6-2.4-1.1-4.8-1.5-7.2c-0.7-4.8-1-9.1-1-13.9l0,0l-31,0.1l0,0 c-0.4,2.8-0.5,5.1-0.5,7.9c-0.1,2.9,0,5.7,0.3,8.6c0.3,2.8,0.8,5.7,1.7,8.3c0.9,2.6,2.3,5.2,4.5,6.9l-0.4-0.1l14.9-0.2 c5-0.1,10-0.1,14.9-0.1c0.1,0,0.3,0.1,0.3,0.3c0,0.1-0.1,0.3-0.2,0.3c-5,0.2-10,0.4-14.9,0.5l-14.9,0.4c-0.1,0-0.3,0-0.4-0.1l0,0 c-2.5-1.9-3.9-4.7-5-7.4c-1-2.8-1.5-5.7-1.9-8.6c-0.3-2.9-0.4-5.8-0.4-8.8c0.1-2.9,0.2-5.8,0.6-8.8c0-0.4,0.4-0.6,0.7-0.6h0 l32.3,0.1h0c0.3,0,0.6,0.3,0.6,0.6v0c0,4.8,0.2,9.6,0.7,14.4c0.3,2.4,0.6,4.8,1.2,7.1c0.5,2.3,1.2,4.7,2.4,6.8c0,0.1,0,0.1,0,0.2 C223.1,85,223,85,222.9,84.9"/>
  497. <path fill="#C8D7E2" d="M192.1,67.1c1.6-0.9,3.4-1.2,5.1-1.3c1.7-0.2,3.5-0.2,5.2-0.2c3.5,0.1,6.9,0.2,10.3,1c0.1,0,0.2,0.2,0.2,0.3 c0,0.1-0.1,0.2-0.2,0.2c-3.4,0.2-6.9,0-10.3,0c-1.7,0-3.4,0-5.1,0c-1.7,0-3.4,0.1-5.1,0.3l0,0c-0.1,0-0.1,0-0.1-0.1 C192,67.2,192.1,67.1,192.1,67.1"/>
  498. <path fill="#C8D7E2" d="M194.1,74c1.4,0,2.7,0,4.1,0c1.4,0,2.7,0,4.1,0c2.7,0,5.4-0.1,8.2-0.2c0.1,0,0.3,0.1,0.3,0.3 c0,0.1-0.1,0.2-0.2,0.3c-1.3,0.5-2.7,0.7-4.1,0.9c-1.4,0.2-2.8,0.2-4.2,0.3c-1.4,0-2.8,0-4.2-0.2c-1.4-0.2-2.8-0.4-4.1-1.1 c-0.1,0-0.1-0.1,0-0.2C193.9,74.1,194,74,194.1,74L194.1,74z"/>
  499. <path fill="#86A6BD" d="M40.2,88.6c-0.5,0-0.8-0.4-0.9-0.9l-0.1-8.2c0-0.7,0-1.4,0-2.1c0.1-0.7,0.2-1.5,0.4-2.2c0.4-1.4,1-2.8,1.9-4 c1.7-2.5,4.3-4.3,7.1-5.1c0.7-0.2,1.5-0.3,2.2-0.5c0.7-0.1,1.5-0.1,2.2-0.1c1.3,0,2.9,0,4.4,0.4c2.9,0.7,5.6,2.5,7.4,4.9 c0.9,1.2,1.6,2.6,2.1,4c0.5,1.4,0.6,3,0.6,4.4l0,16.4c0,0.7-0.6,1.3-1.3,1.3l-6.7,0c-0.7,0-1.3-0.6-1.3-1.3v0l0-10.8l0-5.4 c0-1.4-0.7-2.8-1.8-3.5c-0.6-0.4-1.3-0.6-2-0.7c-0.7,0-1.9,0-2.5,0c-1.4,0.1-2.7,1-3.3,2.3c-0.3,0.7-0.4,1.3-0.4,2.1l0,2.7 l-0.1,5.4l0,0c0,0.5-0.4,0.9-1,0.9"/>
  500. <path fill="#FFFFFF" d="M41.1,86.9l0.1-7.3c-0.1-2.6,0.7-5,2.1-7.1c1.4-2,3.6-3.5,5.9-4.1c0.6-0.2,1.2-0.3,1.8-0.3 c0.6,0,1.2-0.1,1.9,0c1.4,0,2.5,0,3.7,0.4c2.4,0.6,4.5,2,5.9,4c0.7,1,1.3,2.1,1.6,3.2c0.4,1.2,0.5,2.3,0.5,3.7l0,15.1l0,0l-4.2,0 l0-9.5l0-5.4c0-2.2-1.2-4.4-3-5.5c-0.9-0.6-2-0.9-3.1-1c-1.1,0-1.7,0-2.9,0c-2.2,0.2-4.2,1.7-5.1,3.6c-0.5,0.9-0.7,2.1-0.6,3.1 l0,2.7l0.1,4.4l0,0L41.1,86.9L41.1,86.9"/>
  501. <path fill="#86A6BD" d="M36.3,133c-1.9,0-3.8-1.1-4.8-2.8c-0.5-0.8-0.7-1.8-0.7-2.8l0-2.4l0-9.6l-0.1-9.6l0-4.8c0-0.7,0-1.8,0.3-2.8 c0.3-1,0.9-1.8,1.7-2.5c0.8-0.6,1.7-1.1,2.7-1.3c1.1-0.2,1.8-0.1,2.6-0.1l4.8,0l9.6-0.1l19.2,0c2.1,0,4.1,1.2,5.1,3 c0.5,0.9,0.8,2,0.8,3l0,2.4l0,9.6l-0.1,9.6l0,4.8c0,0.7,0,1.8-0.4,2.8c-0.3,0.9-1,1.8-1.7,2.4c-0.8,0.6-1.7,1.1-2.7,1.2 c-1.1,0.1-1.8,0-2.6,0.1l-4.8,0l-9.6-0.1L36.3,133z"/>
  502. <path fill="#FFFFFF" d="M74.8,112.3l-0.1-9.6l0-2.4c0-0.6-0.1-1.1-0.4-1.6c-0.6-1-1.7-1.6-2.8-1.6l-19.2,0L42.7,97l-4.8,0 c-0.8,0-1.7,0-2.2,0c-0.6,0.1-1.1,0.3-1.6,0.7c-0.5,0.4-0.8,0.9-1,1.4c-0.2,0.6-0.2,1.1-0.2,2l0,4.8l-0.1,9.6l0,9.6l0,2.4 c0,0.6,0.2,1.3,0.5,1.8c0.6,1.1,1.9,1.8,3.1,1.8l19.2-0.1l9.6-0.1l4.8,0c0.8,0,1.7,0,2.2-0.1c0.6-0.1,1.2-0.4,1.6-0.8 c0.5-0.4,0.8-0.9,1-1.5c0.2-0.6,0.2-1.1,0.2-2l0-4.8L74.8,112.3z"/>
  503. <path fill="#86A6BD" d="M48.1,121.4l2.9-6.2c0.3-0.6,0.2-1.3-0.3-1.8c-1-1-1.5-2.5-1.2-4c0.3-1.7,1.7-3.1,3.4-3.4 c2.9-0.6,5.4,1.6,5.4,4.4c0,1.2-0.5,2.3-1.3,3.1c-0.5,0.5-0.6,1.2-0.3,1.8l2.9,6.2c0.1,0.2-0.1,0.5-0.3,0.5H48.4 C48.1,121.9,48,121.6,48.1,121.4"/>
  504. </svg>
  505. <?php echo $message; ?>
  506. <?php if ( $recovery_form ) {
  507. echo $this->get_html_recovery_form();
  508. } ?>
  509. </div>
  510. <div id="error-footer">
  511. <?php if ( $back_button && ! $recovery_form ) {
  512. if ( 'rtl' == $text_direction ) {
  513. $back_button_icon = '<svg class="gridicon gridicons-arrow-right" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8-8-8z"/></g></svg>';
  514. } else {
  515. $back_button_icon = '<svg class="gridicon gridicons-arrow-left" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M20 11H7.83l5.59-5.59L12 4l-8 8 8 8 1.41-1.41L7.83 13H20v-2z"/></g></svg>';
  516. }
  517. ?>
  518. <a href='javascript:history.back()'><?php printf( __( '%s Back' ), $back_button_icon ); ?></a>
  519. <?php } else {
  520. $help_icon = '<svg class="gridicon gridicons-help" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M12 2C6.477 2 2 6.477 2 12s4.477 10 10 10 10-4.477 10-10S17.523 2 12 2zm1 16h-2v-2h2v2zm0-4.14V15h-2v-2c0-.552.448-1 1-1 1.103 0 2-.897 2-2s-.897-2-2-2-2 .897-2 2H8c0-2.21 1.79-4 4-4s4 1.79 4 4c0 1.862-1.278 3.413-3 3.86z"/></g></svg>';?>
  521. <a href="<?php echo esc_url( self::HELP_URL ); ?>" rel="noopener noreferrer" target="_blank"><?php printf( __( '%s Get help unlocking your site' ), $help_icon );?></a>
  522. <?php } ?>
  523. </div>
  524. </body>
  525. </html>
  526. <?php
  527. die();
  528. }
  529. }