diff --git a/lib/compat/wordpress-6.5/fonts/fonts.php b/lib/compat/wordpress-6.5/fonts/fonts.php index 0f7ef7809abbeb..302562298fffe1 100644 --- a/lib/compat/wordpress-6.5/fonts/fonts.php +++ b/lib/compat/wordpress-6.5/fonts/fonts.php @@ -282,17 +282,12 @@ function _wp_filter_font_directory( $font_dir ) { return $font_dir; } - $site_path = ''; - if ( is_multisite() && ! ( is_main_network() && is_main_site() ) ) { - $site_path = '/sites/' . get_current_blog_id(); - } - $font_dir = array( - 'path' => path_join( WP_CONTENT_DIR, 'fonts' ) . $site_path, - 'url' => untrailingslashit( content_url( 'fonts' ) ) . $site_path, + 'path' => untrailingslashit( $font_dir['basedir'] ) . '/fonts', + 'url' => untrailingslashit( $font_dir['baseurl'] ) . '/fonts', 'subdir' => '', - 'basedir' => path_join( WP_CONTENT_DIR, 'fonts' ) . $site_path, - 'baseurl' => untrailingslashit( content_url( 'fonts' ) ) . $site_path, + 'basedir' => untrailingslashit( $font_dir['basedir'] ) . '/fonts', + 'baseurl' => untrailingslashit( $font_dir['baseurl'] ) . '/fonts', 'error' => false, ); @@ -374,6 +369,41 @@ function _wp_before_delete_font_face( $post_id, $post ) { add_action( 'before_delete_post', '_wp_before_delete_font_face', 10, 2 ); } +// @core-merge: Do not merge this function, it is for deleting fonts from the wp-content/fonts directory only used in Gutenberg. +/** + * Deletes associated font files from wp-content/fonts, when a font face is deleted. + * + * @param int $post_id Post ID. + * @param WP_Post $post Post object. + */ +function gutenberg_before_delete_font_face( $post_id, $post ) { + if ( 'wp_font_face' !== $post->post_type ) { + return; + } + + $font_files = get_post_meta( $post_id, '_wp_font_face_file', false ); + + if ( empty( $font_files ) ) { + return; + } + + $site_path = ''; + if ( is_multisite() && ! ( is_main_network() && is_main_site() ) ) { + $site_path = '/sites/' . get_current_blog_id(); + } + + $font_dir = path_join( WP_CONTENT_DIR, 'fonts' ) . $site_path; + + foreach ( $font_files as $font_file ) { + $font_path = $font_dir . '/' . $font_file; + + if ( file_exists( $font_path ) ) { + wp_delete_file( $font_path ); + } + } +} +add_action( 'before_delete_post', 'gutenberg_before_delete_font_face', 10, 2 ); + // @core-merge: Do not merge this back compat function, it is for supporting a legacy font family format only in Gutenberg. /** * Convert legacy font family posts to the new format. diff --git a/phpunit/tests/fonts/font-library/fontDeleteFilesFromWpContentFolder.php b/phpunit/tests/fonts/font-library/fontDeleteFilesFromWpContentFolder.php new file mode 100644 index 00000000000000..dfba431306a0d0 --- /dev/null +++ b/phpunit/tests/fonts/font-library/fontDeleteFilesFromWpContentFolder.php @@ -0,0 +1,194 @@ +user->create( + array( + 'role' => 'administrator', + ) + ); + grant_super_admin( self::$super_admin_id ); + } + + /** + * Tears down test class. + */ + public static function wpTearDownAfterClass() { + self::delete_user( self::$super_admin_id ); + } + + /** + * Tests that font files uploaded to wp-content/fonts are deleted when the associated font face post is deleted. + * + * @covers ::gutenberg_before_delete_font_face + */ + public function test_uploaded_font_files_are_deleted_from_wp_content_fonts_folder() { + wp_set_current_user( self::$super_admin_id ); + $font_family_id = $this->create_font_family(); + + // Upload the font file to wp-content/fonts, the previous directory used for fonts. + add_filter( 'font_dir', array( $this, 'filter_font_dir' ) ); + $response = $this->create_font_face( $font_family_id ); + remove_filter( 'font_dir', array( $this, 'filter_font_dir' ) ); + + $this->assertSame( 201, $response->get_status(), 'The font face should be created successfully.' ); + + $data = $response->get_data(); + $font_path = str_replace( content_url(), WP_CONTENT_DIR, $data['font_face_settings']['src'] ); + + // Ensure the file was uploaded to the correct location. + $this->assertFalse( str_contains( $font_path, 'uploads' ), 'The font file should not be in the uploads folder.' ); + $this->assertTrue( file_exists( $font_path ), 'The font file should exist in the wp-content/fonts folder.' ); + + // Ensure the file is deleted when the font face post is deleted, even when the directory is not filtered. + wp_delete_post( $font_family_id, true ); + $this->assertTrue( str_contains( wp_get_font_dir()['path'], 'uploads' ), 'The font directory should be in uploads.' ); + $this->assertFalse( file_exists( $font_path ), 'The font file should be deleted from the wp-content/fonts folder.' ); + } + + /** + * Tests that font files uploaded to wp-content/fonts are deleted when the associated font face post is deleted on multisite. + * + * @group multisite + * @group ms-required + * + * @covers ::gutenberg_before_delete_font_face + */ + public function test_uploaded_font_files_are_deleted_from_wp_content_fonts_folder_multisite() { + wp_set_current_user( self::$super_admin_id ); + $blog_id = self::factory()->blog->create(); + + switch_to_blog( $blog_id ); + $font_family_id = $this->create_font_family(); + + // Upload the font file to wp-content/fonts, the previous directory used for fonts. + add_filter( 'font_dir', array( $this, 'filter_font_dir' ) ); + $response = $this->create_font_face( $font_family_id ); + remove_filter( 'font_dir', array( $this, 'filter_font_dir' ) ); + restore_current_blog(); + + $this->assertSame( 201, $response->get_status(), 'The font face should be created successfully.' ); + + $data = $response->get_data(); + $font_path = str_replace( content_url(), WP_CONTENT_DIR, $data['font_face_settings']['src'] ); + + // Ensure the file was uploaded to the correct location. + $this->assertFalse( str_contains( $font_path, 'uploads' ), 'The font file should not be in the uploads folder.' ); + $this->assertTrue( file_exists( $font_path ), 'The font file should exist in the wp-content/fonts folder.' ); + + // Ensure the file is deleted when the font face post is deleted, even when the directory is not filtered. + switch_to_blog( $blog_id ); + wp_delete_post( $font_family_id, true ); + restore_current_blog(); + + $this->assertTrue( str_contains( wp_get_font_dir()['path'], 'uploads' ), 'The font directory should be in uploads.' ); + $this->assertFalse( file_exists( $font_path ), 'The font file should be deleted from the wp-content/fonts folder.' ); + } + + /** + * Sets the font upload directory to wp-content/fonts, the default previously used in Gutenberg. + * + * @param array $font_dir Font directory settings. + * @return array Filtered font directory settings. + */ + public function filter_font_dir( $font_dir ) { + $site_path = ''; + if ( is_multisite() && ! ( is_main_network() && is_main_site() ) ) { + $site_path = '/sites/' . get_current_blog_id(); + } + + $font_dir['path'] = path_join( WP_CONTENT_DIR, 'fonts' ) . $site_path; + $font_dir['url'] = untrailingslashit( content_url( 'fonts' ) ) . $site_path; + $font_dir['basedir'] = path_join( WP_CONTENT_DIR, 'fonts' ) . $site_path; + $font_dir['baseurl'] = untrailingslashit( content_url( 'fonts' ) ) . $site_path; + + return $font_dir; + } + + /** + * Creates a font family post for testing. + * + * @return int Font family post ID. + */ + protected function create_font_family() { + return self::factory()->post->create( + wp_slash( + array( + 'post_type' => 'wp_font_family', + 'post_status' => 'publish', + 'post_title' => 'Open Sans', + 'post_name' => 'open-sans', + 'post_content' => wp_json_encode( + array( + 'fontFamily' => '"Open Sans"', + ) + ), + ) + ) + ); + } + + /** + * Creates a font face post under the specified font family for testing. + * + * @param int $font_family_id Font family post ID. + * @return WP_REST_Response REST response object for font face creation. + */ + protected function create_font_face( $font_family_id ) { + // Create a new font face with a font file upload. + $font_file = GUTENBERG_DIR_TESTDATA . '/fonts/OpenSans-Regular.woff2'; + $font_path = wp_tempnam( 'OpenSans-Regular.woff2' ); + copy( $font_file, $font_path ); + + $files = array( + 'file-0' => array( + 'name' => 'OpenSans-Regular.woff2', + 'full_path' => 'OpenSans-Regular.woff2', + 'type' => 'font/woff2', + 'tmp_name' => $font_path, + 'error' => 0, + 'size' => filesize( $font_path ), + ), + ); + + $request = new WP_REST_Request( 'POST', '/wp/v2/font-families/' . $font_family_id . '/font-faces' ); + $request->set_param( + 'font_face_settings', + wp_json_encode( + array( + 'fontFamily' => '"Open Sans"', + 'fontWeight' => '200', + 'fontStyle' => 'normal', + 'src' => 'file-0', + ) + ) + ); + $request->set_file_params( $files ); + return rest_get_server()->dispatch( $request ); + } +}