Source: includes/classes/AdminNotices.php

  1. <?php
  2. /**
  3. * ElasticPress admin notice handler
  4. *
  5. * phpcs:disable WordPress.WP.I18n.MissingTranslatorsComment
  6. *
  7. * @since 3.0
  8. * @package elasticpress
  9. */
  10. namespace ElasticPress;
  11. use ElasticPress\Utils;
  12. use ElasticPress\Elasticsearch;
  13. use ElasticPress\Screen;
  14. use ElasticPress\Features;
  15. use ElasticPress\Indexables;
  16. use ElasticPress\Stats;
  17. if ( ! defined( 'ABSPATH' ) ) {
  18. exit; // Exit if accessed directly.
  19. }
  20. /**
  21. * Admin notices class
  22. */
  23. class AdminNotices {
  24. /**
  25. * Notice keys
  26. *
  27. * @since 3.0
  28. * @var array
  29. */
  30. protected $notice_keys = [
  31. 'host_error',
  32. 'es_below_compat',
  33. 'es_above_compat',
  34. 'different_server_type',
  35. 'need_setup',
  36. 'no_sync',
  37. 'upgrade_sync',
  38. 'auto_activate_sync',
  39. 'using_autosuggest_defaults',
  40. 'maybe_wrong_mapping',
  41. 'yellow_health',
  42. 'too_many_fields',
  43. ];
  44. /**
  45. * Notices that should be shown on a screen
  46. *
  47. * @var array
  48. * @since 3.0
  49. */
  50. protected $notices = [];
  51. /**
  52. * Process all notifications and prepare ones that should be displayed
  53. *
  54. * @since 3.0
  55. */
  56. public function process_notices() {
  57. $this->notices = [];
  58. foreach ( $this->notice_keys as $notice ) {
  59. $output = call_user_func( [ $this, 'process_' . $notice . '_notice' ] );
  60. if ( ! empty( $output ) ) {
  61. $this->notices[ $notice ] = $output;
  62. }
  63. }
  64. }
  65. /**
  66. * Autosuggest defaults are being used. Feature must be active.
  67. *
  68. * Type: notice
  69. * Dismiss: Everywhere
  70. * Show: Settings and dashboard only
  71. *
  72. * @since 3.0
  73. * @return array|bool
  74. */
  75. protected function process_using_autosuggest_defaults_notice() {
  76. $feature = Features::factory()->get_registered_feature( 'autosuggest' );
  77. if ( ! $feature instanceof Feature ) {
  78. return false;
  79. }
  80. if ( ! $feature->is_active() ) {
  81. return false;
  82. }
  83. $last_sync = Utils\get_option( 'ep_last_sync', false );
  84. if ( empty( $last_sync ) ) {
  85. return false;
  86. }
  87. $host = Utils\get_host();
  88. if ( empty( $host ) ) {
  89. return false;
  90. }
  91. $dismiss = Utils\get_option( 'ep_hide_using_autosuggest_defaults_notice', false );
  92. if ( $dismiss ) {
  93. return false;
  94. }
  95. $screen = Screen::factory()->get_current_screen();
  96. if ( ! in_array( $screen, [ 'dashboard', 'settings' ], true ) ) {
  97. return false;
  98. }
  99. return [
  100. 'html' => sprintf( esc_html__( 'Autosuggest feature is enabled. If documents feature is enabled, your media will also become searchable in the frontend.', 'elasticpress' ) ),
  101. 'type' => 'info',
  102. 'dismiss' => true,
  103. ];
  104. }
  105. /**
  106. * An EP feature was auto-activated that requires a sync.
  107. *
  108. * Type: warning
  109. * Dismiss: Everywhere except ep dash and settings
  110. * Show: All screens except install. Dont show during sync ever
  111. *
  112. * @since 3.0
  113. * @return array|bool
  114. */
  115. protected function process_auto_activate_sync_notice() {
  116. $need_upgrade_sync = Utils\get_option( 'ep_need_upgrade_sync', false );
  117. // need_upgrade_sync takes priority over this notice
  118. if ( $need_upgrade_sync ) {
  119. return false;
  120. }
  121. $auto_activate_sync = Utils\get_option( 'ep_feature_auto_activated_sync', false );
  122. if ( ! $auto_activate_sync ) {
  123. return false;
  124. }
  125. $last_sync = Utils\get_option( 'ep_last_sync', false );
  126. if ( empty( $last_sync ) ) {
  127. return false;
  128. }
  129. $host = Utils\get_host();
  130. if ( empty( $host ) ) {
  131. return false;
  132. }
  133. $dismiss = Utils\get_option( 'ep_hide_auto_activate_sync_notice', false );
  134. $screen = Screen::factory()->get_current_screen();
  135. if ( 'install' === $screen ) {
  136. return false;
  137. }
  138. if ( $dismiss && ! in_array( $screen, [ 'dashboard', 'settings' ], true ) ) {
  139. return false;
  140. }
  141. if ( Utils\isset_do_sync_parameter() ) {
  142. return false;
  143. }
  144. $url = Utils\get_sync_url( 'features' );
  145. $feature = Features::factory()->get_registered_feature( $auto_activate_sync );
  146. if ( defined( 'EP_DASHBOARD_SYNC' ) && ! EP_DASHBOARD_SYNC ) {
  147. $html = sprintf(
  148. /* translators: Feature name */
  149. esc_html__( 'Dashboard sync is disabled. The ElasticPress %s feature has been auto-activated! You will need to reindex using WP-CLI for it to work.', 'elasticpress' ),
  150. esc_html( is_object( $feature ) ? $feature->get_short_title() : '' )
  151. );
  152. } else {
  153. $html = sprintf(
  154. /* translators: 1. Feature name; 2: Sync page URL */
  155. __( 'The ElasticPress %1$s feature has been auto-activated! You will need to <a href="%2$s">run a sync</a> for it to work.', 'elasticpress' ),
  156. esc_html( is_object( $feature ) ? $feature->get_short_title() : '' ),
  157. esc_url( $url )
  158. );
  159. }
  160. return [
  161. 'html' => $html,
  162. 'type' => 'warning',
  163. 'dismiss' => ! in_array( $screen, [ 'dashboard', 'settings' ], true ),
  164. ];
  165. }
  166. /**
  167. * EP was upgraded to or past a version that requires a sync.
  168. *
  169. * Type: warning
  170. * Dismiss: Everywhere except ep dash and settings
  171. * Show: All screens except install. Dont show during sync ever
  172. *
  173. * @since 3.0
  174. * @return array|bool
  175. */
  176. protected function process_upgrade_sync_notice() {
  177. $need_upgrade_sync = Utils\get_option( 'ep_need_upgrade_sync', false );
  178. if ( ! $need_upgrade_sync ) {
  179. return false;
  180. }
  181. $last_sync = Utils\get_option( 'ep_last_sync', false );
  182. if ( empty( $last_sync ) ) {
  183. return false;
  184. }
  185. $host = Utils\get_host();
  186. if ( empty( $host ) ) {
  187. return false;
  188. }
  189. $dismiss = Utils\get_option( 'ep_hide_upgrade_sync_notice', false );
  190. $screen = Screen::factory()->get_current_screen();
  191. if ( 'install' === $screen ) {
  192. return false;
  193. }
  194. if ( $dismiss && ! in_array( $screen, [ 'dashboard', 'settings' ], true ) ) {
  195. return false;
  196. }
  197. if ( Utils\isset_do_sync_parameter() ) {
  198. return false;
  199. }
  200. $url = Utils\get_sync_url( 'upgrade' );
  201. if ( defined( 'EP_DASHBOARD_SYNC' ) && ! EP_DASHBOARD_SYNC ) {
  202. $html = esc_html__( 'Dashboard sync is disabled. The new version of ElasticPress requires that you delete all data and start a fresh sync using WP-CLI.', 'elasticpress' );
  203. } else {
  204. $html = sprintf(
  205. /* translators: Sync Page URL */
  206. __( 'The new version of ElasticPress requires that you <a href="%s">delete all data and start a fresh sync</a>.', 'elasticpress' ),
  207. esc_url( $url )
  208. );
  209. }
  210. $notice = esc_html__( 'Please note that some ElasticPress functionality may be impaired and/or content may not be searchable until the full sync has been performed.', 'elasticpress' );
  211. return [
  212. 'html' => '<span class="dashicons dashicons-warning"></span> ' . $html . ' ' . $notice,
  213. 'type' => 'error',
  214. 'dismiss' => ! in_array( $screen, [ 'dashboard', 'settings' ], true ),
  215. ];
  216. }
  217. /**
  218. * EP has no never had a sync. We assume the user is on step 3 of the install.
  219. *
  220. * Type: notice
  221. * Dismiss: Everywhere except ep dash and settings
  222. * Show: All screens except install. Dont show during sync ever
  223. *
  224. * @since 3.0
  225. * @return array|bool
  226. */
  227. protected function process_no_sync_notice() {
  228. $last_sync = Utils\get_option( 'ep_last_sync', false );
  229. if ( ! empty( $last_sync ) ) {
  230. return false;
  231. }
  232. $host = Utils\get_host();
  233. if ( empty( $host ) ) {
  234. return false;
  235. }
  236. $dismiss = Utils\get_option( 'ep_hide_no_sync_notice', false );
  237. $screen = Screen::factory()->get_current_screen();
  238. if ( 'install' === $screen ) {
  239. return false;
  240. }
  241. if ( $dismiss && ! in_array( $screen, [ 'dashboard', 'settings' ], true ) ) {
  242. return false;
  243. }
  244. if ( Utils\isset_do_sync_parameter() ) {
  245. return false;
  246. }
  247. if ( defined( 'EP_IS_NETWORK' ) && EP_IS_NETWORK ) {
  248. $url = admin_url( 'network/admin.php?page=elasticpress-sync' );
  249. } else {
  250. $url = admin_url( 'admin.php?page=elasticpress-sync' );
  251. }
  252. if ( defined( 'EP_DASHBOARD_SYNC' ) && ! EP_DASHBOARD_SYNC ) {
  253. $html = esc_html__( 'Dashboard sync is disabled, but ElasticPress is almost ready to go. Trigger a sync from WP-CLI.', 'elasticpress' );
  254. } else {
  255. $html = sprintf(
  256. /* translators: Sync Page URL */
  257. __( 'ElasticPress is almost ready to go. You just need to <a href="%s">sync your content</a>.', 'elasticpress' ),
  258. esc_url( $url )
  259. );
  260. }
  261. return [
  262. 'html' => $html,
  263. 'type' => 'info',
  264. 'dismiss' => ! in_array( $screen, [ 'dashboard', 'settings' ], true ),
  265. ];
  266. }
  267. /**
  268. * EP has no host set. We assume the user needs to run an install.
  269. *
  270. * Type: notice
  271. * Dismiss: Anywhere except EP dashboard
  272. * Show: All screens except settings and install
  273. *
  274. * @since 3.0
  275. * @return array|bool
  276. */
  277. protected function process_need_setup_notice() {
  278. $host = Utils\get_host();
  279. if ( ! empty( $host ) ) {
  280. return false;
  281. }
  282. $dismiss = Utils\get_option( 'ep_hide_need_setup_notice', false );
  283. $screen = Screen::factory()->get_current_screen();
  284. if ( in_array( $screen, [ 'settings', 'install' ], true ) ) {
  285. return false;
  286. }
  287. if ( $dismiss && 'dashboard' !== $screen ) {
  288. return false;
  289. }
  290. if ( defined( 'EP_IS_NETWORK' ) && EP_IS_NETWORK ) {
  291. $url = admin_url( 'network/admin.php?page=elasticpress-settings' );
  292. } else {
  293. $url = admin_url( 'admin.php?page=elasticpress-settings' );
  294. }
  295. return [
  296. 'type' => 'info',
  297. 'dismiss' => 'dashboard' !== $screen,
  298. 'html' => sprintf(
  299. /* translators: Sync Page URL */
  300. __( 'ElasticPress is almost ready to go. You just need to <a href="%s">enter your settings</a>.', 'elasticpress' ),
  301. esc_url( $url )
  302. ),
  303. ];
  304. }
  305. /**
  306. * Below ES compat error.
  307. *
  308. * Type: error
  309. * Dismiss: Anywhere
  310. * Show: All screens
  311. *
  312. * @since 3.0
  313. * @return array|bool
  314. */
  315. protected function process_es_below_compat_notice() {
  316. if ( Utils\is_epio() ) {
  317. return false;
  318. }
  319. $host = Utils\get_host();
  320. if ( empty( $host ) ) {
  321. return false;
  322. }
  323. $es_version = Elasticsearch::factory()->get_elasticsearch_version();
  324. if ( false === $es_version ) {
  325. return false;
  326. }
  327. $dismiss = Utils\get_option( 'ep_hide_es_below_compat_notice', false );
  328. if ( $dismiss ) {
  329. return false;
  330. }
  331. // First reduce version to major version i.e. 7.10 not 7.10.1.
  332. $major_es_version = preg_replace( '#^([0-9]+\.[0-9]+).*#', '$1', $es_version );
  333. // pad a version to have at least two parts (7 -> 7.0)
  334. $parts = explode( '.', $major_es_version );
  335. if ( 1 === count( $parts ) ) {
  336. $parts[] = 0;
  337. }
  338. $major_es_version = implode( '.', $parts );
  339. if ( 1 === version_compare( EP_ES_VERSION_MIN, $major_es_version ) ) {
  340. return [
  341. 'type' => 'error',
  342. 'dismiss' => true,
  343. 'html' => sprintf(
  344. /* translators: 1. Current Elasticsearch version; 2. Minimum required ES version */
  345. __( 'Your Elasticsearch version %1$s is below the minimum required Elasticsearch version %2$s. ElasticPress may or may not work properly.', 'elasticpress' ),
  346. esc_html( $es_version ),
  347. esc_html( EP_ES_VERSION_MIN )
  348. ),
  349. ];
  350. }
  351. return false;
  352. }
  353. /**
  354. * Above ES compat error.
  355. *
  356. * Type: error
  357. * Dismiss: Anywhere
  358. * Show: All screens
  359. *
  360. * @since 3.0
  361. * @return array|bool
  362. */
  363. protected function process_es_above_compat_notice() {
  364. if ( Utils\is_epio() ) {
  365. return false;
  366. }
  367. $host = Utils\get_host();
  368. if ( empty( $host ) ) {
  369. return false;
  370. }
  371. $es_version = Elasticsearch::factory()->get_elasticsearch_version();
  372. if ( false === $es_version ) {
  373. return false;
  374. }
  375. $dismiss = Utils\get_option( 'ep_hide_es_above_compat_notice', false );
  376. if ( $dismiss ) {
  377. return false;
  378. }
  379. // First reduce version to major version i.e. 7.10 not 7.10.1.
  380. $major_es_version = preg_replace( '#^([0-9]+\.[0-9]+).*#', '$1', $es_version );
  381. if ( -1 === version_compare( EP_ES_VERSION_MAX, $major_es_version ) ) {
  382. return [
  383. 'type' => 'warning',
  384. 'dismiss' => true,
  385. 'html' => sprintf(
  386. /* translators: 1. Current Elasticsearch version; 2. Maximum supported ES version */
  387. __( 'Your Elasticsearch version %1$s is above the maximum required Elasticsearch version %2$s. ElasticPress may or may not work properly.', 'elasticpress' ),
  388. esc_html( $es_version ),
  389. esc_html( EP_ES_VERSION_MAX )
  390. ),
  391. ];
  392. }
  393. }
  394. /**
  395. * Server software different from Elasticsearch warning.
  396. *
  397. * Type: warning
  398. * Dismiss: Anywhere
  399. * Show: All screens
  400. *
  401. * @since 4.2.1
  402. * @return array|bool
  403. */
  404. protected function process_different_server_type_notice() {
  405. if ( Utils\is_epio() ) {
  406. return false;
  407. }
  408. $host = Utils\get_host();
  409. if ( empty( $host ) ) {
  410. return false;
  411. }
  412. $server_type = Elasticsearch::factory()->get_server_type();
  413. if ( false === $server_type || 'elasticsearch' === $server_type ) {
  414. return false;
  415. }
  416. $dismiss = Utils\get_option( 'ep_hide_different_server_type_notice', false );
  417. if ( $dismiss ) {
  418. return false;
  419. }
  420. $doc_url = 'https://10up.github.io/ElasticPress/tutorial-compatibility.html';
  421. $html = sprintf(
  422. /* translators: Document page URL */
  423. __( 'Your server software is not supported. To learn more about server compatibility please <a href="%s">visit our documentation</a>.', 'elasticpress' ),
  424. esc_url( $doc_url )
  425. );
  426. return [
  427. 'html' => $html,
  428. 'type' => 'warning',
  429. 'dismiss' => true,
  430. ];
  431. }
  432. /**
  433. * Host error notification. Shows when EP can't reach ES host.
  434. *
  435. * Type: error
  436. * Dismiss: Only on non-EP screens
  437. * Show: All screens except install
  438. *
  439. * @since 3.0
  440. * @return array|bool
  441. */
  442. protected function process_host_error_notice() {
  443. $host = Utils\get_host();
  444. if ( empty( $host ) ) {
  445. return false;
  446. }
  447. $screen = Screen::factory()->get_current_screen();
  448. if ( 'install' === $screen ) {
  449. return false;
  450. }
  451. $es_version = Elasticsearch::factory()->get_elasticsearch_version( false );
  452. if ( false !== $es_version ) {
  453. return false;
  454. }
  455. // Only dismissable on non-EP screens
  456. if ( ! in_array( $screen, [ 'settings', 'dashboard' ], true ) ) {
  457. $dismiss = Utils\get_option( 'ep_hide_host_error_notice', false );
  458. if ( $dismiss ) {
  459. return false;
  460. }
  461. }
  462. if ( defined( 'EP_IS_NETWORK' ) && EP_IS_NETWORK ) {
  463. $url = admin_url( 'network/admin.php?page=elasticpress-settings' );
  464. $response_code = get_site_transient( 'ep_es_info_response_code' );
  465. $response_error = get_site_transient( 'ep_es_info_response_error' );
  466. } else {
  467. $url = admin_url( 'admin.php?page=elasticpress-settings' );
  468. $response_code = get_transient( 'ep_es_info_response_code' );
  469. $response_error = get_transient( 'ep_es_info_response_error' );
  470. }
  471. $retry_url = add_query_arg(
  472. [
  473. 'ep-retry' => 1,
  474. 'ep_retry_nonce' => wp_create_nonce( 'ep_retry_nonce' ),
  475. ]
  476. );
  477. $html = sprintf(
  478. /* translators: 1. Current URL with retry parameter; 2. Settings Page URL */
  479. __( 'There is a problem with connecting to your Elasticsearch host. ElasticPress can <a href="%1$s">try your host again</a>, or you may need to <a href="%2$s">change your settings</a>.', 'elasticpress' ),
  480. esc_url( $retry_url ),
  481. esc_url( $url )
  482. );
  483. if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
  484. if ( ! empty( $response_code ) ) {
  485. /* translators: Response Code Number */
  486. $html .= '<span class="notice-error-es-response-code"> ' . sprintf( __( 'Response Code: %s', 'elasticpress' ), esc_html( $response_code ) ) . '</span>';
  487. }
  488. if ( ! empty( $response_error ) ) {
  489. /* translators: Response Code Message */
  490. $html .= '<span class="notice-error-es-response-error"> ' . sprintf( __( 'Response error: %s', 'elasticpress' ), esc_html( $response_error ) ) . '</span>';
  491. }
  492. }
  493. return [
  494. 'html' => $html,
  495. 'type' => 'error',
  496. 'dismiss' => ( ! in_array( Screen::factory()->get_current_screen(), [ 'settings', 'dashboard' ], true ) ) ? true : false,
  497. ];
  498. }
  499. /**
  500. * Determine if the wrong mapping might be installed
  501. *
  502. * Type: error
  503. * Dismiss: Always dismissable per es_version as custom mapping could exist
  504. * Show: All screens
  505. *
  506. * @since 3.6.2
  507. * @return array|bool
  508. */
  509. protected function process_maybe_wrong_mapping_notice() {
  510. $screen = Screen::factory()->get_current_screen();
  511. if ( 'install' === $screen ) {
  512. return false;
  513. }
  514. // we might have this dismissed
  515. $dismiss = Utils\get_option( 'ep_hide_maybe_wrong_mapping_notice', false );
  516. // we need a host
  517. $host = Utils\get_host();
  518. if ( empty( $host ) ) {
  519. return false;
  520. }
  521. // we also need a version
  522. $es_version = Elasticsearch::factory()->get_elasticsearch_version( false );
  523. if ( false === $es_version || $dismiss === $es_version ) {
  524. return false;
  525. }
  526. // we also likely need a sync to have a mapping
  527. $last_sync = Utils\get_option( 'ep_last_sync', false );
  528. if ( empty( $last_sync ) ) {
  529. return false;
  530. }
  531. $post_indexable = Indexables::factory()->get( 'post' );
  532. $mapping_file_wanted = $post_indexable->get_mapping_name();
  533. $mapping_file_current = $post_indexable->determine_mapping_version();
  534. if ( is_wp_error( $mapping_file_current ) ) {
  535. return false;
  536. }
  537. if ( ! $mapping_file_current || $mapping_file_wanted !== $mapping_file_current ) {
  538. $html = sprintf(
  539. /* translators: 1. <em>; 2. </em> */
  540. esc_html__( 'It seems the mapping data in your index does not match the Elasticsearch version used. We recommend to reindex your content using the sync button on the top of the screen or through wp-cli by adding the %1$s--setup%2$s flag', 'elasticpress' ),
  541. '<em>',
  542. '</em>'
  543. );
  544. if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
  545. /* translators: 1. Current mapping file; 2. Mapping file that should be used */
  546. $html .= '<span class="notice-error-es-response-code"> ' . sprintf( esc_html__( 'Current mapping: %1$s. Expected mapping: %2$s', 'elasticpress' ), esc_html( $mapping_file_current ), esc_html( $mapping_file_wanted ) ) . '</span>';
  547. }
  548. return [
  549. 'html' => $html,
  550. 'type' => 'error',
  551. 'dismiss' => true,
  552. ];
  553. }
  554. }
  555. /**
  556. * Single node notification. Shows when index health is yellow.
  557. *
  558. * Type: warning
  559. * Dismiss: Anywhere
  560. * Show: All screens except install
  561. *
  562. * @since 3.2
  563. * @return array|bool
  564. */
  565. protected function process_yellow_health_notice() {
  566. $host = Utils\get_host();
  567. if ( empty( $host ) ) {
  568. return false;
  569. }
  570. $last_sync = Utils\get_option( 'ep_last_sync', false );
  571. if ( empty( $last_sync ) ) {
  572. return false;
  573. }
  574. $dismiss = Utils\get_option( 'ep_hide_yellow_health_notice', false );
  575. $screen = Screen::factory()->get_current_screen();
  576. if ( ! in_array( $screen, [ 'dashboard', 'settings' ], true ) || $dismiss ) {
  577. return false;
  578. }
  579. $nodes = Stats::factory()->get_nodes();
  580. if ( false !== $nodes && $nodes < 2 && $nodes > 0 ) {
  581. if ( defined( 'EP_IS_NETWORK' ) && EP_IS_NETWORK ) {
  582. $url = network_admin_url( 'admin.php?page=elasticpress-health' );
  583. } else {
  584. $url = admin_url( 'admin.php?page=elasticpress-health' );
  585. }
  586. return [
  587. 'type' => 'warning',
  588. 'dismiss' => true,
  589. 'html' => sprintf(
  590. /* translators: Index Health URL */
  591. __( 'It looks like one or more of your indices are running on a single node. While this won\'t prevent you from using ElasticPress, depending on your site\'s specific needs this can represent a performance issue. Please check the <a href="%s">Index Health</a> page where you can check the health of all of your indices.', 'elasticpress' ),
  592. $url
  593. ),
  594. ];
  595. }
  596. }
  597. /**
  598. * Too many fields notification. Shows when the site has potentially more fields than ES could handle.
  599. *
  600. * Type: warning|error
  601. * Dismiss: Anywhere
  602. * Show: Sync and Install page
  603. *
  604. * @since 4.4.0
  605. * @return array|bool
  606. */
  607. protected function process_too_many_fields_notice() {
  608. $host = Utils\get_host();
  609. if ( empty( $host ) ) {
  610. return false;
  611. }
  612. $dismiss = Utils\get_option( 'ep_hide_too_many_fields_notice', false );
  613. $screen = Screen::factory()->get_current_screen();
  614. if ( ! in_array( $screen, [ 'install', 'sync' ], true ) || $dismiss ) {
  615. return false;
  616. }
  617. $has_error = false;
  618. $has_warning = false;
  619. if ( defined( 'EP_IS_NETWORK' ) && EP_IS_NETWORK ) {
  620. $sites = Utils\get_sites( 0, true );
  621. foreach ( $sites as $site ) {
  622. switch_to_blog( $site['blog_id'] );
  623. list( $has_error, $site_has_warning ) = $this->check_field_count();
  624. restore_current_blog();
  625. $has_warning = $has_warning || $site_has_warning;
  626. if ( $has_error ) {
  627. break;
  628. }
  629. }
  630. } else {
  631. list( $has_error, $has_warning ) = $this->check_field_count();
  632. }
  633. if ( $has_error ) {
  634. $message = sprintf(
  635. /* translators: Elasticsearch or ElasticPress.io; 2. Link to article; 3. Link to article */
  636. __( 'Your website content has more public custom fields than %1$s is able to store. Check our articles about <a href="%2$s">Elasticsearch field limitations</a> and <a href="%3$s">how to index just the custom fields you need</a> before trying to sync.', 'elasticpress' ),
  637. Utils\is_epio() ? __( 'ElasticPress.io', 'elasticpress' ) : __( 'Elasticsearch', 'elasticpress' ),
  638. 'https://www.elasticpress.io/documentation/article/i-get-the-error-limit-of-total-fields-in-index-has-been-exceeded/',
  639. 'https://www.elasticpress.io/documentation/article/how-to-exclude-metadata-from-indexing/'
  640. );
  641. return [
  642. 'type' => 'error',
  643. 'dismiss' => true,
  644. 'html' => $message,
  645. ];
  646. }
  647. if ( $has_warning ) {
  648. $message = sprintf(
  649. /* translators: Elasticsearch or ElasticPress.io; 2. Link to article; 3. Link to article */
  650. __( 'Your website content seems to have more public custom fields than %1$s is able to store. Check our articles about <a href="%2$s">Elasticsearch field limitations</a> and <a href="%3$s">how to index just the custom fields you need</a> if you receive any errors while syncing.', 'elasticpress' ),
  651. Utils\is_epio() ? __( 'ElasticPress.io', 'elasticpress' ) : __( 'Elasticsearch', 'elasticpress' ),
  652. 'https://www.elasticpress.io/documentation/article/i-get-the-error-limit-of-total-fields-in-index-has-been-exceeded/',
  653. 'https://www.elasticpress.io/documentation/article/how-to-exclude-metadata-from-indexing/'
  654. );
  655. return [
  656. 'type' => 'warning',
  657. 'dismiss' => true,
  658. 'html' => $message,
  659. ];
  660. }
  661. return false;
  662. }
  663. /**
  664. * Get notices that should be displayed
  665. *
  666. * @since 3.0
  667. * @return array
  668. */
  669. public function get_notices() {
  670. /**
  671. * Filter admin notices
  672. *
  673. * @hook ep_admin_notices
  674. * @param {array} $notices Admin notices
  675. * @return {array} New notices
  676. */
  677. return apply_filters( 'ep_admin_notices', $this->notices );
  678. }
  679. /**
  680. * Dismiss a notice given a notice key.
  681. *
  682. * @param string $notice Notice key
  683. * @since 3.0
  684. */
  685. public function dismiss_notice( $notice ) {
  686. $value = true;
  687. // allow version dependent dismissal
  688. if ( in_array( $notice, [ 'maybe_wrong_mapping' ], true ) ) {
  689. $value = Elasticsearch::factory()->get_elasticsearch_version( false );
  690. }
  691. Utils\update_option( 'ep_hide_' . $notice . '_notice', $value );
  692. }
  693. /**
  694. * Return singleton instance of class
  695. *
  696. * @return object
  697. * @since 0.1.0
  698. */
  699. public static function factory() {
  700. static $instance = false;
  701. if ( ! $instance ) {
  702. $instance = new self();
  703. }
  704. return $instance;
  705. }
  706. /**
  707. * Compare the number of fields in the site and the number of allowed fields in ES
  708. *
  709. * @since 4.4.0
  710. * @return array
  711. */
  712. protected function check_field_count() {
  713. $post_indexable = Indexables::factory()->get( 'post' );
  714. $search = Features::factory()->get_registered_feature( 'search' );
  715. if ( $search && ! empty( $search->weighting ) && 'manual' === $search->weighting->get_meta_mode() ) {
  716. $indexable_fields = $post_indexable->get_all_allowed_metas_manual();
  717. $count_fields_db = count( $indexable_fields );
  718. } else {
  719. $indexable_fields = $post_indexable->get_predicted_indexable_meta_keys();
  720. $count_fields_db = count( $indexable_fields );
  721. }
  722. $index_name = $post_indexable->get_index_name();
  723. $es_field_limit = Elasticsearch::factory()->get_index_total_fields_limit( $index_name );
  724. $es_field_limit = $es_field_limit ?? apply_filters( 'ep_total_field_limit', 5000 );
  725. $predicted_es_field_count = $count_fields_db * 8;
  726. return [
  727. $predicted_es_field_count > $es_field_limit,
  728. $predicted_es_field_count * 1.2 > $es_field_limit,
  729. ];
  730. }
  731. }