HEX
Server: Apache/2.4.65 (Debian)
System: Linux 88f31f35b0b8 6.1.0-38-amd64 #1 SMP PREEMPT_DYNAMIC Debian 6.1.147-1 (2025-08-02) x86_64
User: www-data (33)
PHP: 8.2.29
Disabled: NONE
Upload Files
File: /var/www/html/wp-content/plugins/wpseo-video/detail-retrieval/class-youtube.php
<?php
/**
 * Yoast SEO Video plugin file.
 *
 * @package    Internals
 * @since      1.7.0
 * @version    1.7.0
 */

// Avoid direct calls to this file.
if ( ! class_exists( 'WPSEO_Video_Sitemap' ) ) {
	header( 'Status: 403 Forbidden' );
	header( 'HTTP/1.1 403 Forbidden' );
	exit();
}


/**
 * YouTube Video SEO Details
 *
 * @link https://www.youtube.com/yt/dev/
 *
 * Also available: oembed at http://www.youtube.com/oembed?url=%s
 */
if ( ! class_exists( 'WPSEO_Video_Details_Youtube' ) ) {

	/**
	 * Class WPSEO_Video_Details_Youtube
	 */
	class WPSEO_Video_Details_Youtube extends WPSEO_Video_Details {

		/**
		 * Sprintf template to create a URL from an ID.
		 *
		 * @var string
		 */
		protected $url_template = 'http://www.youtube.com/v/%s';

		/**
		 * Information on the remote URL to use for retrieving the video details.
		 *
		 * @var string[]
		 */
		protected $remote_url = [
			'pattern'       => 'https://www.googleapis.com/youtube/v3/videos?part=snippet,status,statistics,contentDetails,player&id=%1$s&fields=items&key=%2$s',
			'replace_key'   => 'id',
			'response_type' => 'json',
		];

		/**
		 * Google API access key.
		 *
		 * @var string
		 */
		protected $api_key = 'AIzaSyD5OvaM_lplFbQjDr-tgK9SEkLxdkW79Lw';

		/**
		 * Retrieve the video id from a known video url based on a regex match.
		 * Also change the url based on the new video id.
		 *
		 * @param int $match_nr The captured parenthesized sub-pattern to use from matches. Defaults to 1.
		 *
		 * @return void
		 */
		protected function determine_video_id_from_url( $match_nr = 1 ) {
			if ( isset( $this->vid['url'] ) && is_string( $this->vid['url'] ) && $this->vid['url'] !== '' ) {

				$yt_id = WPSEO_Video_Sitemap::$youtube_id_pattern;

				$patterns = [
					'`youtube\.(?:com|[a-z]{2})/(?:v/|(?:watch)?(?:\?|#!)(?:.*&)?v=)(' . $yt_id . ')`i',
					'`youtube(?:-nocookie)?\.com/(?:embed|v)/(?!videoseries|playlist)(' . $yt_id . ')`i',
					'`https?://youtu\.be/(' . $yt_id . ')`i',
				];

				foreach ( $patterns as $pattern ) {
					if ( preg_match( $pattern, $this->vid['url'], $match ) ) {
						$this->vid['id'] = $match[ $match_nr ];
						break;
					}
				}

				// @todo [JRF => Yoast] shouldn't this be checked against $youtube_id_pattern as well ?
				if ( ( ! isset( $this->vid['id'] ) || empty( $this->vid['id'] ) ) && ! preg_match( '`^(?:http|//)`', $this->vid['url'] ) ) {
					$this->vid['id'] = $this->vid['url'];
				}
			}
		}

		/**
		 * Check if the response is for a video
		 *
		 * @return bool
		 */
		protected function is_video_response() {
			return ( ! empty( $this->decoded_response ) );
		}

		/**
		 * Set the video last fetched.
		 *
		 * @return void
		 */
		protected function set_last_fetched() {
			$this->vid['last_fetched'] = time();
		}

		/**
		 * Set the video duration
		 *
		 * @return void
		 */
		protected function set_duration() {
			if ( ! empty( $this->decoded_response->contentDetails->duration ) ) {
				$date = new DateTime( '@0' );
				$date->add( new DateInterval( $this->decoded_response->contentDetails->duration ) );

				$this->vid['duration'] = (int) $date->format( 'U' );
			}
		}

		/**
		 * Set the video height
		 *
		 * @return void
		 */
		protected function set_height() {
			if ( ! empty( $this->decoded_response->player->embedHtml )
				&& preg_match( '` height="([^"]+)"`i', $this->decoded_response->player->embedHtml, $match )
			) {
				$this->vid['height'] = (int) $match[1];
			}
			else {
				// Fall back to hard-coded default.
				$this->vid['height'] = 390;
			}
		}

		/**
		 * Set the player location
		 *
		 * @return void
		 */
		protected function set_player_loc() {
			// Bow out if video is explicitely not embeddable - falls through if embeddable status not available.
			if ( isset( $this->decoded_response->status->embeddable ) && $this->decoded_response->status->embeddable !== true ) {
				return;
			}

			if ( ! empty( $this->decoded_response->player->embedHtml )
				&& preg_match( '` src="([^"]+)"`i', $this->decoded_response->player->embedHtml, $match )
			) {
				$player_loc = $match[1];
			}
			else {
				// Fall back to hard-coded default.
				$player_loc = '//www.youtube.com/embed/' . rawurlencode( $this->vid['id'] );
			}

			// Add protocol if the resulting player URL would be protocol-less.
			if ( strpos( $player_loc, 'http' ) !== 0 ) {
				$player_loc = 'https:' . $player_loc;
			}

			$this->vid['player_loc'] = $player_loc;
		}

		/**
		 * Set the thumbnail location
		 *
		 * @return void
		 */
		protected function set_thumbnail_loc() {
			$formats = [ 'maxres', 'standard', 'high', 'medium', 'default' ];

			foreach ( $formats as $format ) {
				if ( ! empty( $this->decoded_response->snippet->thumbnails->$format ) && is_object( $this->decoded_response->snippet->thumbnails->$format ) ) {
					$thumbnail = $this->decoded_response->snippet->thumbnails->$format;
					if ( ! empty( $thumbnail->url ) ) {
						$image = $this->make_image_local( $thumbnail->url );
						if ( is_string( $image ) && $image !== '' ) {
							$this->vid['thumbnail_loc'] = $image;

							return;
						}
					}
				}
			}
		}

		/**
		 * Set the video view count
		 *
		 * @return void
		 */
		protected function set_view_count() {
			if ( ! empty( $this->decoded_response->statistics->viewCount ) ) {
				$this->vid['view_count'] = $this->decoded_response->statistics->viewCount;
			}
		}

		/**
		 * Set the video width
		 *
		 * @return void
		 */
		protected function set_width() {
			if ( ! empty( $this->decoded_response->player->embedHtml )
				&& preg_match( '` width="([^"]+)"`i', $this->decoded_response->player->embedHtml, $match )
			) {
				$this->vid['width'] = (int) $match[1];
			}
			else {
				// Fall back to hard-coded default.
				$this->vid['width'] = 640;
			}
		}

		/**
		 * Extends the parent method. By letting the parent set the response and get the first item afterwards
		 *
		 * @return void
		 */
		protected function decode_as_json() {
			parent::decode_as_json();

			if ( ! empty( $this->decoded_response->items[0] ) ) {
				$this->decoded_response = $this->decoded_response->items[0];
			}
			else {
				// Reset if no valid data received.
				$this->decoded_response = null;
			}
		}
	}
}