File "Image_Downloader.php"

Full Path: /home/romayxjt/public_html/wp-content/plugins/kadence-starter-templates/inc/resources/Image_Downloader/Image_Downloader.php
File size: 5.16 KB
MIME-type: text/x-php
Charset: utf-8

<?php declare( strict_types=1 );

namespace KadenceWP\KadenceStarterTemplates\Image_Downloader;

use KadenceWP\KadenceStarterTemplates\Psr\Log\LoggerInterface;
use KadenceWP\KadenceStarterTemplates\StellarWP\ProphecyMonorepo\ImageDownloader\Exceptions\ImageDownloadException;
use KadenceWP\KadenceStarterTemplates\StellarWP\ProphecyMonorepo\ImageDownloader\ImageDownloader;
use Throwable;

final class Image_Downloader {

	/**
	 * @var ImageDownloader
	 */
	private $downloader;

	/**
	 * @var WordPress_Importer
	 */
	private $importer;

	/**
	 * @var Pexels_ID_Registry
	 */
	private $registry;

	/**
	 * @var LoggerInterface
	 */
	private $logger;

	public function __construct(
		ImageDownloader $downloader,
		WordPress_Importer $importer,
		Pexels_ID_Registry $registry,
		LoggerInterface $logger
	) {
		$this->downloader = $downloader;
		$this->importer   = $importer;
		$this->registry   = $registry;
		$this->logger     = $logger;
	}

	/**
	 * Download and import a collection of images to the WordPress library.
	 *
	 * @param array<array{
	 *      collection_slug?: string,
	 *      image_type?: string,
	 *      images: array<int, array{
	 *           id: int,
	 *           post_id?: int,
	 *           width: int,
	 *           height: int,
	 *           alt: string,
	 *           url: string,
	 *           photographer: string,
	 *           photographer_url: string,
	 *           avg_color: string,
	 *           sizes: non-empty-array<int,array{name: string, src: string}>
	 *      }>
	 *     }> $images
	 *
	 * @return array<array{id: int, url: string}>
	 * @throws ImageDownloadException
	 * @throws Throwable
	 */
	public function download( array $images ): array {
		$existing   = $this->get_existing_images( $images );
		$downloaded = empty( $images['images'] ) ? [] : $this->download_images( $images );

		// Merge any imported images with existing images.
		return array_merge( $existing, $downloaded );
	}

	/**
	 * Based on the incoming request, determine which images have already been
	 * downloaded.
	 *
	 * This is weird because the frontend replaces the id and the URL, which is normally
	 * the pexels_id and pexels URL with the attachment_id and full URL to the image.
	 *
	 * Now local images will pass a post_id, but depending on the image state, we still
	 * have to check the meta for the pexels_id, because the image may exist in the media
	 * library.
	 *
	 * @param array<array{
	 *       collection_slug?: string,
	 *       image_type?: string,
	 *       images: array<int, array{
	 *            id: int,
	 *            post_id?: int,
	 *            width: int,
	 *            height: int,
	 *            alt: string,
	 *            url: string,
	 *            photographer: string,
	 *            photographer_url: string,
	 *            avg_color: string,
	 *            sizes: non-empty-array<int,array{name: string, src: string}>
	 *       }>
	 *      }> $images
	 *
	 * @return array<array{id: int, url: string}>
	 */
	private function get_existing_images( array &$images ): array {
		$existing = [];
		$ids      = $this->registry->all();

		foreach ( $images['images'] as $key => $image ) {
			$post_id   = $image['post_id'] ?? false;
			$pexels_id = $image['id'] ?? false;

			if ( $post_id !== false ) {
				// We were provided a post_id, but it's not in our list of attachments.
				if ( ! isset( $ids['post_ids'][ $post_id ] ) ) {
					continue;
				}
			} elseif ( $pexels_id !== false ) {
				$post_id = $ids['pexels_ids'][ $pexels_id ] ?? false;

				if ( $post_id === false ) {
					continue;
				}
			}

			$existing[] = [
				'id'  => $post_id,
				'url' => wp_get_attachment_image_url( $post_id ),
			];

			unset( $images['images'][ $key ] );
		}

		return $existing;
	}

	/**
	 * Download and import Pexels images to the WordPress library.
	 *
	 * @param array<array{
	 *        collection_slug?: string,
	 *        image_type?: string,
	 *        images: array<int, array{
	 *             id: int,
	 *             post_id?: int,
	 *             width: int,
	 *             height: int,
	 *             alt: string,
	 *             url: string,
	 *             photographer: string,
	 *             photographer_url: string,
	 *             avg_color: string,
	 *             sizes: non-empty-array<int,array{name: string, src: string}>
	 *        }>
	 *       }> $images
	 *
	 * @return array<array{id: int, url: string}>
	 * @throws Throwable
	 */
	private function download_images( array $images ): array {
		if ( ! current_user_can( 'upload_files' ) ) {
			$this->logger->warning( 'User doesn\'t have permission to upload files. Aborting image download' );

			return [];
		}

		$path = wp_get_upload_dir()['path'] ?? '';

		if ( ! $path ) {
			return [];
		}

		if ( ! isset( $images['image_type'] ) ) {
			$images['image_type'] = 'jpg';
		}

		if ( ! isset( $images['collection_slug'] ) ) {
			$images['collection_slug'] = wp_generate_password( 12, false );
		}

		$collection[] = $images;

		try {
			$downloaded = $this->downloader->download( $collection, $path );

			return $this->importer->import( $downloaded );
		} catch ( Throwable $e ) {
			$this->logger->error( 'Image download or import error', [
				'message' => $e->getMessage(),
				'trace'   => $e->getTraceAsString(),
			] );

			return [];
		}
	}

}