41 private const MAX_PROCESSING_TIME = 43;
49 public function __construct( Title $title, User $user, $group =
null,
string $code =
'en' ) {
50 $this->setTitle( $title );
51 $this->setUser( $user );
53 $this->setCode( $code );
61 public function setTitle( Title $title ): void {
62 $this->title = $title;
65 public function getUser(): User {
69 public function setUser( User $user ): void {
80 $this->group = $group;
82 $this->group = MessageGroups::getGroup( $group );
86 public function getCode(): string {
90 public function setCode(
string $code =
'en' ): void {
94 protected function getAction(): string {
95 return $this->getTitle()->getLocalURL();
98 protected function doHeader(): string {
101 'action' => $this->getAction(),
102 'class' =>
'mw-translate-manage'
105 $csrfTokenSet = RequestContext::getMain()->getCsrfTokenSet();
106 return Xml::openElement(
'form', $formParams ) .
107 Html::hidden(
'title', $this->getTitle()->getPrefixedText() ) .
108 Html::hidden(
'token', $csrfTokenSet->getToken() ) .
109 Html::hidden(
'process', 1 );
112 protected function doFooter(): string {
116 protected function allowProcess(): bool {
117 $context = RequestContext::getMain();
118 $request = $context->getRequest();
119 $csrfTokenSet = $context->getCsrfTokenSet();
121 return $request->wasPosted()
122 && $request->getBool(
'process' )
123 && $csrfTokenSet->matchTokenField(
'token' );
126 protected function getActions(): array {
129 $this->code ===
'en' ?
'fuzzy' :
'conflict',
134 public function execute( array $messages ): bool {
135 $context = RequestContext::getMain();
136 $output = $context->getOutput();
139 $diff =
new DifferenceEngine();
140 $diff->showDiffStyle();
141 $diff->setReducedLineNumbers();
144 $process = $this->allowProcess();
147 $group = $this->getGroup();
148 $code = $this->getCode();
149 $collection = $group->initCollection( $code );
150 $collection->loadTranslations();
152 $output->addHTML( $this->doHeader() );
161 foreach ( $messages as $key => $value ) {
163 $isExistingMessageFuzzy =
false;
165 if ( isset( $collection[$key] ) ) {
167 $old = $collection[$key]->translation();
168 $isExistingMessageFuzzy = $collection[$key]->hasTag(
'fuzzy' );
171 if ( $old ===
null ) {
184 $para =
'<code class="mw-tmi-new">' . htmlspecialchars( $key ) .
'</code>';
185 $name = $context->msg(
'translate-manage-import-new' )->rawParams( $para )
187 $text = Utilities::convertWhiteSpaceToHTML( $value );
188 $changed[] = self::makeSectionElement( $name,
'new', $text );
191 if ( $old === (
string)$value ) {
196 $oldTextForDiff = $old;
197 if ( $isExistingMessageFuzzy ) {
198 if ( MessageHandle::makeFuzzyString( $old ) === (
string)$value ) {
207 $oldContent = ContentHandler::makeContent( $oldTextForDiff, $diff->getTitle() );
208 $newContent = ContentHandler::makeContent( $value, $diff->getTitle() );
209 $diff->setContent( $oldContent, $newContent );
210 $text = $diff->getDiff(
'',
'' );
217 $action = $context->getRequest()
218 ->getVal( self::escapeNameForPHP(
"action-$type-$key" ) );
221 if ( $changed === [] ) {
226 if ( $action ===
null ) {
229 $message = $context->msg(
230 'translate-manage-inconsistent',
231 wfEscapeWikiText(
"action-$type-$key" )
233 $changed[] =
"<li>$message</li></ul>";
239 if ( !isset( $this->time ) ) {
240 $this->time = (int)wfTimestamp();
245 $messageKeyAndParams = $this->doAction(
253 $msgKey = array_shift( $messageKeyAndParams );
254 $params = $messageKeyAndParams;
255 $message = $context->msg( $msgKey, $params )->parse();
256 $changed[] =
"<li>$message</li>";
260 if ( $this->checkProcessTime() ) {
262 $message = $context->msg(
'translate-manage-toolong' )
263 ->numParams( self::MAX_PROCESSING_TIME )->parse();
264 $changed[] =
"<li>$message</li></ul>";
276 $actions = $this->getActions();
277 $defaultAction = $action ?:
'import';
284 foreach ( $actions as $action ) {
285 $label = $context->msg(
"translate-manage-action-$action" )->text();
286 $name = self::escapeNameForPHP(
"action-$type-$key" );
287 $id = Sanitizer::escapeIdForAttribute(
"action-$key-$action" );
288 $act[] = Xml::radioLabel( $label, $name, $action, $id, $action === $defaultAction );
291 $param =
'<code class="mw-tmi-diff">' . htmlspecialchars( $key ) .
'</code>';
292 $name = $context->msg(
'translate-manage-import-diff' )
293 ->rawParams( $param, implode(
' ', $act ) )
296 $changed[] = self::makeSectionElement( $name, $type, $text );
301 $collection->filter(
'hastranslation',
false );
302 $keys = $collection->getMessageKeys();
304 $diff = array_diff( $keys, array_keys( $messages ) );
306 foreach ( $diff as $s ) {
307 $para =
'<code class="mw-tmi-deleted">' . htmlspecialchars( $s ) .
'</code>';
308 $name = $context->msg(
'translate-manage-import-deleted' )->rawParams( $para )->escaped();
309 $text = Utilities::convertWhiteSpaceToHTML( $collection[$s]->translation() );
310 $changed[] = self::makeSectionElement( $name,
'deleted', $text );
314 if ( $process || ( $changed === [] && $code !==
'en' ) ) {
315 if ( $changed === [] ) {
316 $output->addWikiMsg(
'translate-manage-nochanges-other' );
319 if ( $changed === [] || !str_starts_with( end( $changed ),
'<li>' ) ) {
323 $changed[] =
'</ul>';
325 $languageName = Utilities::getLanguageName( $code, $context->getLanguage()->getCode() );
327 ->msg(
'translate-manage-import-done', $group->getId(), $group->getLabel(), $languageName )
329 $changed[] = Html::successBox( $message );
330 $output->addHTML( implode(
"\n", $changed ) );
333 if ( $changed !== [] ) {
334 if ( $code ===
'en' ) {
335 $output->addWikiMsg(
'translate-manage-intro-en' );
337 $lang = Utilities::getLanguageName(
339 $context->getLanguage()->getCode()
341 $output->addWikiMsg(
'translate-manage-intro-other', $lang );
343 $output->addHTML( Html::hidden(
'language', $code ) );
344 $output->addHTML( implode(
"\n", $changed ) );
345 $output->addHTML( Xml::submitButton( $context->msg(
'translate-manage-submit' )->text() ) );
347 $output->addWikiMsg(
'translate-manage-nochanges' );
351 $output->addHTML( $this->doFooter() );
365 private function doAction(
371 global $wgTranslateDocumentationLanguageCode;
374 $code = $this->getCode();
375 $title = $this->makeTranslationTitle( $group, $key, $code );
377 if ( $action ===
'import' || $action ===
'conflict' ) {
378 if ( $action ===
'import' ) {
379 $comment = wfMessage(
'translate-manage-import-summary' )->inContentLanguage()->plain();
381 $comment = wfMessage(
'translate-manage-conflict-summary' )->inContentLanguage()->plain();
385 return self::doImport( $title, $message, $comment, $this->getUser() );
386 } elseif ( $action ===
'ignore' ) {
387 return [
'translate-manage-import-ignore', $key ];
388 } elseif ( $action ===
'fuzzy' && $code !==
'en' &&
389 $code !== $wgTranslateDocumentationLanguageCode
393 return self::doImport( $title, $message, $comment, $this->getUser() );
394 } elseif ( $action ===
'fuzzy' && $code ===
'en' ) {
395 return self::doFuzzy( $title, $message, $comment, $this->getUser() );
397 throw new InvalidArgumentException(
"Unhandled action $action" );
401 protected function checkProcessTime() {
402 return (
int)wfTimestamp() - $this->time >= self::MAX_PROCESSING_TIME;
406 private static function doImport(
412 $wikiPage = MediaWikiServices::getInstance()->getWikiPageFactory()->newFromTitle( $title );
413 $content = ContentHandler::makeContent( $message, $title );
414 $status = $wikiPage->doUserEditContent(
419 $success = $status->isOK();
422 return [
'translate-manage-import-ok',
423 wfEscapeWikiText( $title->getPrefixedText() )
427 $text =
"Failed to import new version of page {$title->getPrefixedText()}\n";
428 $text .= $status->getWikiText();
429 throw new RuntimeException( $text );
439 $context = RequestContext::getMain();
440 $services = MediaWikiServices::getInstance();
442 if ( !$context->getUser()->isAllowed(
'translate-manage' ) ) {
443 return [
'badaccess-group0' ];
448 $user = FuzzyBot::getUser();
453 $titleText = $handle->getKey();
455 $revStore = $services->getRevisionStore();
456 $queryInfo = $revStore->getQueryInfo( [
'page' ] );
457 $dbw = $services->getDBLoadBalancer()->getConnection( DB_PRIMARY );
458 $rows = $dbw->select(
459 $queryInfo[
'tables'],
460 $queryInfo[
'fields'],
462 'page_namespace' => $title->getNamespace(),
463 'page_latest=rev_id',
464 'page_title' . $dbw->buildLike(
"$titleText/", $dbw->anyString() ),
472 $slots = $revStore->getContentBlobsForBatch( $rows, [ SlotRecord::MAIN ] )->getValue();
474 foreach ( $rows as $row ) {
475 global $wgTranslateDocumentationLanguageCode;
477 $translationTitle = Title::makeTitle( (
int)$row->page_namespace, $row->page_title );
480 if ( $translationTitle->getSubpageText() ===
'en' ||
481 $translationTitle->getSubpageText() === $wgTranslateDocumentationLanguageCode
485 } elseif ( isset( $slots[$row->rev_id] ) ) {
486 $slot = $slots[$row->rev_id][SlotRecord::MAIN];
487 $text = MessageHandle::makeFuzzyString( $slot->blob_data );
489 $text = MessageHandle::makeFuzzyString(
490 Utilities::getTextFromTextContent(
491 $revStore->newRevisionFromRow( $row )->getContent( SlotRecord::MAIN )
497 $changed[] = self::doImport(
507 foreach ( $changed as $c ) {
508 $key = array_shift( $c );
509 $text .=
'* ' . $context->msg( $key, $c )->plain() .
"\n";
512 return [
'translate-manage-import-fuzzy',
"\n" . $text ];
524 private function makeTranslationTitle(
MessageGroup $group,
string $key,
string $code ): Title {
525 $ns = $group->getNamespace();
527 return Title::makeTitleSafe( $ns,
"$key/$code" );
543 Language $lang =
null
545 $containerParams = [
'class' =>
"mw-tpt-sp-section mw-tpt-sp-section-type-{$type}" ];
546 $legendParams = [
'class' =>
'mw-tpt-sp-legend' ];
547 $contentParams = [
'class' =>
'mw-tpt-sp-content' ];
549 $contentParams[
'dir'] = $lang->getDir();
550 $contentParams[
'lang'] = $lang->getCode();
553 return Html::rawElement(
'div', $containerParams,
554 Html::rawElement(
'div', $legendParams, $legend ) .
555 Html::rawElement(
'div', $contentParams, $content )
564 private static function escapeNameForPHP(
string $name ): string {
576 return strtr( $name, $replacements );
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