Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
RenderTranslationPageJob.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\PageTranslation;
5
6use JobQueueGroup;
7use MediaWiki\CommentStore\CommentStoreComment;
12use MediaWiki\MediaWikiServices;
13use MediaWiki\Revision\RevisionStore;
14use MediaWiki\Revision\SlotRecord;
15use MediaWiki\Title\Title;
16use MediaWiki\User\User;
17use MediaWiki\User\UserIdentity;
18use MediaWiki\User\UserRigorOptions;
19use RecentChange;
20
28 public const ACTION_DELETE = 'delete';
29
30 public static function newJob(
31 Title $target,
32 ?string $triggerAction = null,
33 ?string $unitTitleText = null
34 ): self {
35 $job = new self( $target, [ 'triggerAction' => $triggerAction, 'unitTitle' => $unitTitleText ] );
36 $job->setUser( FuzzyBot::getUser() );
37 $job->setFlags( EDIT_FORCE_BOT );
38 $job->setSummary( wfMessage( 'tpt-render-summary' )->inContentLanguage()->text() );
39
40 return $job;
41 }
42
43 public static function newNonPrioritizedJob(
44 Title $target,
45 ?string $triggerAction = null,
46 ?string $unitTitleText = null
47 ): self {
48 $job = self::newJob( $target, $triggerAction, $unitTitleText );
49 $job->command = 'NonPrioritizedRenderTranslationPageJob';
50 return $job;
51 }
52
53 public function __construct( Title $title, array $params = [] ) {
54 parent::__construct( 'RenderTranslationPageJob', $title, $params );
55 $this->removeDuplicates = true;
56 }
57
58 public function run(): bool {
59 $this->logJobStart();
60 $mwServices = MediaWikiServices::getInstance();
61 // We may be doing double wait here if this job was spawned by TranslationUpdateJob
62 $lb = $mwServices->getDBLoadBalancerFactory();
63 if ( !$lb->waitForReplication() ) {
64 $this->logWarning( 'Continuing despite replication lag' );
65 }
66
67 // Initialization
68 $translationPageTitle = $this->title;
69
70 $tpPage = TranslatablePage::getTranslationPageFromTitle( $translationPageTitle );
71 if ( !$tpPage ) {
72 $this->logError( 'Cannot render translation page!' );
73 return false;
74 }
75
76 // Other stuff
77 $user = $this->getUser();
78 $summary = $this->getSummary();
79 $flags = $this->getFlags();
80
81 // We should not re-create the translation page if a translation unit is being deleted
82 // because it is possible that the translation page may also be queued for deletion.
83 // Hence, set the flag to EDIT_UPDATE and remove EDIT_NEW if its added
84 if ( $this->isDeleteTrigger() ) {
85 $flags = ( $flags | EDIT_UPDATE ) & ~EDIT_NEW;
86 }
87
88 // @todo FuzzyBot hack
89 Hooks::$allowTargetEdit = true;
90
91 $commentStoreComment = CommentStoreComment::newUnsavedComment( $summary );
92 // $percentageTranslated is modified by reference
93 $content = $tpPage->getPageContent( $mwServices->getParser(), $percentageTranslated );
94 $translationPageTitleExists = $translationPageTitle->exists();
95 if ( $percentageTranslated === 0 && !$translationPageTitleExists ) {
96 Hooks::$allowTargetEdit = false;
97 $this->logInfo( 'No translations found and translation page does not exist. Nothing to do.' );
98 return true;
99 }
100
101 if (
102 $percentageTranslated === 0 &&
103 $translationPageTitleExists &&
104 $this->hasOnlyFuzzyBotAsAuthor( $mwServices->getRevisionStore(), $translationPageTitle )
105 ) {
106 $this->logInfo( 'Deleting translation page having no translations and modified only by Fuzzybot' );
107 // Page is not translated at all but the translation page exists and has been only edited by FuzzyBot
108 $this->deleteTranslationPage( $mwServices->getJobQueueGroup(), $translationPageTitle, FuzzyBot::getUser() );
109 } else {
110 $pageUpdater = $mwServices->getWikiPageFactory()
111 ->newFromTitle( $translationPageTitle )
112 ->newPageUpdater( $user );
113 $pageUpdater->setContent( SlotRecord::MAIN, $content );
114
115 if ( $user->authorizeWrite( 'autopatrol', $translationPageTitle ) ) {
116 $pageUpdater->setRcPatrolStatus( RecentChange::PRC_AUTOPATROLLED );
117 }
118
119 $pageUpdater->addTag( 'translate-translation-pages' );
120 $pageUpdater->saveRevision( $commentStoreComment, $flags );
121 $status = $pageUpdater->getStatus();
122
123 if ( !$status->isOK() ) {
124 if ( $this->isDeleteTrigger() && $status->hasMessage( 'edit-gone-missing' ) ) {
125 $this->logInfo( 'Translation page missing with delete trigger' );
126 } else {
127 $this->logError(
128 'Error while editing content in page.',
129 [
130 'content' => $content->getTextForSummary(),
131 'errors' => $status->getErrors()
132 ]
133 );
134 }
135 }
136 }
137
138 Hooks::$allowTargetEdit = false;
139
140 $this->logInfo( 'Finished TranslateRenderJob' );
141 return true;
142 }
143
144 public function setFlags( int $flags ): void {
145 $this->params['flags'] = $flags;
146 }
147
148 private function getFlags(): int {
149 return $this->params['flags'];
150 }
151
152 public function setSummary( string $summary ): void {
153 $this->params['summary'] = $summary;
154 }
155
157 public function getDeduplicationInfo(): array {
158 $info = parent::getDeduplicationInfo();
159 // Unit title is only passed for logging and should not be used for de-duplication
160 unset( $info['params']['unitTitle'] );
161 return $info;
162 }
163
164 private function getSummary(): string {
165 return $this->params['summary'];
166 }
167
169 public function setUser( $user ): void {
170 if ( $user instanceof UserIdentity ) {
171 $this->params['user'] = $user->getName();
172 } else {
173 $this->params['user'] = $user;
174 }
175 }
176
178 private function getUser(): User {
179 $userFactory = MediaWikiServices::getInstance()->getUserFactory();
180 return $userFactory->newFromName( $this->params['user'], UserRigorOptions::RIGOR_NONE );
181 }
182
183 private function isDeleteTrigger(): bool {
184 $triggerAction = $this->params['triggerAction'] ?? null;
185 return $triggerAction === self::ACTION_DELETE;
186 }
187
188 private function logJobStart(): void {
189 $unitTitleText = $this->params['unitTitle'] ?? null;
190 $logMessage = 'Starting TranslateRenderJob ';
191 if ( $unitTitleText ) {
192 $logMessage .= "trigged by $unitTitleText ";
193 }
194
195 if ( $this->isDeleteTrigger() ) {
196 $logMessage .= '- [deletion] ';
197 }
198
199 $this->logInfo( trim( $logMessage ) );
200 }
201
202 private function deleteTranslationPage(
203 JobQueueGroup $jobQueueGroup,
204 Title $translationPageTitle,
205 UserIdentity $performer
206 ): void {
207 $translatablePageTitle = ( new MessageHandle( $translationPageTitle ) )->getTitleForBase();
208 $isTranslationPage = true;
209
210 $job = DeleteTranslatableBundleJob::newJob(
211 $translationPageTitle,
212 $translatablePageTitle->getPrefixedText(),
213 TranslatablePage::class,
214 $isTranslationPage,
215 $performer,
216 wfMessage( 'pt-deletepage-lang-outdated-logreason' )->inContentLanguage()->text()
217 );
218
219 $jobQueueGroup->push( $job );
220 }
221
222 private function hasOnlyFuzzyBotAsAuthor( RevisionStore $revisionStore, Title $title ): bool {
223 $fuzzyBot = FuzzyBot::getUser();
224 $pageAuthors = $revisionStore->getAuthorsBetween( $title->getId() );
225 foreach ( $pageAuthors as $author ) {
226 if ( !$author->equals( $fuzzyBot ) ) {
227 return false;
228 }
229 }
230 return true;
231 }
232}
return[ 'Translate:AggregateGroupManager'=> static function(MediaWikiServices $services):AggregateGroupManager { return new AggregateGroupManager( $services->getTitleFactory());}, '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( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'), $services->getTitleFactory(), new ServiceOptions(ExternalMessageSourceStateImporter::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:FileBasedMessageGroupFactory'=> static function(MediaWikiServices $services):FileBasedMessageGroupFactory { return new FileBasedMessageGroupFactory(new MessageGroupConfigurationParser(), 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: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->getDBLoadBalancer());}, '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->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( 'Translate.MessageGroupSubscription'), new ServiceOptions(MessageGroupSubscription::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:MessageGroupSubscriptionHookHandler'=> static function(MediaWikiServices $services):MessageGroupSubscriptionHookHandler { return new MessageGroupSubscriptionHookHandler($services->get( 'Translate:MessageGroupSubscription'), $services->getUserFactory());}, 'Translate:MessageGroupSubscriptionStore'=> static function(MediaWikiServices $services):MessageGroupSubscriptionStore { return new MessageGroupSubscriptionStore( $services->getDBLoadBalancerFactory());}, '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( 'Translate'), $services->getMainObjectStash(), $services->getDBLoadBalancerFactory(), $services->get( 'Translate:MessageGroupSubscription'), 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->getDBLoadBalancer(), $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->getDBLoadBalancer());}, '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->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(), $services->getNamespaceInfo(), $services->getTitleFactory());}, '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->getDBLoadBalancerFactory(), $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:TranslatablePageMarker'=> static function(MediaWikiServices $services):TranslatablePageMarker { return new TranslatablePageMarker($services->getDBLoadBalancer(), $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'));}, '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->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:TranslatablePageView'=> static function(MediaWikiServices $services):TranslatablePageView { return new TranslatablePageView($services->getDBLoadBalancerFactory(), $services->get( 'Translate:TranslatablePageStateStore'), new ServiceOptions(TranslatablePageView::SERVICE_OPTIONS, $services->getMainConfig()));}, 'Translate:TranslateSandbox'=> static function(MediaWikiServices $services):TranslateSandbox { return new TranslateSandbox($services->getUserFactory(), $services->getDBLoadBalancer(), $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 { $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
Class for pointing to messages, like Title class is for titles.
Job for updating translation pages when translation or template changes.
FuzzyBot - the misunderstood workhorse.
Definition FuzzyBot.php:15