123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- <?php
- class WebPHandler extends BitmapHandler {
- const BROKEN_FILE = '0';
-
- const MINIMUM_CHUNK_HEADER_LENGTH = 18;
-
- const _MW_WEBP_VERSION = 1;
- const VP8X_ICC = 32;
- const VP8X_ALPHA = 16;
- const VP8X_EXIF = 8;
- const VP8X_XMP = 4;
- const VP8X_ANIM = 2;
- public function getMetadata( $image, $filename ) {
- $parsedWebPData = self::extractMetadata( $filename );
- if ( !$parsedWebPData ) {
- return self::BROKEN_FILE;
- }
- $parsedWebPData['metadata']['_MW_WEBP_VERSION'] = self::_MW_WEBP_VERSION;
- return serialize( $parsedWebPData );
- }
- public function getMetadataType( $image ) {
- return 'parsed-webp';
- }
- public function isMetadataValid( $image, $metadata ) {
- if ( $metadata === self::BROKEN_FILE ) {
-
- return self::METADATA_GOOD;
- }
- Wikimedia\suppressWarnings();
- $data = unserialize( $metadata );
- Wikimedia\restoreWarnings();
- if ( !$data || !is_array( $data ) ) {
- wfDebug( __METHOD__ . " invalid WebP metadata\n" );
- return self::METADATA_BAD;
- }
- if ( !isset( $data['metadata']['_MW_WEBP_VERSION'] )
- || $data['metadata']['_MW_WEBP_VERSION'] != self::_MW_WEBP_VERSION
- ) {
- wfDebug( __METHOD__ . " old but compatible WebP metadata\n" );
- return self::METADATA_COMPATIBLE;
- }
- return self::METADATA_GOOD;
- }
-
- public static function extractMetadata( $filename ) {
- wfDebugLog( 'WebP', __METHOD__ . ": Extracting metadata from $filename\n" );
- $info = RiffExtractor::findChunksFromFile( $filename, 100 );
- if ( $info === false ) {
- wfDebugLog( 'WebP', __METHOD__ . ": Not a valid RIFF file\n" );
- return false;
- }
- if ( $info['fourCC'] != 'WEBP' ) {
- wfDebugLog( 'WebP', __METHOD__ . ': FourCC was not WEBP: ' .
- bin2hex( $info['fourCC'] ) . " \n" );
- return false;
- }
- $metadata = self::extractMetadataFromChunks( $info['chunks'], $filename );
- if ( !$metadata ) {
- wfDebugLog( 'WebP', __METHOD__ . ": No VP8 chunks found\n" );
- return false;
- }
- return $metadata;
- }
-
- public static function extractMetadataFromChunks( $chunks, $filename ) {
- $vp8Info = [];
- foreach ( $chunks as $chunk ) {
- if ( !in_array( $chunk['fourCC'], [ 'VP8 ', 'VP8L', 'VP8X' ] ) ) {
-
- continue;
- }
- $chunkHeader = file_get_contents( $filename, false, null,
- $chunk['start'], self::MINIMUM_CHUNK_HEADER_LENGTH );
- wfDebugLog( 'WebP', __METHOD__ . ": {$chunk['fourCC']}\n" );
- switch ( $chunk['fourCC'] ) {
- case 'VP8 ':
- return array_merge( $vp8Info,
- self::decodeLossyChunkHeader( $chunkHeader ) );
- case 'VP8L':
- return array_merge( $vp8Info,
- self::decodeLosslessChunkHeader( $chunkHeader ) );
- case 'VP8X':
- $vp8Info = array_merge( $vp8Info,
- self::decodeExtendedChunkHeader( $chunkHeader ) );
-
- break;
- }
- }
- return $vp8Info;
- }
-
- protected static function decodeLossyChunkHeader( $header ) {
-
-
-
-
- $syncCode = substr( $header, 11, 3 );
- if ( $syncCode != "\x9D\x01\x2A" ) {
- wfDebugLog( 'WebP', __METHOD__ . ': Invalid sync code: ' .
- bin2hex( $syncCode ) . "\n" );
- return [];
- }
-
- $imageSize = unpack( 'v2', substr( $header, 14, 4 ) );
-
- return [
- 'compression' => 'lossy',
- 'width' => $imageSize[1] & 0x3FFF,
- 'height' => $imageSize[2] & 0x3FFF
- ];
- }
-
- public static function decodeLosslessChunkHeader( $header ) {
-
-
-
- if ( $header{8} != "\x2F" ) {
- wfDebugLog( 'WebP', __METHOD__ . ': Invalid signature: ' .
- bin2hex( $header{8} ) . "\n" );
- return [];
- }
-
-
- $imageSize = unpack( 'C4', substr( $header, 9, 4 ) );
- return [
- 'compression' => 'lossless',
- 'width' => ( $imageSize[1] | ( ( $imageSize[2] & 0x3F ) << 8 ) ) + 1,
- 'height' => ( ( ( $imageSize[2] & 0xC0 ) >> 6 ) |
- ( $imageSize[3] << 2 ) | ( ( $imageSize[4] & 0x03 ) << 10 ) ) + 1
- ];
- }
-
- public static function decodeExtendedChunkHeader( $header ) {
-
-
-
- $flags = unpack( 'c', substr( $header, 8, 1 ) );
-
- $width = unpack( 'V', substr( $header, 12, 3 ) . "\x00" );
- $height = unpack( 'V', substr( $header, 15, 3 ) . "\x00" );
- return [
- 'compression' => 'unknown',
- 'animated' => ( $flags[1] & self::VP8X_ANIM ) == self::VP8X_ANIM,
- 'transparency' => ( $flags[1] & self::VP8X_ALPHA ) == self::VP8X_ALPHA,
- 'width' => ( $width[1] & 0xFFFFFF ) + 1,
- 'height' => ( $height[1] & 0xFFFFFF ) + 1
- ];
- }
- public function getImageSize( $file, $path, $metadata = false ) {
- if ( $file === null ) {
- $metadata = self::getMetadata( $file, $path );
- }
- if ( $metadata === false && $file instanceof File ) {
- $metadata = $file->getMetadata();
- }
- Wikimedia\suppressWarnings();
- $metadata = unserialize( $metadata );
- Wikimedia\restoreWarnings();
- if ( $metadata == false ) {
- return false;
- }
- return [ $metadata['width'], $metadata['height'] ];
- }
-
- public function mustRender( $file ) {
- return true;
- }
-
- public function canRender( $file ) {
- if ( self::isAnimatedImage( $file ) ) {
- return false;
- }
- return true;
- }
-
- public function isAnimatedImage( $image ) {
- $ser = $image->getMetadata();
- if ( $ser ) {
- $metadata = unserialize( $ser );
- if ( isset( $metadata['animated'] ) && $metadata['animated'] === true ) {
- return true;
- }
- }
- return false;
- }
- public function canAnimateThumbnail( $file ) {
- return false;
- }
-
- public function getThumbType( $ext, $mime, $params = null ) {
- return [ 'png', 'image/png' ];
- }
-
- protected function getScalerType( $dstPath, $checkDstPath = true ) {
- return 'im';
- }
- }
|