Skip to content

Commit

Permalink
Moves font directory into uploads to match WP 6.5 (WordPress#60354)
Browse files Browse the repository at this point in the history
* Updates default font directory location to wp-content/uploads/fonts when using Gutenberg with WP 6.5 and below
* Deletes font files from wp-content/fonts when font face posts are deleted

Unlinked contributors: joehoyle, Zealth57.

Co-authored-by: creativecoder <[email protected]>
Co-authored-by: peterwilsoncc <[email protected]>
Co-authored-by: azaozz <[email protected]>
Co-authored-by: priethor <[email protected]>
Co-authored-by: hellofromtonya <[email protected]>
Co-authored-by: costdev <[email protected]>
Co-authored-by: swissspidy <[email protected]>
Co-authored-by: andreilupu <[email protected]>
Co-authored-by: youknowriad <[email protected]>
Co-authored-by: aaronjorbin <[email protected]>
Co-authored-by: cbirdsong <[email protected]>
Co-authored-by: mcsf <[email protected]>
Co-authored-by: jazzsequence <[email protected]>
Co-authored-by: joemcgill <[email protected]>
Co-authored-by: pwtyler <[email protected]>
Co-authored-by: getdave <[email protected]>
Co-authored-by: mtias <[email protected]>
Co-authored-by: justlevine <[email protected]>
Co-authored-by: samuelsidler <[email protected]>
Co-authored-by: johnbillion <[email protected]>
  • Loading branch information
21 people authored Jun 30, 2024
1 parent 9bfeea1 commit b76965d
Show file tree
Hide file tree
Showing 2 changed files with 233 additions and 9 deletions.
48 changes: 39 additions & 9 deletions lib/compat/wordpress-6.5/fonts/fonts.php
Original file line number Diff line number Diff line change
Expand Up @@ -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,
);

Expand Down Expand Up @@ -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.
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
<?php
// @core-merge: Do not include these tests, they are for Gutenberg only.

/**
* Tests deleting font files from the previous fonts folder default (wp-content/fonts).
*
* This covers cases where fonts were installed with older versions of Gutenberg, but are then deleted with
* newer versions of Gutenberg and/or WordPress.
*
* @package WordPress
* @subpackage Font Library
*
* @group fonts
* @group font-library
*/
class Tests_Font_Delete_Files_From_Wp_Content_Folder extends WP_UnitTestCase {
/**
* ID of a super admin user.
*
* @var int
*/
protected static $super_admin_id;

/**
* Sets up test class.
*
* @param WP_UnitTest_Factory $factory Unit test factory instance.
*/
public static function wpSetUpBeforeClass( WP_UnitTest_Factory $factory ) {
self::$super_admin_id = $factory->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 );
}
}

0 comments on commit b76965d

Please sign in to comment.