Adds a new 'S3-Compatible Storage' provider that works with any
S3-API-compatible object storage service, including MinIO, Ceph,
Cloudflare R2, Backblaze B2, and others.
Changes:
- New provider class: classes/providers/storage/s3-compatible-provider.php
- Provider key: s3compatible
- Reads user-configured endpoint URL from settings
- Uses path-style URL access (required by most S3-compatible services)
- Supports credentials via AS3CF_S3COMPAT_ACCESS_KEY_ID /
AS3CF_S3COMPAT_SECRET_ACCESS_KEY wp-config.php constants
- Disables AWS-specific features (Block Public Access, Object Ownership)
- New provider SVG icons (s3compatible.svg, -link.svg, -round.svg)
- Registered provider in main plugin class with endpoint setting support
- Updated StorageProviderSubPage to show endpoint URL input for S3-compatible
- Built pro settings bundle with rollup (Svelte 4.2.19)
- Added package.json and updated rollup.config.mjs for pro-only builds
172 lines
4.7 KiB
PHP
172 lines
4.7 KiB
PHP
<?php
|
|
/**
|
|
* Update Broken _wp_attachment_metadata introduced in 0.9.4
|
|
*
|
|
* @package amazon-s3-and-cloudfront
|
|
* @subpackage Classes/Upgrades/Meta-WP-Error
|
|
* @copyright Copyright (c) 2015, Delicious Brains
|
|
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
|
* @since 0.9.5
|
|
*/
|
|
|
|
namespace DeliciousBrains\WP_Offload_Media\Upgrades;
|
|
|
|
use AS3CF_Error;
|
|
use AS3CF_Utils;
|
|
use Exception;
|
|
|
|
/**
|
|
* Upgrade_Meta_WP_Error Class
|
|
*
|
|
* This class handles updating the _wp_attachment_metadata
|
|
* for attachments that have been removed from the local server
|
|
* and have had it corrupted by another plugin
|
|
*
|
|
* @since 0.9.5
|
|
*/
|
|
class Upgrade_Meta_WP_Error extends Upgrade {
|
|
|
|
/**
|
|
* @var int
|
|
*/
|
|
protected $upgrade_id = 3;
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $upgrade_name = 'meta_error';
|
|
|
|
/**
|
|
* @var string 'metadata', 'attachment'
|
|
*/
|
|
protected $upgrade_type = 'attachments';
|
|
|
|
/**
|
|
* Get running update text.
|
|
*
|
|
* @return string
|
|
*/
|
|
protected function get_running_update_text() {
|
|
return __( 'and rebuilding the metadata for attachments that may have been corrupted.', 'amazon-s3-and-cloudfront' );
|
|
}
|
|
|
|
/**
|
|
* Rebuild the attachment metadata for an attachment
|
|
*
|
|
* @param mixed $item
|
|
*
|
|
* @return bool
|
|
*/
|
|
protected function upgrade_item( $item ) {
|
|
$provider_object = AS3CF_Utils::maybe_fix_serialized_string( $item->provider_object );
|
|
$fixed = $item->provider_object !== $provider_object;
|
|
|
|
$provider_object = AS3CF_Utils::maybe_unserialize( $provider_object );
|
|
|
|
if ( false === $provider_object ) {
|
|
AS3CF_Error::log( 'Failed to unserialize offload meta for attachment ' . $item->ID . ': ' . $item->provider_object );
|
|
$this->error_count++;
|
|
|
|
return false;
|
|
}
|
|
|
|
if ( $fixed ) {
|
|
if ( update_post_meta( $item->ID, 'amazonS3_info', $provider_object ) ) {
|
|
$msg = sprintf( __( 'Fixed legacy amazonS3_info metadata when rebuilding corrupted attachment metadata, please check bucket and path for attachment ID %1$s', 'amazon-s3-and-cloudfront' ), $item->ID );
|
|
AS3CF_Error::log( $msg );
|
|
} else {
|
|
AS3CF_Error::log( 'Failed to fix broken serialized legacy offload metadata for attachment ' . $item->ID . ': ' . $item->provider_object );
|
|
$this->error_count++;
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
$file = get_attached_file( $item->ID, true );
|
|
|
|
if ( ! file_exists( $file ) ) {
|
|
// Copy back the file to the server if doesn't exist so we can successfully
|
|
// regenerate the attachment metadata
|
|
try {
|
|
$args = array(
|
|
'Bucket' => $provider_object['bucket'],
|
|
'Key' => $provider_object['key'],
|
|
'SaveAs' => $file,
|
|
);
|
|
$this->as3cf->get_provider_client( $provider_object['region'], true )->get_object( $args );
|
|
} catch ( Exception $e ) {
|
|
AS3CF_Error::log( sprintf( __( 'There was an error attempting to download the file %s from the bucket: %s', 'amazon-s3-and-cloudfront' ), $provider_object['key'], $e->getMessage() ) );
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Remove corrupted meta
|
|
delete_post_meta( $item->ID, '_wp_attachment_metadata' );
|
|
|
|
require_once ABSPATH . '/wp-admin/includes/image.php';
|
|
// Generate new attachment meta
|
|
wp_update_attachment_metadata( $item->ID, wp_generate_attachment_metadata( $item->ID, $file ) );
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Get a count of all attachments without region in their S3 metadata.
|
|
*
|
|
* @return int
|
|
*/
|
|
protected function count_items_to_process() {
|
|
return (int) $this->get_attachments_with_error_metadata( $this->blog_prefix, true );
|
|
}
|
|
|
|
/**
|
|
* Get all attachments that don't have region in their S3 meta data for a blog
|
|
*
|
|
* @param string $prefix
|
|
* @param int $limit
|
|
* @param bool|mixed $offset
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function get_items_to_process( $prefix, $limit, $offset = false ) {
|
|
$attachments = $this->get_attachments_with_error_metadata( $prefix, false, $limit );
|
|
|
|
return $attachments;
|
|
}
|
|
|
|
/**
|
|
* Get S3 attachments that have had their _wp_attachment_metadata corrupted
|
|
*
|
|
* @param string $prefix
|
|
* @param bool|false $count
|
|
* @param null|int $limit
|
|
*
|
|
* @return array|int
|
|
*/
|
|
protected function get_attachments_with_error_metadata( $prefix, $count = false, $limit = null ) {
|
|
global $wpdb;
|
|
|
|
$sql = "FROM `{$prefix}postmeta` pm1
|
|
LEFT OUTER JOIN `{$prefix}postmeta` pm2
|
|
ON pm1.`post_id` = pm2.`post_id`
|
|
AND pm2.`meta_key` = '_wp_attachment_metadata'
|
|
WHERE pm1.`meta_key` = 'amazonS3_info'
|
|
AND pm2.`meta_value` like '%%WP_Error%%'";
|
|
|
|
if ( $count ) {
|
|
$sql = 'SELECT COUNT(*)' . $sql;
|
|
|
|
return $wpdb->get_var( $sql );
|
|
}
|
|
|
|
$sql = "SELECT pm1.`post_id` as `ID`, pm1.`meta_value` AS 'provider_object'" . $sql;
|
|
|
|
if ( $limit && $limit > 0 ) {
|
|
$sql .= sprintf( ' LIMIT %d', (int) $limit );
|
|
}
|
|
|
|
return $wpdb->get_results( $sql, OBJECT );
|
|
}
|
|
}
|