Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
GettextFormat.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\FileFormatSupport;
5
6use InvalidArgumentException;
14use MediaWiki\Language\LanguageCode;
15use MediaWiki\Logger\LoggerFactory;
16use MediaWiki\MediaWikiServices;
17use MediaWiki\Specials\SpecialVersion;
18use MediaWiki\Title\Title;
19use UtfNormal\Validator;
20
31 private bool $allowPotMode = false;
32 private bool $offlineMode = false;
33
34 public function supportsFuzzy(): string {
35 return 'yes';
36 }
37
38 public function getFileExtensions(): array {
39 return [ '.pot', '.po' ];
40 }
41
42 public function setOfflineMode( bool $value ): void {
43 $this->offlineMode = $value;
44 }
45
47 public function read( $languageCode ) {
48 // This is somewhat hacky, but pot mode should only ever be used for the source language.
49 // See https://phabricator.wikimedia.org/T230361
50 $this->allowPotMode = $this->getGroup()->getSourceLanguage() === $languageCode;
51
52 try {
53 return parent::read( $languageCode );
54 } finally {
55 $this->allowPotMode = false;
56 }
57 }
58
60 public function readFromVariable( string $data ): array {
61 if ( !mb_check_encoding( $data, 'UTF-8' ) ) {
62 throw new GettextParseException( 'invalid-utf8' );
63 }
64
65 $data = Validator::cleanUp( $data );
66
67 # Authors first
68 $matches = [];
69 preg_match_all( '/^#\s*Author:\s*(.*)$/m', $data, $matches );
70 $authors = $matches[1];
71
72 # Then messages and everything else
73 $parsedData = $this->parseGettext( $data );
74 $parsedData['AUTHORS'] = $authors;
75
76 foreach ( $parsedData['MESSAGES'] as $key => $value ) {
77 if ( $value === '' ) {
78 unset( $parsedData['MESSAGES'][$key] );
79 }
80 }
81
82 return $parsedData;
83 }
84
86 private function parseGettext( string $data ): array {
87 $mangler = $this->group->getMangler();
88 $useCtxtAsKey = $this->extra['CtxtAsKey'] ?? false;
89 $keyAlgorithm = 'simple';
90 if ( isset( $this->extra['keyAlgorithm'] ) ) {
91 $keyAlgorithm = $this->extra['keyAlgorithm'];
92 }
93
94 $potmode = false;
95
96 // Normalise newlines, to make processing easier
97 $data = str_replace( "\r\n", "\n", $data );
98
99 /* Delimit the file into sections, which are separated by two newlines.
100 * We are permissive and accept more than two. This parsing method isn't
101 * efficient wrt memory, but was easy to implement */
102 $sections = preg_split( '/\n{2,}/', $data );
103
104 /* First one isn't an actual message. We'll handle it specially below */
105 $headerSection = array_shift( $sections );
106 /* Since this is the header section, we are only interested in the tags
107 * and msgid is empty. Somewhere we should extract the header comments
108 * too */
109 $match = $this->expectKeyword( 'msgstr', $headerSection );
110 if ( $match !== null ) {
111 $headerBlock = $this->formatForWiki( $match, 'trim' );
112 $headers = $this->parseHeaderTags( $headerBlock );
113
114 // Check for pot-mode by checking if the header is fuzzy
115 $flags = $this->parseFlags( $headerSection );
116 if ( in_array( 'fuzzy', $flags, true ) ) {
117 $potmode = $this->allowPotMode;
118 }
119 } else {
120 $message = "Gettext file header was not found:\n\n$headerSection";
121 throw new GettextParseException( $message );
122 }
123
124 $template = [];
125 $messages = [];
126
127 // Extract some metadata from headers for easier use
128 $metadata = [];
129 if ( isset( $headers['X-Language-Code'] ) ) {
130 $metadata['code'] = $headers['X-Language-Code'];
131 }
132
133 if ( isset( $headers['X-Message-Group'] ) ) {
134 $metadata['group'] = $headers['X-Message-Group'];
135 }
136
137 /* At this stage we are only interested how many plurals forms we should
138 * be expecting when parsing the rest of this file. */
139 $pluralCount = null;
140 if ( $potmode ) {
141 $pluralCount = 2;
142 } elseif ( isset( $headers['Plural-Forms'] ) ) {
143 $pluralCount = $metadata['plural'] = GettextPlural::getPluralCount( $headers['Plural-Forms'] );
144 }
145
146 $metadata['plural'] = $pluralCount;
147
148 // Then parse the messages
149 foreach ( $sections as $section ) {
150 $item = $this->parseGettextSection( $section, $pluralCount );
151 if ( $item === null ) {
152 continue;
153 }
154
155 if ( $useCtxtAsKey ) {
156 if ( $item['ctxt'] === false ) {
157 error_log( "ctxt missing for: $section" );
158 continue;
159 }
160 $key = $item['ctxt'];
161 } else {
162 $key = $this->generateKeyFromItem( $item, $keyAlgorithm );
163 }
164
165 $key = $mangler->mangle( $key );
166 $messages[$key] = $potmode ? $item['id'] : $item['str'];
167 $template[$key] = $item;
168 }
169
170 return [
171 'MESSAGES' => $messages,
172 'EXTRA' => [
173 'TEMPLATE' => $template,
174 'METADATA' => $metadata,
175 'HEADERS' => $headers,
176 ],
177 ];
178 }
179
181 private function parseGettextSection( string $section, ?int $pluralCount ): ?array {
182 if ( trim( $section ) === '' ) {
183 return null;
184 }
185
186 /* These inactive sections are of no interest to us. Multiline mode
187 * is needed because there may be flags or other annoying stuff
188 * before the commented out sections.
189 */
190 if ( preg_match( '/^#~/m', $section ) ) {
191 return null;
192 }
193
194 $item = [
195 'ctxt' => false,
196 'id' => '',
197 'str' => '',
198 'flags' => [],
199 'comments' => [],
200 ];
201
202 $match = $this->expectKeyword( 'msgid', $section );
203 if ( $match !== null ) {
204 $item['id'] = $this->formatForWiki( $match );
205 } else {
206 throw new GettextParseException( "Unable to parse msgid:\n\n$section" );
207 }
208
209 $match = $this->expectKeyword( 'msgctxt', $section );
210 if ( $match !== null ) {
211 $item['ctxt'] = $this->formatForWiki( $match );
212 }
213
214 $pluralMessage = false;
215 $match = $this->expectKeyword( 'msgid_plural', $section );
216 if ( $match !== null ) {
217 $pluralMessage = true;
218 $plural = $this->formatForWiki( $match );
219 $item['id'] = GettextPlural::flatten( [ $item['id'], $plural ] );
220 }
221
222 if ( $pluralMessage ) {
223 $pluralMessageText = $this->processGettextPluralMessage( $pluralCount, $section );
224
225 // Keep the translation empty if no form has translation
226 if ( $pluralMessageText !== '' ) {
227 $item['str'] = $pluralMessageText;
228 }
229 } else {
230 $match = $this->expectKeyword( 'msgstr', $section );
231 if ( $match !== null ) {
232 $item['str'] = $this->formatForWiki( $match );
233 } else {
234 throw new GettextParseException( "Unable to parse msgstr:\n\n$section" );
235 }
236 }
237
238 // Parse flags
239 $flags = $this->parseFlags( $section );
240 foreach ( $flags as $key => $flag ) {
241 if ( $flag === 'fuzzy' ) {
242 $item['str'] = TRANSLATE_FUZZY . $item['str'];
243 unset( $flags[$key] );
244 }
245 }
246 $item['flags'] = $flags;
247
248 // Rest of the comments
249 $matches = [];
250 if ( preg_match_all( '/^#(.?) (.*)$/m', $section, $matches, PREG_SET_ORDER ) ) {
251 foreach ( $matches as $match ) {
252 if ( $match[1] !== ',' && !str_starts_with( $match[1], '[Wiki]' ) ) {
253 $item['comments'][$match[1]][] = $match[2];
254 }
255 }
256 }
257
258 return $item;
259 }
260
261 private function processGettextPluralMessage( ?int $pluralCount, string $section ): string {
262 $actualForms = [];
263
264 for ( $i = 0; $i < $pluralCount; $i++ ) {
265 $match = $this->expectKeyword( "msgstr\\[$i\\]", $section );
266
267 if ( $match !== null ) {
268 $actualForms[] = $this->formatForWiki( $match );
269 } else {
270 $actualForms[] = '';
271 error_log( "Plural $i not found, expecting total of $pluralCount for $section" );
272 }
273 }
274
275 if ( array_sum( array_map( 'strlen', $actualForms ) ) > 0 ) {
276 return GettextPlural::flatten( $actualForms );
277 } else {
278 return '';
279 }
280 }
281
282 private function parseFlags( string $section ): array {
283 $matches = [];
284 if ( preg_match( '/^#,(.*)$/mu', $section, $matches ) ) {
285 return array_map( 'trim', explode( ',', $matches[1] ) );
286 } else {
287 return [];
288 }
289 }
290
291 private function expectKeyword( string $name, string $section ): ?string {
292 /* Catches the multiline textblock that comes after keywords msgid,
293 * msgstr, msgid_plural, msgctxt.
294 */
295 $poformat = '".*"\n?(^".*"$\n?)*';
296
297 $matches = [];
298 if ( preg_match( "/^$name\s($poformat)/mx", $section, $matches ) ) {
299 return $matches[1];
300 } else {
301 return null;
302 }
303 }
304
311 public function generateKeyFromItem( array $item, string $algorithm = 'simple' ): string {
312 $lang = MediaWikiServices::getInstance()->getLanguageFactory()->getLanguage( 'en' );
313
314 if ( $item['ctxt'] === '' ) {
315 /* Messages with msgctxt as empty string should be different
316 * from messages without any msgctxt. To avoid BC break make
317 * the empty ctxt a special case */
318 $hash = sha1( $item['id'] . 'MSGEMPTYCTXT' );
319 } else {
320 $hash = sha1( $item['ctxt'] . $item['id'] );
321 }
322
323 if ( $algorithm === 'simple' ) {
324 $hash = substr( $hash, 0, 6 );
325 $snippet = $lang->truncateForDatabase( $item['id'], 30, '' );
326 $snippet = str_replace( ' ', '_', trim( $snippet ) );
327 } else { // legacy
328 $legalChars = Title::legalChars();
329 $snippet = $item['id'];
330 $snippet = preg_replace( "/[^$legalChars]/", ' ', $snippet );
331 $snippet = preg_replace( '/[:&%\/_]/', ' ', $snippet );
332 $snippet = preg_replace( '/ {2,}/', ' ', $snippet );
333 $snippet = $lang->truncateForDatabase( $snippet, 30, '' );
334 $snippet = str_replace( ' ', '_', trim( $snippet ) );
335 }
336
337 return "$hash-$snippet";
338 }
339
343 private function processData( string $data ): string {
344 $quotePattern = '/(^"|"$\n?)/m';
345 $data = preg_replace( $quotePattern, '', $data );
346 return stripcslashes( $data );
347 }
348
353 private function handleWhitespace( string $data, string $whitespace ): string {
354 if ( preg_match( '/\s$/', $data ) ) {
355 if ( $whitespace === 'mark' ) {
356 $data .= '\\';
357 } elseif ( $whitespace === 'trim' ) {
358 $data = rtrim( $data );
359 } else {
360 // This condition will never happen as long as $whitespace is 'mark' or 'trim'
361 throw new InvalidArgumentException( "Unknown action for whitespace: $whitespace" );
362 }
363 }
364
365 return $data;
366 }
367
374 private function formatForWiki( string $data, string $whitespace = 'mark' ): string {
375 $data = $this->processData( $data );
376 return $this->handleWhitespace( $data, $whitespace );
377 }
378
379 private function parseHeaderTags( string $headers ): array {
380 $tags = [];
381 foreach ( explode( "\n", $headers ) as $line ) {
382 if ( !str_contains( $line, ':' ) ) {
383 error_log( __METHOD__ . ": $line" );
384 }
385 [ $key, $value ] = explode( ':', $line, 2 );
386 $tags[trim( $key )] = trim( $value );
387 }
388
389 return $tags;
390 }
391
392 protected function writeReal( MessageCollection $collection ): string {
393 // FIXME: this should be the source language
394 $pot = $this->read( 'en' ) ?? [];
395 $code = $collection->code;
396 $template = $this->read( $code ) ?? [];
397 $output = $this->doGettextHeader( $collection, $template['EXTRA'] ?? [] );
398
399 $pluralRule = GettextPlural::getPluralRule( $code );
400 if ( !$pluralRule ) {
401 $pluralRule = GettextPlural::getPluralRule( 'en' );
402 LoggerFactory::getInstance( LogNames::MAIN )->warning(
403 "T235180: Missing Gettext plural rule for '{languagecode}'",
404 [ 'languagecode' => $code ]
405 );
406 }
407 $pluralCount = GettextPlural::getPluralCount( $pluralRule );
408
409 $documentationLanguageCode = MediaWikiServices::getInstance()
410 ->getMainConfig()
411 ->get( 'TranslateDocumentationLanguageCode' );
412 $documentationCollection = null;
413 if ( is_string( $documentationLanguageCode ) ) {
414 $documentationCollection = clone $collection;
415 $documentationCollection->resetForNewLanguage( $documentationLanguageCode );
416 $documentationCollection->loadTranslations();
417 }
418
420 foreach ( $collection as $key => $m ) {
421 $transTemplate = $template['EXTRA']['TEMPLATE'][$key] ?? [];
422 $potTemplate = $pot['EXTRA']['TEMPLATE'][$key] ?? [];
423 $documentation = isset( $documentationCollection[$key] ) ?
424 $documentationCollection[$key]->translation() : null;
425
426 $output .= $this->formatMessageBlock(
427 $key,
428 $m,
429 $transTemplate,
430 $potTemplate,
431 $pluralCount,
432 $documentation
433 );
434 }
435
436 return $output;
437 }
438
439 private function doGettextHeader( MessageCollection $collection, array $template ): string {
440 global $wgSitename;
441
442 $code = $collection->code;
443 $name = Utilities::getLanguageName( $code );
444 $native = Utilities::getLanguageName( $code, $code );
445 $authors = $this->doAuthors( $collection );
446 if ( isset( $this->extra['header'] ) ) {
447 $extra = "# --\n" . $this->extra['header'];
448 } else {
449 $extra = '';
450 }
451
452 $group = $this->getGroup();
453 $output =
454 <<<EOT
455 # Translation of {$group->getLabel()} to $name ($native)
456 # Exported from $wgSitename
457 #
458 $authors$extra
459 EOT;
460
461 // Make sure there is no empty line before msgid
462 $output = trim( $output ) . "\n";
463
464 $specs = $template['HEADERS'] ?? [];
465
466 $timestamp = wfTimestampNow();
467 $specs['PO-Revision-Date'] = $this->formatTime( $timestamp );
468 if ( $this->offlineMode ) {
469 $specs['POT-Creation-Date'] = $this->formatTime( $timestamp );
470 } else {
471 $specs['X-POT-Import-Date'] = $this->formatTime( wfTimestamp( TS_MW, $this->getPotTime() ) );
472 }
473 $specs['Content-Type'] = 'text/plain; charset=UTF-8';
474 $specs['Content-Transfer-Encoding'] = '8bit';
475
476 $specs['Language'] = LanguageCode::bcp47( $this->group->mapCode( $code ) );
477
478 Services::getInstance()->getHookRunner()->onTranslate_GettextFormat_headerFields(
479 $specs,
480 $this->group,
481 $code
482 );
483
484 $specs['X-Generator'] = 'MediaWiki '
485 . SpecialVersion::getVersion()
486 . '; Translate '
487 . Utilities::getVersion();
488
489 if ( $this->offlineMode ) {
490 $specs['X-Language-Code'] = $code;
491 $specs['X-Message-Group'] = $group->getId();
492 }
493
494 $specs['Plural-Forms'] = GettextPlural::getPluralRule( $code )
495 ?: GettextPlural::getPluralRule( 'en' );
496
497 $output .= 'msgid ""' . "\n";
498 $output .= 'msgstr ""' . "\n";
499 $output .= '""' . "\n";
500
501 foreach ( $specs as $k => $v ) {
502 $output .= $this->escape( "$k: $v\n" ) . "\n";
503 }
504
505 $output .= "\n";
506
507 return $output;
508 }
509
510 private function doAuthors( MessageCollection $collection ): string {
511 $output = '';
512 $authors = $collection->getAuthors();
513 $authors = $this->filterAuthors( $authors, $collection->code );
514
515 foreach ( $authors as $author ) {
516 $output .= "# Author: $author\n";
517 }
518
519 return $output;
520 }
521
522 private function formatMessageBlock(
523 string $key,
524 Message $message,
525 array $trans,
526 array $pot,
527 int $pluralCount,
528 ?string $documentation
529 ): string {
530 $header = $this->formatDocumentation( $documentation );
531 $content = '';
532
533 $comments = $pot['comments'] ?? $trans['comments'] ?? [];
534 foreach ( $comments as $type => $typecomments ) {
535 foreach ( $typecomments as $comment ) {
536 $header .= "#$type $comment\n";
537 }
538 }
539
540 $flags = $pot['flags'] ?? $trans['flags'] ?? [];
541 $flags = array_merge( $message->getTags(), $flags );
542
543 if ( $this->offlineMode ) {
544 $content .= 'msgctxt ' . $this->escape( $key ) . "\n";
545 } else {
546 $ctxt = $pot['ctxt'] ?? $trans['ctxt'] ?? false;
547 if ( $ctxt !== false ) {
548 $content .= 'msgctxt ' . $this->escape( $ctxt ) . "\n";
549 }
550 }
551
552 $msgid = $message->definition();
553 $msgstr = $message->translation() ?? '';
554 if ( str_contains( $msgstr, TRANSLATE_FUZZY ) ) {
555 $msgstr = str_replace( TRANSLATE_FUZZY, '', $msgstr );
556 // Might be fuzzy infile
557 $flags[] = 'fuzzy';
558 }
559
560 if ( GettextPlural::hasPlural( $msgid ) ) {
561 $forms = GettextPlural::unflatten( $msgid, 2 );
562 $content .= 'msgid ' . $this->escape( $forms[0] ) . "\n";
563 $content .= 'msgid_plural ' . $this->escape( $forms[1] ) . "\n";
564
565 try {
566 $forms = GettextPlural::unflatten( $msgstr, $pluralCount );
567 foreach ( $forms as $index => $form ) {
568 $content .= "msgstr[$index] " . $this->escape( $form ) . "\n";
569 }
570 } catch ( GettextPluralException ) {
571 $flags[] = 'invalid-plural';
572 for ( $i = 0; $i < $pluralCount; $i++ ) {
573 $content .= "msgstr[$i] \"\"\n";
574 }
575 }
576 } else {
577 $content .= 'msgid ' . $this->escape( $msgid ) . "\n";
578 $content .= 'msgstr ' . $this->escape( $msgstr ) . "\n";
579 }
580
581 if ( $flags ) {
582 sort( $flags );
583 $header .= '#, ' . implode( ', ', array_unique( $flags ) ) . "\n";
584 }
585
586 $output = $header ?: "#\n";
587 $output .= $content . "\n";
588
589 return $output;
590 }
591
592 private function formatTime( string $time ): string {
593 $lang = MediaWikiServices::getInstance()->getLanguageFactory()->getLanguage( 'en' );
594
595 return $lang->sprintfDate( 'xnY-xnm-xnd xnH:xni:xns+0000', $time );
596 }
597
598 private function getPotTime(): string {
599 $cache = $this->group->getMessageGroupCache( $this->group->getSourceLanguage() );
600
601 return $cache->exists() ? $cache->getTimestamp() : wfTimestampNow();
602 }
603
604 private function formatDocumentation( ?string $documentation ): string {
605 if ( !is_string( $documentation ) ) {
606 return '';
607 }
608
609 if ( !$this->offlineMode ) {
610 return '';
611 }
612
613 $lines = explode( "\n", $documentation );
614 $out = '';
615 foreach ( $lines as $line ) {
616 $out .= "#. [Wiki] $line\n";
617 }
618
619 return $out;
620 }
621
622 private function escape( string $line ): string {
623 // There may be \ as a last character, for keeping trailing whitespace
624 $line = preg_replace( '/(\s)\\\\$/', '\1', $line );
625 $line = addcslashes( $line, '\\"' );
626 $line = str_replace( "\n", '\n', $line );
627 return '"' . $line . '"';
628 }
629
630 public function shouldOverwrite( string $a, string $b ): bool {
631 $regex = '/^"(.+)-Date: \d\d\d\d-\d\d-\d\d \d\d:\d\d:\d\d\+\d\d\d\d\\\\n"$/m';
632
633 $a = preg_replace( $regex, '', $a );
634 $b = preg_replace( $regex, '', $b );
635
636 return $a !== $b;
637 }
638
639 public static function getExtraSchema(): array {
640 return [
641 'root' => [
642 '_type' => 'array',
643 '_children' => [
644 'FILES' => [
645 '_type' => 'array',
646 '_children' => [
647 'header' => [
648 '_type' => 'text',
649 ],
650 'keyAlgorithm' => [
651 '_type' => 'enum',
652 '_values' => [ 'simple', 'legacy' ],
653 ],
654 'CtxtAsKey' => [
655 '_type' => 'boolean',
656 ],
657 ]
658 ]
659 ]
660 ]
661 ];
662 }
663
664 public function isContentEqual( ?string $a, ?string $b ): bool {
665 if ( $a === $b ) {
666 return true;
667 }
668
669 if ( $a === null || $b === null ) {
670 return false;
671 }
672
673 try {
674 $parsedA = GettextPlural::parsePluralForms( $a );
675 $parsedB = GettextPlural::parsePluralForms( $b );
676
677 // if they have the different number of plural forms, just fail
678 if ( count( $parsedA[1] ) !== count( $parsedB[1] ) ) {
679 return false;
680 }
681
682 } catch ( GettextPluralException ) {
683 // Something failed, invalid syntax?
684 return false;
685 }
686
687 $expectedPluralCount = count( $parsedA[1] );
688
689 // GettextPlural::unflatten() will return an empty array when $expectedPluralCount is 0
690 // So if they do not have translations and are different strings, they are not equal
691 if ( $expectedPluralCount === 0 ) {
692 return false;
693 }
694
695 return GettextPlural::unflatten( $a, $expectedPluralCount )
696 === GettextPlural::unflatten( $b, $expectedPluralCount );
697 }
698}
699
700class_alias( GettextFormat::class, 'GettextFFS' );
return[ 'Translate:AggregateGroupManager'=> static function(MediaWikiServices $services):AggregateGroupManager { return new AggregateGroupManager($services->getTitleFactory(), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:AggregateGroupMessageGroupFactory'=> static function(MediaWikiServices $services):AggregateGroupMessageGroupFactory { return new AggregateGroupMessageGroupFactory($services->get( 'Translate:MessageGroupMetadata'));}, '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:ExternalMessageSourceStateComparator'=> static function(MediaWikiServices $services):ExternalMessageSourceStateComparator { return new ExternalMessageSourceStateComparator(new SimpleStringComparator(), $services->getRevisionLookup(), $services->getPageStore());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance(LogNames::GROUP_SYNCHRONIZATION), $services->get( 'Translate:MessageIndex'), $services->getTitleFactory(), $services->get( 'Translate:MessageGroupSubscription'), new ServiceOptions(ExternalMessageSourceStateImporter::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:FileBasedMessageGroupFactory'=> static function(MediaWikiServices $services):FileBasedMessageGroupFactory { return new FileBasedMessageGroupFactory(new MessageGroupConfigurationParser(), $services->getContentLanguageCode() ->toString(), new ServiceOptions(FileBasedMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, '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:HookDefinedMessageGroupFactory'=> static function(MediaWikiServices $services):HookDefinedMessageGroupFactory { return new HookDefinedMessageGroupFactory( $services->get( 'Translate:HookRunner'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleDependencyPurger'=> static function(MediaWikiServices $services):MessageBundleDependencyPurger { return new MessageBundleDependencyPurger( $services->get( 'Translate:TranslatableBundleFactory'));}, 'Translate:MessageBundleMessageGroupFactory'=> static function(MediaWikiServices $services):MessageBundleMessageGroupFactory { return new MessageBundleMessageGroupFactory($services->get( 'Translate:MessageGroupMetadata'), new ServiceOptions(MessageBundleMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:MessageBundleTranslationLoader'=> static function(MediaWikiServices $services):MessageBundleTranslationLoader { return new MessageBundleTranslationLoader( $services->getLanguageFallback());}, 'Translate:MessageGroupMetadata'=> static function(MediaWikiServices $services):MessageGroupMetadata { return new MessageGroupMetadata( $services->getConnectionProvider());}, 'Translate:MessageGroupReviewStore'=> static function(MediaWikiServices $services):MessageGroupReviewStore { return new MessageGroupReviewStore($services->getConnectionProvider(), $services->get( 'Translate:HookRunner'));}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getLinkRenderer(), $services->get( 'Translate:MessageGroupReviewStore'), $services->get( 'Translate:MessageGroupMetadata'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageGroupSubscription'=> static function(MediaWikiServices $services):MessageGroupSubscription { return new MessageGroupSubscription($services->get( 'Translate:MessageGroupSubscriptionStore'), $services->getJobQueueGroup(), $services->getUserIdentityLookup(), LoggerFactory::getInstance(LogNames::GROUP_SUBSCRIPTION), new ServiceOptions(MessageGroupSubscription::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:MessageGroupSubscriptionHookHandler'=> static function(MediaWikiServices $services):?MessageGroupSubscriptionHookHandler { if(! $services->getExtensionRegistry() ->isLoaded( 'Echo')) { return null;} return new MessageGroupSubscriptionHookHandler($services->get( 'Translate:MessageGroupSubscription'), $services->getUserFactory());}, 'Translate:MessageGroupSubscriptionStore'=> static function(MediaWikiServices $services):MessageGroupSubscriptionStore { return new MessageGroupSubscriptionStore( $services->getConnectionProvider());}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=(array) $services->getMainConfig() ->get( 'TranslateMessageIndex');$class=array_shift( $params);$implementationMap=['HashMessageIndex'=> HashMessageIndex::class, 'CDBMessageIndex'=> CDBMessageIndex::class, 'DatabaseMessageIndex'=> DatabaseMessageIndex::class, 'hash'=> HashMessageIndex::class, 'cdb'=> CDBMessageIndex::class, 'database'=> DatabaseMessageIndex::class,];$messageIndexStoreClass=$implementationMap[$class] ?? $implementationMap['database'];return new MessageIndex(new $messageIndexStoreClass, $services->getMainWANObjectCache(), $services->getJobQueueGroup(), $services->get( 'Translate:HookRunner'), LoggerFactory::getInstance(LogNames::MAIN), $services->getMainObjectStash(), $services->getConnectionProvider(), new ServiceOptions(MessageIndex::SERVICE_OPTIONS, $services->getMainConfig()),);}, '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->getConnectionProvider(), $services->getJsonCodec());}, 'Translate:ProgressStatsTableFactory'=> static function(MediaWikiServices $services):ProgressStatsTableFactory { return new ProgressStatsTableFactory($services->getLinkRenderer(), $services->get( 'Translate:ConfigHelper'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore( $services->getConnectionProvider());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleDeleter'=> static function(MediaWikiServices $services):TranslatableBundleDeleter { return new TranslatableBundleDeleter($services->getMainObjectStash(), $services->getJobQueueGroup(), $services->get( 'Translate:SubpageListBuilder'), $services->get( 'Translate:TranslatableBundleFactory'));}, 'Translate:TranslatableBundleExporter'=> static function(MediaWikiServices $services):TranslatableBundleExporter { return new TranslatableBundleExporter($services->get( 'Translate:SubpageListBuilder'), $services->getWikiExporterFactory(), $services->getConnectionProvider());}, '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(), $services->getNamespaceInfo(), $services->getTitleFactory(), $services->getFormatterFactory());}, '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->getConnectionProvider(), $services->getObjectCacheFactory(), $services->getMainConfig() ->get( 'TranslatePageMoveLimit'));}, 'Translate:TranslatableBundleStatusStore'=> static function(MediaWikiServices $services):TranslatableBundleStatusStore { return new TranslatableBundleStatusStore($services->getConnectionProvider() ->getPrimaryDatabase(), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), $services->getDBLoadBalancer() ->getMaintenanceConnectionRef(DB_PRIMARY));}, 'Translate:TranslatablePageMarker'=> static function(MediaWikiServices $services):TranslatablePageMarker { return new TranslatablePageMarker($services->getConnectionProvider(), $services->getJobQueueGroup(), $services->getLinkRenderer(), MessageGroups::singleton(), $services->get( 'Translate:MessageIndex'), $services->getTitleFormatter(), $services->getTitleParser(), $services->get( 'Translate:TranslatablePageParser'), $services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:TranslatablePageStateStore'), $services->get( 'Translate:TranslationUnitStoreFactory'), $services->get( 'Translate:MessageGroupMetadata'), $services->getWikiPageFactory(), $services->get( 'Translate:TranslatablePageView'), $services->get( 'Translate:MessageGroupSubscription'), $services->getFormatterFactory(), $services->get( 'Translate:HookRunner'),);}, 'Translate:TranslatablePageMessageGroupFactory'=> static function(MediaWikiServices $services):TranslatablePageMessageGroupFactory { return new TranslatablePageMessageGroupFactory(new ServiceOptions(TranslatablePageMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStateStore'=> static function(MediaWikiServices $services):TranslatablePageStateStore { return new TranslatablePageStateStore($services->get( 'Translate:PersistentCache'), $services->getPageStore());}, 'Translate:TranslatablePageStore'=> static function(MediaWikiServices $services):TranslatablePageStore { return new TranslatablePageStore($services->get( 'Translate:MessageIndex'), $services->getJobQueueGroup(), $services->get( 'Translate:RevTagStore'), $services->getConnectionProvider(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:TranslatablePageView'=> static function(MediaWikiServices $services):TranslatablePageView { return new TranslatablePageView($services->getConnectionProvider(), $services->get( 'Translate:TranslatablePageStateStore'), new ServiceOptions(TranslatablePageView::SERVICE_OPTIONS, $services->getMainConfig()));}, 'Translate:TranslateSandbox'=> static function(MediaWikiServices $services):TranslateSandbox { return new TranslateSandbox($services->getUserFactory(), $services->getConnectionProvider(), $services->getPermissionManager(), $services->getAuthManager(), $services->getUserGroupManager(), $services->getActorStore(), $services->getUserOptionsManager(), $services->getJobQueueGroup(), $services->get( 'Translate:HookRunner'), new ServiceOptions(TranslateSandbox::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { return new TranslationStashStorage( $services->getConnectionProvider() ->getPrimaryDatabase());}, 'Translate:TranslationStatsDataProvider'=> static function(MediaWikiServices $services):TranslationStatsDataProvider { return new TranslationStatsDataProvider(new ServiceOptions(TranslationStatsDataProvider::CONSTRUCTOR_OPTIONS, $services->getMainConfig()), $services->getObjectFactory(), $services->getConnectionProvider());}, '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->getConnectionProvider());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);}, 'Translate:WorkflowStatesMessageGroupLoader'=> static function(MediaWikiServices $services):WorkflowStatesMessageGroupLoader { return new WorkflowStatesMessageGroupLoader(new ServiceOptions(WorkflowStatesMessageGroupLoader::CONSTRUCTOR_OPTIONS, $services->getMainConfig()),);},]
@phpcs-require-sorted-array
FileFormat class that implements support for gettext file format.
generateKeyFromItem(array $item, string $algorithm='simple')
Generates unique key for each message.
getFileExtensions()
Return the commonly used file extensions for these formats.
Exception thrown when a Gettext file could not be parsed, such as when missing required headers.
A very basic FileFormatSupport module that implements some basic functionality and a simple binary ba...
Constants for log channel names used in this extension.
Definition LogNames.php:13
This file contains the class for core message collections implementation.
Interface for message objects used by MessageCollection.
Definition Message.php:13
Minimal service container.
Definition Services.php:60
Essentially random collection of helper functions, similar to GlobalFunctions.php.
Definition Utilities.php:30
Message groups are usually configured in YAML, though the actual storage format does not matter,...