mirror of
https://github.com/WordPress/WordPress.git
synced 2025-01-15 12:51:20 +01:00
3af51954e0
Ensures that string lookup in MO files only uses the singular string. This matches expected behavior with gettext files and improves compatibility for cases where for example both `__( 'Product' )` and `_n( 'Product', 'Products’, num )` are used in a project, where both will use the same translation for the singular version. Maintains backward compatibility and feature parity with the pomo library and the PHP translation file format. Replaces [57386], which was reverted in [57505], with a more accurate and performant solution. See #59656. Built from https://develop.svn.wordpress.org/trunk@57513 git-svn-id: http://core.svn.wordpress.org/trunk@57014 1a063a9b-81f0-0310-95a4-ce76da25c4cd
228 lines
5.8 KiB
PHP
228 lines
5.8 KiB
PHP
<?php
|
|
/**
|
|
* I18N: WP_Translation_File_MO class.
|
|
*
|
|
* @package WordPress
|
|
* @subpackage I18N
|
|
* @since 6.5.0
|
|
*/
|
|
|
|
/**
|
|
* Class WP_Translation_File_MO.
|
|
*
|
|
* @since 6.5.0
|
|
*/
|
|
class WP_Translation_File_MO extends WP_Translation_File {
|
|
/**
|
|
* Endian value.
|
|
*
|
|
* V for little endian, N for big endian, or false.
|
|
*
|
|
* Used for unpack().
|
|
*
|
|
* @since 6.5.0
|
|
* @var false|'V'|'N'
|
|
*/
|
|
protected $uint32 = false;
|
|
|
|
/**
|
|
* The magic number of the GNU message catalog format.
|
|
*
|
|
* @since 6.5.0
|
|
* @var int
|
|
*/
|
|
const MAGIC_MARKER = 0x950412de;
|
|
|
|
/**
|
|
* Detects endian and validates file.
|
|
*
|
|
* @since 6.5.0
|
|
*
|
|
* @param string $header File contents.
|
|
* @return false|'V'|'N' V for little endian, N for big endian, or false on failure.
|
|
*/
|
|
protected function detect_endian_and_validate_file( string $header ) {
|
|
$big = unpack( 'N', $header );
|
|
|
|
if ( false === $big ) {
|
|
return false;
|
|
}
|
|
|
|
$big = reset( $big );
|
|
|
|
if ( false === $big ) {
|
|
return false;
|
|
}
|
|
|
|
$little = unpack( 'V', $header );
|
|
|
|
if ( false === $little ) {
|
|
return false;
|
|
}
|
|
|
|
$little = reset( $little );
|
|
|
|
if ( false === $little ) {
|
|
return false;
|
|
}
|
|
|
|
if ( self::MAGIC_MARKER === $big ) {
|
|
return 'N';
|
|
}
|
|
|
|
if ( self::MAGIC_MARKER === $little ) {
|
|
return 'V';
|
|
}
|
|
|
|
$this->error = 'Magic marker does not exist';
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Parses the file.
|
|
*
|
|
* @since 6.5.0
|
|
*
|
|
* @return bool True on success, false otherwise.
|
|
*/
|
|
protected function parse_file(): bool {
|
|
$this->parsed = true;
|
|
|
|
$file_contents = file_get_contents( $this->file ); // phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents
|
|
|
|
if ( false === $file_contents ) {
|
|
return false;
|
|
}
|
|
|
|
$file_length = strlen( $file_contents );
|
|
|
|
if ( $file_length < 24 ) {
|
|
$this->error = 'Invalid data';
|
|
return false;
|
|
}
|
|
|
|
$this->uint32 = $this->detect_endian_and_validate_file( substr( $file_contents, 0, 4 ) );
|
|
|
|
if ( false === $this->uint32 ) {
|
|
return false;
|
|
}
|
|
|
|
$offsets = substr( $file_contents, 4, 24 );
|
|
|
|
if ( false === $offsets ) {
|
|
return false;
|
|
}
|
|
|
|
$offsets = unpack( "{$this->uint32}rev/{$this->uint32}total/{$this->uint32}originals_addr/{$this->uint32}translations_addr/{$this->uint32}hash_length/{$this->uint32}hash_addr", $offsets );
|
|
|
|
if ( false === $offsets ) {
|
|
return false;
|
|
}
|
|
|
|
$offsets['originals_length'] = $offsets['translations_addr'] - $offsets['originals_addr'];
|
|
$offsets['translations_length'] = $offsets['hash_addr'] - $offsets['translations_addr'];
|
|
|
|
if ( $offsets['rev'] > 0 ) {
|
|
$this->error = 'Unsupported revision';
|
|
return false;
|
|
}
|
|
|
|
if ( $offsets['translations_addr'] > $file_length || $offsets['originals_addr'] > $file_length ) {
|
|
$this->error = 'Invalid data';
|
|
return false;
|
|
}
|
|
|
|
// Load the Originals.
|
|
$original_data = str_split( substr( $file_contents, $offsets['originals_addr'], $offsets['originals_length'] ), 8 );
|
|
$translations_data = str_split( substr( $file_contents, $offsets['translations_addr'], $offsets['translations_length'] ), 8 );
|
|
|
|
foreach ( array_keys( $original_data ) as $i ) {
|
|
$o = unpack( "{$this->uint32}length/{$this->uint32}pos", $original_data[ $i ] );
|
|
$t = unpack( "{$this->uint32}length/{$this->uint32}pos", $translations_data[ $i ] );
|
|
|
|
if ( false === $o || false === $t ) {
|
|
continue;
|
|
}
|
|
|
|
$original = substr( $file_contents, $o['pos'], $o['length'] );
|
|
$translation = substr( $file_contents, $t['pos'], $t['length'] );
|
|
// GlotPress bug.
|
|
$translation = rtrim( $translation, "\0" );
|
|
|
|
// Metadata about the MO file is stored in the first translation entry.
|
|
if ( '' === $original ) {
|
|
foreach ( explode( "\n", $translation ) as $meta_line ) {
|
|
if ( '' === $meta_line ) {
|
|
continue;
|
|
}
|
|
|
|
list( $name, $value ) = array_map( 'trim', explode( ':', $meta_line, 2 ) );
|
|
|
|
$this->headers[ strtolower( $name ) ] = $value;
|
|
}
|
|
} else {
|
|
/*
|
|
* In MO files, the key normally contains both singular and plural versions.
|
|
* However, this just adds the singular string for lookup,
|
|
* which caters for cases where both __( 'Product' ) and _n( 'Product', 'Products' )
|
|
* are used and the translation is expected to be the same for both.
|
|
*/
|
|
$parts = explode( "\0", (string) $original );
|
|
|
|
$this->entries[ $parts[0] ] = $translation;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Exports translation contents as a string.
|
|
*
|
|
* @since 6.5.0
|
|
*
|
|
* @return string Translation file contents.
|
|
*/
|
|
public function export(): string {
|
|
// Prefix the headers as the first key.
|
|
$headers_string = '';
|
|
foreach ( $this->headers as $header => $value ) {
|
|
$headers_string .= "{$header}: $value\n";
|
|
}
|
|
$entries = array_merge( array( '' => $headers_string ), $this->entries );
|
|
$entry_count = count( $entries );
|
|
|
|
if ( false === $this->uint32 ) {
|
|
$this->uint32 = 'V';
|
|
}
|
|
|
|
$bytes_for_entries = $entry_count * 4 * 2;
|
|
// Pair of 32bit ints per entry.
|
|
$originals_addr = 28; /* header */
|
|
$translations_addr = $originals_addr + $bytes_for_entries;
|
|
$hash_addr = $translations_addr + $bytes_for_entries;
|
|
$entry_offsets = $hash_addr;
|
|
|
|
$file_header = pack( $this->uint32 . '*', self::MAGIC_MARKER, 0 /* rev */, $entry_count, $originals_addr, $translations_addr, 0 /* hash_length */, $hash_addr );
|
|
|
|
$o_entries = '';
|
|
$t_entries = '';
|
|
$o_addr = '';
|
|
$t_addr = '';
|
|
|
|
foreach ( array_keys( $entries ) as $original ) {
|
|
$o_addr .= pack( $this->uint32 . '*', strlen( $original ), $entry_offsets );
|
|
$entry_offsets += strlen( $original ) + 1;
|
|
$o_entries .= $original . "\0";
|
|
}
|
|
|
|
foreach ( $entries as $translations ) {
|
|
$t_addr .= pack( $this->uint32 . '*', strlen( $translations ), $entry_offsets );
|
|
$entry_offsets += strlen( $translations ) + 1;
|
|
$t_entries .= $translations . "\0";
|
|
}
|
|
|
|
return $file_header . $o_addr . $t_addr . $o_entries . $t_entries;
|
|
}
|
|
}
|