23use CLDRPluralRuleParser\Error as CLDRPluralRuleError;
24use CLDRPluralRuleParser\Evaluator;
29use Psr\Log\LoggerInterface;
124 'fallback',
'namespaceNames',
'bookstoreList',
125 'magicWords',
'messages',
'rtl',
126 'digitTransformTable',
'separatorTransformTable',
127 'minimumGroupingDigits',
'fallback8bitEncoding',
128 'linkPrefixExtension',
'linkTrail',
'linkPrefixCharset',
129 'namespaceAliases',
'dateFormats',
'datePreferences',
130 'datePreferenceMigrationMap',
'defaultDateFormat',
131 'specialPageAliases',
'imageFiles',
'preloadedMessages',
132 'namespaceGenderAliases',
'digitGroupingPattern',
'pluralRules',
133 'pluralRuleTypes',
'compiledPluralRules',
141 'namespaceAliases',
'dateFormats',
'imageFiles',
'preloadedMessages'
209 $storeArg[
'directory'] =
210 $conf[
'storeDirectory'] ?: $fallbackCacheDir;
212 if ( !empty( $conf[
'storeClass'] ) ) {
213 $storeClass = $conf[
'storeClass'];
214 } elseif ( $conf[
'store'] ===
'files' || $conf[
'store'] ===
'file' ||
215 ( $conf[
'store'] ===
'detect' && $storeArg[
'directory'] )
217 $storeClass = LCStoreCDB::class;
218 } elseif ( $conf[
'store'] ===
'db' || $conf[
'store'] ===
'detect' ) {
219 $storeClass = LCStoreDB::class;
220 $storeArg[
'server'] = $conf[
'storeServer'] ?? [];
221 } elseif ( $conf[
'store'] ===
'array' ) {
222 $storeClass = LCStoreStaticArray::class;
225 'Please set $wgLocalisationCacheConf[\'store\'] to something sensible.'
229 return new $storeClass( $storeArg );
235 public const CONSTRUCTOR_OPTIONS = [
239 'ExtensionMessagesFiles',
262 LoggerInterface $logger,
263 array $clearStoreCallbacks,
269 $this->options = $options;
270 $this->store = $store;
271 $this->logger = $logger;
272 $this->clearStoreCallbacks = $clearStoreCallbacks;
273 $this->langNameUtils = $langNameUtils;
274 $this->hookRunner =
new HookRunner( $hookContainer );
277 $this->manualRecache = $options->
get(
'manualRecache' );
287 if ( $this->mergeableKeys ===
null ) {
288 $this->mergeableKeys = array_fill_keys( array_merge(
289 self::$mergeableMapKeys,
290 self::$mergeableListKeys,
291 self::$mergeableAliasListKeys,
292 self::$optionalMergeKeys,
297 return isset( $this->mergeableKeys[$key] );
310 if ( !isset( $this->loadedItems[$code][$key] ) ) {
311 $this->loadItem( $code, $key );
314 if ( $key ===
'fallback' && isset( $this->shallowFallbacks[$code] ) ) {
315 return $this->shallowFallbacks[$code];
318 return $this->data[$code][$key];
329 if ( !isset( $this->loadedSubitems[$code][$key][$subkey] ) &&
330 !isset( $this->loadedItems[$code][$key] )
332 $this->loadSubitem( $code, $key, $subkey );
335 return $this->data[$code][$key][$subkey] ??
null;
351 if ( in_array( $key, self::$splitKeys ) ) {
352 return $this->getSubitem( $code,
'list', $key );
354 $item = $this->getItem( $code, $key );
355 if ( is_array( $item ) ) {
356 return array_keys( $item );
369 if ( !isset( $this->initialisedLangs[$code] ) ) {
370 $this->initLanguage( $code );
374 if ( isset( $this->loadedItems[$code][$key] ) ) {
378 if ( isset( $this->shallowFallbacks[$code] ) ) {
379 $this->loadItem( $this->shallowFallbacks[$code], $key );
384 if ( in_array( $key, self::$splitKeys ) ) {
385 $subkeyList = $this->getSubitem( $code,
'list', $key );
386 foreach ( $subkeyList as $subkey ) {
387 if ( isset( $this->data[$code][$key][$subkey] ) ) {
390 $this->data[$code][$key][$subkey] = $this->getSubitem( $code, $key, $subkey );
393 $this->data[$code][$key] = $this->store->get( $code, $key );
396 $this->loadedItems[$code][$key] =
true;
406 if ( !in_array( $key, self::$splitKeys ) ) {
407 $this->loadItem( $code, $key );
412 if ( !isset( $this->initialisedLangs[$code] ) ) {
413 $this->initLanguage( $code );
417 if ( isset( $this->loadedItems[$code][$key] ) ||
418 isset( $this->loadedSubitems[$code][$key][$subkey] )
423 if ( isset( $this->shallowFallbacks[$code] ) ) {
424 $this->loadSubitem( $this->shallowFallbacks[$code], $key, $subkey );
429 $value = $this->store->get( $code,
"$key:$subkey" );
430 $this->data[$code][$key][$subkey] = $value;
431 $this->loadedSubitems[$code][$key][$subkey] =
true;
442 if ( $this->options->get(
'forceRecache' ) && !isset( $this->recachedLangs[$code] ) ) {
443 $this->logger->debug( __METHOD__ .
"($code): forced reload" );
448 $deps = $this->store->get( $code,
'deps' );
449 $keys = $this->store->get( $code,
'list' );
450 $preload = $this->store->get( $code,
'preload' );
452 if ( $deps ===
null ||
$keys ===
null || $preload ===
null ) {
453 $this->logger->debug( __METHOD__ .
"($code): cache missing, need to make one" );
458 foreach ( $deps as $dep ) {
464 $this->logger->debug( __METHOD__ .
"($code): cache for $code expired due to " .
480 if ( isset( $this->initialisedLangs[$code] ) ) {
484 $this->initialisedLangs[$code] =
true;
486 # If the code is of the wrong form for a Messages*.php file, do a shallow fallback
487 if ( !$this->langNameUtils->isValidBuiltInCode( $code ) ) {
488 $this->initShallowFallback( $code,
'en' );
493 # Recache the data if necessary
494 if ( !$this->manualRecache && $this->isExpired( $code ) ) {
495 if ( $this->langNameUtils->isSupportedLanguage( $code ) ) {
496 $this->recache( $code );
497 } elseif ( $code ===
'en' ) {
498 throw new MWException(
'MessagesEn.php is missing.' );
500 $this->initShallowFallback( $code,
'en' );
507 $preload = $this->getItem( $code,
'preload' );
508 if ( $preload ===
null ) {
509 if ( $this->manualRecache ) {
511 if ( $code ===
'en' ) {
512 throw new MWException(
'No localisation cache found for English. ' .
513 'Please run maintenance/rebuildLocalisationCache.php.' );
515 $this->initShallowFallback( $code,
'en' );
519 throw new MWException(
'Invalid or missing localisation cache.' );
522 $this->data[$code] = $preload;
523 foreach ( $preload as $key => $item ) {
524 if ( in_array( $key, self::$splitKeys ) ) {
525 foreach ( $item as $subkey => $subitem ) {
526 $this->loadedSubitems[$code][$key][$subkey] =
true;
529 $this->loadedItems[$code][$key] =
true;
541 $this->data[$primaryCode] =& $this->data[$fallbackCode];
542 $this->loadedItems[$primaryCode] =& $this->loadedItems[$fallbackCode];
543 $this->loadedSubitems[$primaryCode] =& $this->loadedSubitems[$fallbackCode];
544 $this->shallowFallbacks[$primaryCode] = $fallbackCode;
558 if ( $_fileType ==
'core' || $_fileType ==
'extension' ) {
559 foreach ( self::$allKeys as $key ) {
562 if ( isset( $$key ) ) {
566 } elseif ( $_fileType ==
'aliases' ) {
568 if ( isset( $aliases ) ) {
569 $data[
'aliases'] = $aliases;
572 throw new MWException( __METHOD__ .
": Invalid file type: $_fileType" );
585 if ( !is_readable( $fileName ) ) {
589 $json = file_get_contents( $fileName );
590 if ( $json ===
false ) {
594 $data = FormatJson::decode( $json,
true );
595 if ( $data ===
null ) {
596 throw new MWException( __METHOD__ .
": Invalid JSON file: $fileName" );
600 foreach ( $data as $key => $unused ) {
601 if ( $key ===
'' || $key[0] ===
'@' ) {
602 unset( $data[$key] );
607 return [
'messages' => $data ];
617 $rules = $this->getPluralRules( $code );
618 if ( $rules ===
null ) {
622 $compiledRules = Evaluator::compile( $rules );
623 }
catch ( CLDRPluralRuleError $e ) {
624 $this->logger->debug( $e->getMessage() );
629 return $compiledRules;
640 if ( $this->pluralRules ===
null ) {
641 $this->loadPluralFiles();
643 return $this->pluralRules[$code] ??
null;
654 if ( $this->pluralRuleTypes ===
null ) {
655 $this->loadPluralFiles();
657 return $this->pluralRuleTypes[$code] ??
null;
664 foreach ( $this->getPluralFiles() as $fileName ) {
665 $this->loadPluralFile( $fileName );
673 "$IP/languages/data/plurals.xml",
675 "$IP/languages/data/plurals-mediawiki.xml",
688 $xml = file_get_contents( $fileName );
690 throw new MWException(
"Unable to read plurals file $fileName" );
692 $doc =
new DOMDocument;
693 $doc->loadXML( $xml );
694 $rulesets = $doc->getElementsByTagName(
"pluralRules" );
695 foreach ( $rulesets as $ruleset ) {
696 $codes = $ruleset->getAttribute(
'locales' );
699 $ruleElements = $ruleset->getElementsByTagName(
"pluralRule" );
700 foreach ( $ruleElements as $elt ) {
701 $ruleType = $elt->getAttribute(
'count' );
702 if ( $ruleType ===
'other' ) {
706 $rules[] = $elt->nodeValue;
707 $ruleTypes[] = $ruleType;
709 foreach ( explode(
' ', $codes ) as $code ) {
710 $this->pluralRules[$code] = $rules;
711 $this->pluralRuleTypes[$code] = $ruleTypes;
727 $fileName = $this->langNameUtils->getMessagesFileName( $code );
728 if ( !file_exists( $fileName ) ) {
732 $data = $this->readPHPFile( $fileName,
'core' );
736 $data[
'pluralRules'] = $this->getPluralRules( $code );
738 $data[
'compiledPluralRules'] = $this->getCompiledPluralRules( $code );
740 $data[
'pluralRuleTypes'] = $this->getPluralRuleTypes( $code );
742 foreach ( $this->getPluralFiles() as $fileName ) {
756 protected function mergeItem( $key, &$value, $fallbackValue ) {
757 if ( $value !==
null ) {
758 if ( $fallbackValue !==
null ) {
759 if ( in_array( $key, self::$mergeableMapKeys ) ) {
760 $value += $fallbackValue;
761 } elseif ( in_array( $key, self::$mergeableListKeys ) ) {
762 $value = array_unique( array_merge( $fallbackValue, $value ) );
763 } elseif ( in_array( $key, self::$mergeableAliasListKeys ) ) {
764 $value = array_merge_recursive( $value, $fallbackValue );
765 } elseif ( in_array( $key, self::$optionalMergeKeys ) ) {
766 if ( !empty( $value[
'inherit'] ) ) {
767 $value = array_merge( $fallbackValue, $value );
770 unset( $value[
'inherit'] );
771 } elseif ( in_array( $key, self::$magicWordKeys ) ) {
772 $this->mergeMagicWords( $value, $fallbackValue );
776 $value = $fallbackValue;
785 foreach ( $fallbackValue as $magicName => $fallbackInfo ) {
786 if ( !isset( $value[$magicName] ) ) {
787 $value[$magicName] = $fallbackInfo;
789 $oldSynonyms = array_slice( $fallbackInfo, 1 );
790 $newSynonyms = array_slice( $value[$magicName], 1 );
791 $synonyms = array_values( array_unique( array_merge(
792 $newSynonyms, $oldSynonyms ) ) );
793 $value[$magicName] = array_merge( [ $fallbackInfo[0] ], $synonyms );
813 foreach ( $codeSequence as $code ) {
814 if ( isset( $fallbackValue[$code] ) ) {
815 $this->mergeItem( $key, $value, $fallbackValue[$code] );
834 'core' =>
"$IP/languages/i18n",
835 'exif' =>
"$IP/languages/i18n/exif",
836 'api' =>
"$IP/includes/api/i18n",
837 'rest' =>
"$IP/includes/Rest/i18n",
838 'oojs-ui' =>
"$IP/resources/lib/ooui/i18n",
839 'paramvalidator' =>
"$IP/includes/libs/ParamValidator/i18n",
840 ] + $this->options->get(
'MessagesDirs' );
851 throw new MWException(
"Invalid language code requested" );
853 $this->recachedLangs[ $code ] =
true;
856 $initialData = array_fill_keys( self::$allKeys,
null );
857 $coreData = $initialData;
860 # Load the primary localisation from the source file
861 $data = $this->readSourceFilesAndRegisterDeps( $code, $deps );
862 $this->logger->debug( __METHOD__ .
": got localisation for $code from source" );
864 # Merge primary localisation
865 foreach ( $data as $key => $value ) {
866 $this->mergeItem( $key, $coreData[ $key ], $value );
869 # Fill in the fallback if it's not there already
871 if ( ( $coreData[
'fallback'] ===
null || $coreData[
'fallback'] ===
false ) && $code ===
'en' ) {
872 $coreData[
'fallback'] =
false;
873 $coreData[
'originalFallbackSequence'] = $coreData[
'fallbackSequence'] = [];
875 if ( $coreData[
'fallback'] !==
null ) {
876 $coreData[
'fallbackSequence'] = array_map(
'trim', explode(
',', $coreData[
'fallback'] ) );
878 $coreData[
'fallbackSequence'] = [];
880 $len = count( $coreData[
'fallbackSequence'] );
882 # Before we add the 'en' fallback for messages, keep a copy of
883 # the original fallback sequence
884 $coreData[
'originalFallbackSequence'] = $coreData[
'fallbackSequence'];
886 # Ensure that the sequence ends at 'en' for messages
887 if ( !$len || $coreData[
'fallbackSequence'][$len - 1] !==
'en' ) {
888 $coreData[
'fallbackSequence'][] =
'en';
892 $codeSequence = array_merge( [ $code ], $coreData[
'fallbackSequence'] );
893 $messageDirs = $this->getMessagesDirs();
895 # Load non-JSON localisation data for extensions
896 $extensionData = array_fill_keys( $codeSequence, $initialData );
897 foreach ( $this->options->get(
'ExtensionMessagesFiles' ) as $extension => $fileName ) {
898 if ( isset( $messageDirs[$extension] ) ) {
899 # This extension has JSON message data; skip the PHP shim
903 $data = $this->readPHPFile( $fileName,
'extension' );
906 foreach ( $data as $key => $item ) {
907 foreach ( $codeSequence as $csCode ) {
908 if ( isset( $item[$csCode] ) ) {
909 $this->mergeItem( $key, $extensionData[$csCode][$key], $item[$csCode] );
920 # Load the localisation data for each fallback, then merge it into the full array
921 $allData = $initialData;
922 foreach ( $codeSequence as $csCode ) {
923 $csData = $initialData;
925 # Load core messages and the extension localisations.
926 foreach ( $messageDirs as $dirs ) {
927 foreach ( (array)$dirs as $dir ) {
928 $fileName =
"$dir/$csCode.json";
929 $data = $this->readJSONFile( $fileName );
931 foreach ( $data as $key => $item ) {
932 $this->mergeItem( $key, $csData[$key], $item );
939 # Merge non-JSON extension data
940 if ( isset( $extensionData[$csCode] ) ) {
941 foreach ( $extensionData[$csCode] as $key => $item ) {
942 $this->mergeItem( $key, $csData[$key], $item );
946 if ( $csCode === $code ) {
947 # Merge core data into extension data
948 foreach ( $coreData as $key => $item ) {
949 $this->mergeItem( $key, $csData[$key], $item );
952 # Load the secondary localisation from the source file to
953 # avoid infinite cycles on cyclic fallbacks
954 $fbData = $this->readSourceFilesAndRegisterDeps( $csCode, $deps );
955 # Only merge the keys that make sense to merge
956 foreach ( self::$allKeys as $key ) {
957 if ( !isset( $fbData[ $key ] ) ) {
961 if ( ( $coreData[ $key ] ) ===
null || $this->isMergeableKey( $key ) ) {
962 $this->mergeItem( $key, $csData[ $key ], $fbData[ $key ] );
967 # Allow extensions an opportunity to adjust the data for this
969 $this->hookRunner->onLocalisationCacheRecacheFallback( $this, $csCode, $csData );
971 # Merge the data for this fallback into the final array
972 if ( $csCode === $code ) {
975 foreach ( self::$allKeys as $key ) {
976 if ( !isset( $csData[$key] ) ) {
981 if ( $allData[$key] ===
null || $this->isMergeableKey( $key ) ) {
982 $this->mergeItem( $key, $allData[$key], $csData[$key] );
988 # Add cache dependencies for any referenced globals
989 $deps[
'wgExtensionMessagesFiles'] =
new GlobalDependency(
'wgExtensionMessagesFiles' );
995 # Add dependencies to the cache entry
996 $allData[
'deps'] = $deps;
998 # Replace spaces with underscores in namespace names
999 $allData[
'namespaceNames'] = str_replace(
' ',
'_', $allData[
'namespaceNames'] );
1001 # And do the same for special page aliases. $page is an array.
1002 foreach ( $allData[
'specialPageAliases'] as &$page ) {
1003 $page = str_replace(
' ',
'_', $page );
1005 # Decouple the reference to prevent accidental damage
1008 # If there were no plural rules, return an empty array
1009 if ( $allData[
'pluralRules'] ===
null ) {
1010 $allData[
'pluralRules'] = [];
1012 if ( $allData[
'compiledPluralRules'] ===
null ) {
1013 $allData[
'compiledPluralRules'] = [];
1015 # If there were no plural rule types, return an empty array
1016 if ( $allData[
'pluralRuleTypes'] ===
null ) {
1017 $allData[
'pluralRuleTypes'] = [];
1021 $allData[
'list'] = [];
1022 foreach ( self::$splitKeys as $key ) {
1023 $allData[
'list'][$key] = array_keys( $allData[$key] );
1027 $this->hookRunner->onLocalisationCacheRecache( $this, $code, $allData, $unused );
1029 if ( $allData[
'namespaceNames'] ===
null ) {
1030 throw new MWException( __METHOD__ .
': Localisation data failed sanity check! ' .
1031 'Check that your languages/messages/MessagesEn.php file is intact.' );
1034 # Set the preload key
1035 $allData[
'preload'] = $this->buildPreload( $allData );
1037 # Save to the process cache and register the items loaded
1038 $this->data[$code] = $allData;
1039 foreach ( $allData as $key => $item ) {
1040 $this->loadedItems[$code][$key] =
true;
1043 # Save to the persistent cache
1044 $this->store->startWrite( $code );
1045 foreach ( $allData as $key => $value ) {
1046 if ( in_array( $key, self::$splitKeys ) ) {
1047 foreach ( $value as $subkey => $subvalue ) {
1048 $this->store->set(
"$key:$subkey", $subvalue );
1051 $this->store->set( $key, $value );
1054 $this->store->finishWrite();
1056 # Clear out the MessageBlobStore
1057 # HACK: If using a null (i.e. disabled) storage backend, we
1058 # can't write to the MessageBlobStore either
1060 foreach ( $this->clearStoreCallbacks as $callback ) {
1075 $preload = [
'messages' => [] ];
1076 foreach ( self::$preloadedKeys as $key ) {
1077 $preload[$key] = $data[$key];
1080 foreach ( $data[
'preloadedMessages'] as $subkey ) {
1081 $subitem = $data[
'messages'][$subkey] ??
null;
1082 $preload[
'messages'][$subkey] = $subitem;
1094 unset( $this->data[$code] );
1095 unset( $this->loadedItems[$code] );
1096 unset( $this->loadedSubitems[$code] );
1097 unset( $this->initialisedLangs[$code] );
1098 unset( $this->shallowFallbacks[$code] );
1100 foreach ( $this->shallowFallbacks as $shallowCode => $fbCode ) {
1101 if ( $fbCode === $code ) {
1102 $this->unload( $shallowCode );
1111 foreach ( $this->initialisedLangs as
$lang => $unused ) {
1112 $this->unload(
$lang );
1121 $this->manualRecache =
false;
isExpired()
Returns true if the dependency is expired, false otherwise.
Null store backend, used to avoid DB errors during install.
Class for caching the contents of localisation files, Messages*.php and *.i18n.php.
$manualRecache
True if recaching should only be done on an explicit call to recache().
static $magicWordKeys
Keys for items that are formatted like $magicWords.
buildPreload( $data)
Build the preload item from the given pre-cache data.
initLanguage( $code)
Initialise a language in this object.
static $mergeableMapKeys
Keys for items which consist of associative arrays, which may be merged by a fallback sequence.
readSourceFilesAndRegisterDeps( $code, &$deps)
Read the data from the source files for a given language, and register the relevant dependencies in t...
isMergeableKey( $key)
Returns true if the given key is mergeable, that is, if it is an associative array which can be merge...
loadPluralFile( $fileName)
Load a plural XML file with the given filename, compile the relevant rules, and save the compiled rul...
getPluralRules( $code)
Get the plural rules for a given language from the XML files.
$loadedItems
A 2-d associative array, code/key, where presence indicates that the item is loaded.
readPHPFile( $_fileName, $_fileType)
Read a PHP file containing localisation data.
$initialisedLangs
An array where presence of a key indicates that that language has been initialised.
readJSONFile( $fileName)
Read a JSON file containing localisation messages.
loadPluralFiles()
Load the plural XML files.
unload( $code)
Unload the data for a given language from the object cache.
unloadAll()
Unload all data.
disableBackend()
Disable the storage backend.
static $mergeableAliasListKeys
Keys for items which contain an array of arrays of equivalent aliases for each subitem.
getSubitemList( $code, $key)
Get the list of subitem keys for a given item.
$recachedLangs
An array where the keys are codes that have been recached by this instance.
$loadedSubitems
A 3-d associative array, code/key/subkey, where presence indicates that the subitem is loaded.
__construct(ServiceOptions $options, LCStore $store, LoggerInterface $logger, array $clearStoreCallbacks, LanguageNameUtils $langNameUtils, HookContainer $hookContainer)
For constructor parameters, see the documentation in DefaultSettings.php for $wgLocalisationCacheConf...
static $mergeableListKeys
Keys for items which are a numbered array.
recache( $code)
Load localisation data for a given language for both core and extensions and save it to the persisten...
isExpired( $code)
Returns true if the cache identified by $code is missing or expired.
mergeItem( $key, &$value, $fallbackValue)
Merge two localisation values, a primary and a fallback, overwriting the primary value in place.
getSubitem( $code, $key, $subkey)
Get a subitem, for instance a single message for a given language.
mergeMagicWords(&$value, $fallbackValue)
$shallowFallbacks
An array mapping non-existent pseudo-languages to fallback languages.
initShallowFallback( $primaryCode, $fallbackCode)
Create a fallback from one language to another, without creating a complete persistent cache.
static getStoreFromConf(array $conf, $fallbackCacheDir)
Return a suitable LCStore as specified by the given configuration.
LanguageNameUtils $langNameUtils
static $allKeys
All item keys.
getCompiledPluralRules( $code)
Get the compiled plural rules for a given language from the XML files.
static $optionalMergeKeys
Keys for items which contain an associative array, and may be merged if the primary value contains th...
getMessagesDirs()
Gets the combined list of messages dirs from core and extensions.
callable[] $clearStoreCallbacks
See comment for parameter in constructor.
static $preloadedKeys
Keys which are loaded automatically by initLanguage()
getItem( $code, $key)
Get a cache item.
static $splitKeys
Keys for items where the subitems are stored in the backend separately.
loadItem( $code, $key)
Load an item into the cache.
$pluralRuleTypes
Associative array of cached plural rule types.
getPluralRuleTypes( $code)
Get the plural rule types for a given language from the XML files.
LCStore $store
The persistent store object.
loadSubitem( $code, $key, $subkey)
Load a subitem into the cache.
$pluralRules
Associative array of cached plural rules.
mergeExtensionItem( $codeSequence, $key, &$value, $fallbackValue)
Given an array mapping language code to localisation value, such as is found in extension *....
Interface for the persistence layer of LocalisationCache.
if(!isset( $args[0])) $lang