k % 1 l % l1 @% 1 `% 1 % 1 % 1 p% ň1 % ҈1 % ߈1 0% 1 % 1 % 1 % 1 % 1 % 01 % 91 % B1 % S1 % e1 @% n1 % x1 % 1 % 1 % 1 0% 1 @% 1 P% ȉ1 P% Ӊ1 P% މ1 % 1 p% 1 % 1 `% 1 % &1 % - 1 % ;1 % J1 #`1 &`1 U1 `[1 U1 R1 S1 0Y1 R1 S1 0O1 PO1 W1 S1 pB1 S1 R1 pR1 B1 D1 `W1 `B1 W1 PB1 V1 1 a1 1 a1 1 a1 1 0 b1 1 ,b1 1 8b1 1 `1 1 kb1 1 1 1 b1 b1 b1 b1 b1 b1 c1 c1 c1 c1 c1 #c1 )c1 /c1 5c1 ;c1 s ,1 1 "1 d B1 61 D1 ]1 1 u y1 _1 v 1 w 1 Ʈ1 ֮1 x 1 y 1 e 1 p 1 n 41 m M1 e1 s1 1 f 1 g 1 h 1 z 1 t U_1 ¯1 د1 1 { 1 1 P1 d1 /1 C1 V1 | Ԅ1 } 1 j1 1 ~ 1 o 1 1 1 1 1 1 1 ڰ1 j 1 1 k 1 1 1 l 1 q %1 91 I1 Z1 r h1 v1 1 1 1 1 ȱ1 ۱1 1 1 1 ` &1 71 O1 `1 p1 ~1 1 1 1 1 @ ˲1 P 1 ` 1 p 1 1 $1 0 51 @ H1 P ^1 ` s1 1 0 1 0 1 ٳ1 @ 1 1 %1 01 @1 O1 c1 1 y1 (1 0 P1 1 1 Ǵ1 1 1 p1 0 1 @ н1 P 1 ` ,1 E1 0 [1 q1 @ 1 1 1 @ 1 1 ` P1 p ۵1 p1 1 1 1 1 '1 =1 1 Y1 P 1 ` q1 ` 1 p 1 1 e_deleted', $indexable ); } /** * Updates the relations when the post indexable is built. * * @param Indexable $indexable The indexable. * @param WP_Post $post The post. */ public function updated_indexable( $indexable, $post ) { // Only interested in post indexables. if ( $indexable->object_type !== 'post' ) { return; } if ( \is_a( $post, Indexable::class ) ) { \_deprecated_argument( __FUNCTION__, '17.7', 'The $old_indexable argument has been deprecated.' ); $post = $this->post->get_post( $indexable->object_id ); } $this->update_relations( $post ); $indexable->save(); } /** * Saves post meta. * * @param int $post_id Post ID. * * @return void */ public function build_indexable( $post_id ) { // Bail if this is a multisite installation and the site has been switched. if ( $this->is_multisite_and_switched() ) { return; } try { $indexable = $this->repository->find_by_id_and_type( $post_id, 'post', false ); $indexable = $this->builder->build_for_id_and_type( $post_id, 'post', $indexable ); $post = $this->post->get_post( $post_id ); /* * Update whether an author has public posts. * For example this post could be set to Draft or Private, * which can influence if its author has any public posts at all. */ if ( $indexable ) { $this->update_has_public_posts( $indexable ); } // Build links for this post. if ( $post && $indexable && \in_array( $post->post_status, $this->post->get_public_post_statuses(), true ) ) { $this->link_builder->build( $indexable, $post->post_content ); // Save indexable to persist the updated link count. $indexable->save(); $this->updated_indexable( $indexable, $post ); } } catch ( Exception $exception ) { $this->logger->log( LogLevel::ERROR, $exception->getMessage() ); } } /** * Updates the has_public_posts when the post indexable is built. * * @param Indexable $indexable The indexable to check. */ protected function update_has_public_posts( $indexable ) { // Update the author indexable's has public posts value. try { $author_indexable = $this->repository->find_by_id_and_type( $indexable->author_id, 'user' ); if ( $author_indexable ) { $author_indexable->has_public_posts = $this->author_archive->author_has_public_posts( $author_indexable->object_id ); $author_indexable->save(); $this->reschedule_cleanup_if_author_has_no_posts( $author_indexable ); } } catch ( Exception $exception ) { $this->logger->log( LogLevel::ERROR, $exception->getMessage() ); } // Update possible attachment's has public posts value. $this->post->update_has_public_posts_on_attachments( $indexable->object_id, $indexable->is_public ); } /** * Reschedule indexable cleanup if the author does not have any public posts. * This should remove the author from the indexable table, since we do not * want to store authors without public facing posts in the table. * * @param Indexable $author_indexable The author indexable. * * @return void */ protected function reschedule_cleanup_if_author_has_no_posts( $author_indexable ) { if ( $author_indexable->has_public_posts === false ) { $cleanup_not_yet_scheduled = ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ); if ( $cleanup_not_yet_scheduled ) { \wp_schedule_single_event( ( \time() + ( \MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK ); } } } /** * Updates the relations on post save or post status change. * * @param WP_Post $post The post that has been updated. */ protected function update_relations( $post ) { $related_indexables = $this->get_related_indexables( $post ); foreach ( $related_indexables as $indexable ) { // Ignore everything that is not an actual indexable. if ( \is_a( $indexable, Indexable::class ) ) { $indexable->object_last_modified = \max( $indexable->object_last_modified, $post->post_modified_gmt ); $indexable->save(); } } } /** * Retrieves the related indexables for given post. * * @param WP_Post $post The post to get the indexables for. * * @return Indexable[] The indexables. */ protected function get_related_indexables( $post ) { /** * The related indexables. * * @var Indexable[] $related_indexables . */ $related_indexables = []; $related_indexables[] = $this->repository->find_by_id_and_type( $post->post_author, 'user', false ); $related_indexables[] = $this->repository->find_for_post_type_archive( $post->post_type, false ); $related_indexables[] = $this->repository->find_for_home_page( false ); $taxonomies = \get_post_taxonomies( $post->ID ); $taxonomies = \array_filter( $taxonomies, 'is_taxonomy_viewable' ); $term_ids = []; foreach ( $taxonomies as $taxonomy ) { $terms = \get_the_terms( $post->ID, $taxonomy ); if ( empty( $terms ) || \is_wp_error( $terms ) ) { continue; } $term_ids = \array_merge( $term_ids, \wp_list_pluck( $terms, 'term_id' ) ); } $related_indexables = \array_merge( $related_indexables, $this->repository->find_by_multiple_ids_and_type( $term_ids, 'term', false ) ); return \array_filter( $related_indexables ); } /** * Tests if the site is multisite and switched. * * @return bool True when the site is multisite and switched */ protected function is_multisite_and_switched() { return \is_multisite() && \ms_is_switched(); } }