src/MessageBundleTranslation/MessageBundleContent.php
<?php
declare( strict_types = 1 );
namespace MediaWiki\Extension\Translate\MessageBundleTranslation;
use FormatJson;
use JsonContent;
use MediaWiki\Message\Message;
use MediaWiki\Status\Status;
use MediaWiki\User\User;
use WikiPage;
/**
* @author Niklas Laxström
* @license GPL-2.0-or-later
* @since 2021.05
*/
class MessageBundleContent extends JsonContent {
public const CONTENT_MODEL_ID = 'translate-messagebundle';
// List of supported metadata keys
/** @phpcs-require-sorted-array */
public const METADATA_KEYS = [
'allowOnlyPriorityLanguages',
'description',
'label',
'priorityLanguages',
'sourceLanguage'
];
private ?array $messages = null;
private ?MessageBundleMetadata $metadata = null;
public function __construct( $text, $modelId = self::CONTENT_MODEL_ID ) {
parent::__construct( $text, $modelId );
}
public function isValid(): bool {
try {
$this->getMessages();
$this->getMetadata();
return parent::isValid();
} catch ( MalformedBundle $e ) {
return false;
}
}
/** @throws MalformedBundle */
public function validate(): void {
$this->getMessages();
$this->getMetadata();
}
public function prepareSave( WikiPage $page, $flags, $parentRevId, User $user ) {
// TODO: Should be removed when it is no longer needed for backwards compatibility.
// This will give an informative error message when trying to change the content model
try {
$this->getMessages();
$this->getMetadata();
return Status::newGood();
} catch ( MalformedBundle $e ) {
// XXX: We have no context source nor is there Message::messageParam :(
return Status::newFatal( 'translate-messagebundle-validation-error', wfMessage( $e ) );
}
}
/** @throws MalformedBundle */
public function getMessages(): array {
if ( $this->messages !== null ) {
return $this->messages;
}
$data = $this->getRawData();
// Remove the metadata since we are not concerned with it.
unset( $data['@metadata'] );
foreach ( $data as $key => $value ) {
if ( $key === '' ) {
throw new MalformedBundle( 'translate-messagebundle-error-key-empty' );
}
if ( strlen( $key ) > 100 ) {
throw new MalformedBundle(
'translate-messagebundle-error-key-too-long',
[ $key ]
);
}
if ( !preg_match( '/^[a-zA-Z0-9-_.]+$/', $key ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-key-invalid-characters',
[ $key ]
);
}
if ( !is_string( $value ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-value',
[ $key ]
);
}
if ( trim( $value ) === '' ) {
throw new MalformedBundle(
'translate-messagebundle-error-empty-value',
[ $key ]
);
}
}
$this->messages = $data;
return $this->messages;
}
public function getMetadata(): MessageBundleMetadata {
if ( $this->metadata !== null ) {
return $this->metadata;
}
$data = $this->getRawData();
$metadata = $data['@metadata'] ?? [];
if ( !is_array( $metadata ) ) {
throw new MalformedBundle( 'translate-messagebundle-error-metadata-type' );
}
foreach ( $metadata as $key => $value ) {
if ( !in_array( $key, self::METADATA_KEYS ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-metadata',
[ $key, Message::listParam( self::METADATA_KEYS ) ]
);
}
}
$sourceLanguage = $metadata['sourceLanguage'] ?? null;
if ( $sourceLanguage && !is_string( $sourceLanguage ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-sourcelanguage', [ $sourceLanguage ]
);
}
$priorityLanguageCodes = $metadata['priorityLanguages'] ?? null;
if ( $priorityLanguageCodes ) {
if ( !is_array( $priorityLanguageCodes ) ) {
throw new MalformedBundle( 'translate-messagebundle-error-invalid-prioritylanguage-format' );
}
$priorityLanguageCodes = array_unique( $priorityLanguageCodes );
}
$description = $metadata['description'] ?? null;
if ( $description !== null ) {
if ( !is_string( $description ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-description'
);
}
$description = trim( $description ) === '' ? null : trim( $description );
}
$label = $metadata['label'] ?? null;
if ( $label !== null ) {
if ( !is_string( $label ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-label'
);
}
$label = trim( $label ) === '' ? null : trim( $label );
}
$this->metadata = new MessageBundleMetadata(
$sourceLanguage,
$priorityLanguageCodes,
(bool)( $metadata['allowOnlyPriorityLanguages'] ?? false ),
$description,
$label
);
return $this->metadata;
}
private function getRawData(): array {
$status = FormatJson::parse( $this->getText(), FormatJson::FORCE_ASSOC );
if ( !$status->isOK() ) {
throw new MalformedBundle(
'translate-messagebundle-error-parsing',
[ $status->getMessage()->text() ]
);
}
$data = $status->getValue();
// Crude check that we have an associative array (or empty array)
if ( !is_array( $data ) || ( $data !== [] && array_values( $data ) === $data ) ) {
throw new MalformedBundle(
'translate-messagebundle-error-invalid-array',
[ gettype( $data ) ]
);
}
return $data;
}
}