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\Category\Category;
8use MediaWiki\CommentStore\CommentStoreComment;
9use MediaWiki\Context\RequestContext;
14use MediaWiki\MediaWikiServices;
15use MediaWiki\Revision\RevisionStore;
16use MediaWiki\Revision\SlotRecord;
17use MediaWiki\Title\Title;
18use MediaWiki\User\User;
19use MediaWiki\User\UserIdentity;
20use MediaWiki\User\UserRigorOptions;
21use RecentChange;
22use Wikimedia\ScopedCallback;
23
31 public const ACTION_DELETE = 'delete';
32 public const ACTION_CATEGORIZATION = 'categorization';
33
34 public static function newJob(
35 Title $target,
36 ?string $triggerAction = null,
37 ?string $unitTitleText = null
38 ): self {
39 $job = new self( $target, [ 'triggerAction' => $triggerAction, 'unitTitle' => $unitTitleText ] );
40 $job->setUser( FuzzyBot::getUser() );
41 $job->setFlags( EDIT_FORCE_BOT );
42 $job->setSummary( wfMessage( 'tpt-render-summary' )->inContentLanguage()->text() );
43
44 return $job;
45 }
46
47 public static function newNonPrioritizedJob(
48 Title $target,
49 ?string $triggerAction = null,
50 ?string $unitTitleText = null
51 ): self {
52 $job = self::newJob( $target, $triggerAction, $unitTitleText );
53 $job->command = 'NonPrioritizedRenderTranslationPageJob';
54 return $job;
55 }
56
57 public function __construct( Title $title, array $params = [] ) {
58 parent::__construct( 'RenderTranslationPageJob', $title, $params );
59 $this->removeDuplicates = true;
60 }
61
62 public function run(): bool {
63 $this->logJobStart();
64 $mwServices = MediaWikiServices::getInstance();
65 // We may be doing double wait here if this job was spawned by TranslationUpdateJob
66 $lb = $mwServices->getDBLoadBalancerFactory();
67 if ( !$lb->waitForReplication() ) {
68 $this->logWarning( 'Continuing despite replication lag' );
69 }
70
71 // Initialization
72 $translationPageTitle = $this->title;
73
74 $tpPage = TranslatablePage::getTranslationPageFromTitle( $translationPageTitle );
75 if ( !$tpPage ) {
76 $this->logError( 'Cannot render translation page!' );
77 return false;
78 }
79
80 // Other stuff
81 $user = $this->getUser();
82 $summary = $this->getSummary();
83 $flags = $this->getFlags();
84
85 if ( isset( $this->params['session'] ) ) {
86 $scope = RequestContext::importScopedSession( $this->params['session'] );
87 $this->addTeardownCallback( static function () use ( &$scope ) {
88 ScopedCallback::consume( $scope );
89 } );
90 }
91
92 // We should not re-create the translation page if a translation unit is being deleted
93 // because it is possible that the translation page may also be queued for deletion.
94 // Hence, set the flag to EDIT_UPDATE and remove EDIT_NEW if its added
95 if ( $this->isDeleteTrigger() ) {
96 $flags = ( $flags | EDIT_UPDATE ) & ~EDIT_NEW;
97 }
98
99 // @todo FuzzyBot hack
100 Hooks::$allowTargetEdit = true;
101
102 $commentStoreComment = CommentStoreComment::newUnsavedComment( $summary );
103 // $percentageTranslated is modified by reference
104 $content = $tpPage->getPageContent( $mwServices->getParser(), $percentageTranslated );
105 $translationPageTitleExists = $translationPageTitle->exists();
106 if ( $this->isCategoryTrigger() ) {
107 $isNonEmptyCategory = true;
108 } elseif ( $translationPageTitle->inNamespace( NS_CATEGORY ) ) {
109 $cat = Category::newFromTitle( $translationPageTitle );
110 $isNonEmptyCategory = $cat->getMemberCount() > 0;
111 } else {
112 $isNonEmptyCategory = false;
113 }
114 if ( $percentageTranslated === 0 && !$translationPageTitleExists && !$isNonEmptyCategory ) {
115 Hooks::$allowTargetEdit = false;
116 $this->logInfo( 'No translations found and translation page does not exist. Nothing to do.' );
117 return true;
118 }
119
120 if (
121 $percentageTranslated === 0 &&
122 $translationPageTitleExists &&
123 $this->hasOnlyFuzzyBotAsAuthor( $mwServices->getRevisionStore(), $translationPageTitle ) &&
124 !$isNonEmptyCategory
125 ) {
126 $this->logInfo( 'Deleting translation page having no translations and modified only by Fuzzybot' );
127 // Page is not translated at all but the translation page exists and has been only edited by FuzzyBot
128 $this->deleteTranslationPage( $mwServices->getJobQueueGroup(), $translationPageTitle, FuzzyBot::getUser() );
129 } else {
130 $pageUpdater = $mwServices->getWikiPageFactory()
131 ->newFromTitle( $translationPageTitle )
132 ->newPageUpdater( $user );
133 $pageUpdater->setContent( SlotRecord::MAIN, $content );
134
135 if ( $user->authorizeWrite( 'autopatrol', $translationPageTitle ) ) {
136 $pageUpdater->setRcPatrolStatus( RecentChange::PRC_AUTOPATROLLED );
137 }
138
139 $pageUpdater->addTag( 'translate-translation-pages' );
140 $pageUpdater->saveRevision( $commentStoreComment, $flags );
141 $status = $pageUpdater->getStatus();
142
143 if ( !$status->isOK() ) {
144 if ( $this->isDeleteTrigger() && $status->hasMessage( 'edit-gone-missing' ) ) {
145 $this->logInfo( 'Translation page missing with delete trigger' );
146 } else {
147 $this->logError(
148 'Error while editing content in page.',
149 [
150 'content' => $content->getTextForSummary(),
151 'errors' => $status->getErrors()
152 ]
153 );
154 }
155 }
156 }
157
158 Hooks::$allowTargetEdit = false;
159
160 $this->logInfo( 'Finished TranslateRenderJob' );
161 return true;
162 }
163
164 public function setFlags( int $flags ): void {
165 $this->params['flags'] = $flags;
166 }
167
168 private function getFlags(): int {
169 return $this->params['flags'];
170 }
171
172 public function setSummary( string $summary ): void {
173 $this->params['summary'] = $summary;
174 }
175
177 public function getDeduplicationInfo(): array {
178 $info = parent::getDeduplicationInfo();
179 // Unit title is only passed for logging and should not be used for de-duplication
180 unset( $info['params']['unitTitle'] );
181 return $info;
182 }
183
184 private function getSummary(): string {
185 return $this->params['summary'];
186 }
187
192 public function setUser( $user, ?array $session = null ): void {
193 if ( $user instanceof UserIdentity ) {
194 $this->params['user'] = $user->getName();
195 } else {
196 $this->params['user'] = $user;
197 }
198
199 if ( $session ) {
200 $this->params['session'] = $session;
201 }
202 }
203
205 private function getUser(): User {
206 $userFactory = MediaWikiServices::getInstance()->getUserFactory();
207 return $userFactory->newFromName( $this->params['user'], UserRigorOptions::RIGOR_NONE );
208 }
209
210 private function isDeleteTrigger(): bool {
211 return ( $this->params['triggerAction'] ?? null ) === self::ACTION_DELETE;
212 }
213
214 private function isCategoryTrigger(): bool {
215 return ( $this->params['triggerAction'] ?? null ) === self::ACTION_CATEGORIZATION;
216 }
217
218 private function logJobStart(): void {
219 $unitTitleText = $this->params['unitTitle'] ?? null;
220 $logMessage = 'Starting TranslateRenderJob ';
221 if ( $unitTitleText ) {
222 $logMessage .= "trigged by $unitTitleText ";
223 }
224
225 if ( $this->isDeleteTrigger() ) {
226 $logMessage .= '- [deletion] ';
227 }
228
229 $this->logInfo( trim( $logMessage ) );
230 }
231
232 private function deleteTranslationPage(
233 JobQueueGroup $jobQueueGroup,
234 Title $translationPageTitle,
235 UserIdentity $performer
236 ): void {
237 $translatablePageTitle = ( new MessageHandle( $translationPageTitle ) )->getTitleForBase();
238 $isTranslationPage = true;
239
240 $job = DeleteTranslatableBundleJob::newJob(
241 $translationPageTitle,
242 $translatablePageTitle->getPrefixedText(),
243 TranslatablePage::class,
244 $isTranslationPage,
245 $performer,
246 wfMessage( 'pt-deletepage-lang-outdated-logreason' )->inContentLanguage()->text(),
247 $this->params['session'] ?? null
248 );
249
250 $jobQueueGroup->push( $job );
251 }
252
253 private function hasOnlyFuzzyBotAsAuthor( RevisionStore $revisionStore, Title $title ): bool {
254 $fuzzyBot = FuzzyBot::getUser();
255 $pageAuthors = $revisionStore->getAuthorsBetween( $title->getId() );
256 foreach ( $pageAuthors as $author ) {
257 if ( !$author->equals( $fuzzyBot ) ) {
258 return false;
259 }
260 }
261 return true;
262 }
263}
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->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);}, 'Translate:WorkflowStatesMessageGroupLoader'=> static function(MediaWikiServices $services):WorkflowStatesMessageGroupLoader { return new WorkflowStatesMessageGroupLoader(new ServiceOptions(WorkflowStatesMessageGroupLoader::CONSTRUCTOR_OPTIONS, $services->getMainConfig()),);},]
@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