Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
YamlFormat.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\FileFormatSupport;
5
13
22 private ArrayFlattener $flattener;
23
24 public function __construct( FileBasedMessageGroup $group ) {
25 parent::__construct( $group );
26
27 // Obtains object used to flatten and unflatten arrays. In this implementation
28 // we use the ArrayFlattener class which also supports CLDR pluralization rules.
29 $this->flattener = new ArrayFlattener(
30 $this->extra['nestingSeparator'] ?? '.',
31 $this->extra['parseCLDRPlurals'] ?? false
32 );
33 }
34
35 public function getFileExtensions(): array {
36 return [ '.yaml', '.yml' ];
37 }
38
40 public function readFromVariable( string $data ): array {
41 // Authors first.
42 $matches = [];
43 preg_match_all( '/^#\s*Author:\s*(.*)$/m', $data, $matches );
44 $authors = $matches[1];
45
46 // Then messages.
47 $messages = TranslateYaml::loadString( $data ) ?? [];
48
49 // Some groups have messages under language code
50 if ( isset( $this->extra['codeAsRoot'] ) ) {
51 $messages = array_shift( $messages ) ?? [];
52 }
53
54 $messages = $this->flattener->flatten( $messages );
55 $messages = $this->group->getMangler()->mangleArray( $messages );
56 foreach ( $messages as &$value ) {
57 $value = rtrim( $value, "\n" );
58 }
59
60 return [
61 'AUTHORS' => $authors,
62 'MESSAGES' => $messages,
63 ];
64 }
65
66 protected function writeReal( MessageCollection $collection ): string {
67 $output = $this->doHeader( $collection );
68 $output .= $this->doAuthors( $collection );
69
70 $mangler = $this->group->getMangler();
71
72 $messages = [];
73
75 foreach ( $collection as $key => $m ) {
76 $key = $mangler->unmangle( $key );
77 $value = $m->translation();
78 $value = str_replace( TRANSLATE_FUZZY, '', $value );
79
80 if ( $value === '' ) {
81 continue;
82 }
83
84 $messages[$key] = $value;
85 }
86
87 if ( !count( $messages ) ) {
88 return '';
89 }
90 $messages = $this->flattener->unflatten( $messages );
91
92 // Some groups have messages under language code.
93 if ( isset( $this->extra['codeAsRoot'] ) ) {
94 $code = $this->group->mapCode( $collection->code );
95 $messages = [ $code => $messages ];
96 }
97
98 $output .= TranslateYaml::dump( $messages );
99
100 return $output;
101 }
102
103 private function doHeader( MessageCollection $collection ): string {
104 global $wgSitename;
105 global $wgTranslateYamlLibrary;
106
107 $code = $collection->code;
108 $name = Utilities::getLanguageName( $code );
109 $native = Utilities::getLanguageName( $code, $code );
110 $output = "# Messages for $name ($native)\n";
111 $output .= "# Exported from $wgSitename\n";
112
113 if ( isset( $wgTranslateYamlLibrary ) ) {
114 $output .= "# Export driver: $wgTranslateYamlLibrary\n";
115 }
116
117 return $output;
118 }
119
120 private function doAuthors( MessageCollection $collection ): string {
121 $output = '';
122 $authors = $collection->getAuthors();
123 $authors = $this->filterAuthors( $authors, $collection->code );
124
125 foreach ( $authors as $author ) {
126 $output .= "# Author: $author\n";
127 }
128
129 return $output;
130 }
131
132 public function isContentEqual( ?string $a, ?string $b ): bool {
133 return $this->flattener->compareContent( $a, $b );
134 }
135
136 public static function getExtraSchema(): array {
137 return [
138 'root' => [
139 '_type' => 'array',
140 '_children' => [
141 'FILES' => [
142 '_type' => 'array',
143 '_children' => [
144 'codeAsRoot' => [
145 '_type' => 'boolean',
146 ],
147 'nestingSeparator' => [
148 '_type' => 'text',
149 ],
150 'parseCLDRPlurals' => [
151 '_type' => 'boolean',
152 ]
153 ]
154 ]
155 ]
156 ]
157 ];
158 }
159}
160
161class_alias( YamlFormat::class, 'YamlFFS' );
return[ 'Translate:ConfigHelper'=> static function():ConfigHelper { return new ConfigHelper();}, 'Translate:CsvTranslationImporter'=> static function(MediaWikiServices $services):CsvTranslationImporter { return new CsvTranslationImporter( $services->getWikiPageFactory());}, 'Translate:EntitySearch'=> static function(MediaWikiServices $services):EntitySearch { return new EntitySearch($services->getMainWANObjectCache(), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), MessageGroups::singleton(), $services->getNamespaceInfo(), $services->get( 'Translate:MessageIndex'), $services->getTitleParser(), $services->getTitleFormatter());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->getMainConfig(), $services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'));}, 'Translate:FileFormatFactory'=> static function(MediaWikiServices $services):FileFormatFactory { return new FileFormatFactory( $services->getObjectFactory());}, 'Translate:GroupSynchronizationCache'=> static function(MediaWikiServices $services):GroupSynchronizationCache { return new GroupSynchronizationCache( $services->get( 'Translate:PersistentCache'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'));}, 'Translate:MessageGroupReviewStore'=> static function(MediaWikiServices $services):MessageGroupReviewStore { return new MessageGroupReviewStore($services->getDBLoadBalancer(), $services->get( 'Translate:HookRunner'));}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getDBLoadBalancer(), $services->getLinkRenderer(), $services->get( 'Translate:MessageGroupReviewStore'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=$services->getMainConfig() ->get( 'TranslateMessageIndex');if(is_string( $params)) { $params=(array) $params;} $class=array_shift( $params);return new $class( $params);}, 'Translate:MessagePrefixStats'=> static function(MediaWikiServices $services):MessagePrefixStats { return new MessagePrefixStats( $services->getTitleParser());}, 'Translate:ParsingPlaceholderFactory'=> static function():ParsingPlaceholderFactory { return new ParsingPlaceholderFactory();}, 'Translate:PersistentCache'=> static function(MediaWikiServices $services):PersistentCache { return new PersistentDatabaseCache($services->getDBLoadBalancer(), $services->getJsonCodec());}, 'Translate:ProgressStatsTableFactory'=> static function(MediaWikiServices $services):ProgressStatsTableFactory { return new ProgressStatsTableFactory($services->getLinkRenderer(), $services->get( 'Translate:ConfigHelper'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore($services->getDBLoadBalancerFactory());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleExporter'=> static function(MediaWikiServices $services):TranslatableBundleExporter { return new TranslatableBundleExporter($services->get( 'Translate:SubpageListBuilder'), $services->getWikiExporterFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslatableBundleFactory'=> static function(MediaWikiServices $services):TranslatableBundleFactory { return new TranslatableBundleFactory($services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:MessageBundleStore'));}, 'Translate:TranslatableBundleImporter'=> static function(MediaWikiServices $services):TranslatableBundleImporter { return new TranslatableBundleImporter($services->getWikiImporterFactory(), $services->get( 'Translate:TranslatablePageParser'), $services->getRevisionLookup());}, 'Translate:TranslatableBundleMover'=> static function(MediaWikiServices $services):TranslatableBundleMover { return new TranslatableBundleMover($services->getMovePageFactory(), $services->getJobQueueGroup(), $services->getLinkBatchFactory(), $services->get( 'Translate:TranslatableBundleFactory'), $services->get( 'Translate:SubpageListBuilder'), $services->getMainConfig() ->get( 'TranslatePageMoveLimit'));}, 'Translate:TranslatableBundleStatusStore'=> static function(MediaWikiServices $services):TranslatableBundleStatusStore { return new TranslatableBundleStatusStore($services->getDBLoadBalancer() ->getConnection(DB_PRIMARY), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), $services->getDBLoadBalancer() ->getMaintenanceConnectionRef(DB_PRIMARY));}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStore'=> static function(MediaWikiServices $services):TranslatablePageStore { return new TranslatablePageStore($services->get( 'Translate:MessageIndex'), $services->getJobQueueGroup(), $services->get( 'Translate:RevTagStore'), $services->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'),);}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { $db=$services->getDBLoadBalancer() ->getConnection(DB_REPLICA);return new TranslationStashStorage( $db);}, 'Translate:TranslationStatsDataProvider'=> static function(MediaWikiServices $services):TranslationStatsDataProvider { return new TranslationStatsDataProvider(new ServiceOptions(TranslationStatsDataProvider::CONSTRUCTOR_OPTIONS, $services->getMainConfig()), $services->getObjectFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslationUnitStoreFactory'=> static function(MediaWikiServices $services):TranslationUnitStoreFactory { return new TranslationUnitStoreFactory( $services->getDBLoadBalancer());}, 'Translate:TranslatorActivity'=> static function(MediaWikiServices $services):TranslatorActivity { $query=new TranslatorActivityQuery($services->getMainConfig(), $services->getDBLoadBalancer());return new TranslatorActivity($services->getMainObjectStash(), $query, $services->getJobQueueGroup());}, 'Translate:TtmServerFactory'=> static function(MediaWikiServices $services):TtmServerFactory { $config=$services->getMainConfig();$default=$config->get( 'TranslateTranslationDefaultService');if( $default===false) { $default=null;} return new TtmServerFactory( $config->get( 'TranslateTranslationServices'), $default);}]
@phpcs-require-sorted-array
This class implements default behavior for file based message groups.
A very basic FileFormatSupport module that implements some basic functionality and a simple binary ba...
Implements support for message storage in YAML format.
getFileExtensions()
Return the commonly used file extensions for these formats.
static getExtraSchema()
Return a data structure that will be merged with the base schema.
isContentEqual(?string $a, ?string $b)
Checks whether two strings are equal.
This file contains the class for core message collections implementation.
Interface for message objects used by MessageCollection.
Definition Message.php:13
Flattens message arrays for further processing.
Essentially random collection of helper functions, similar to GlobalFunctions.php.
Definition Utilities.php:31
This class is a wrapper class to provide interface to parse and generate YAML files with phpyaml or s...
static loadString( $text)
Message groups are usually configured in YAML, though the actual storage format does not matter,...