Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
TranslatableBundleMover.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\PageTranslation;
5
6use JobQueueGroup;
7use LogicException;
8use MediaWiki\Cache\LinkBatchFactory;
14use MediaWiki\Message\Message;
15use MediaWiki\Page\MovePageFactory;
16use MediaWiki\Status\Status;
17use MediaWiki\Title\Title;
18use MediaWiki\User\User;
19use ObjectCacheFactory;
20use SplObjectStorage;
21use Wikimedia\ObjectCache\BagOStuff;
22use Wikimedia\Rdbms\IConnectionProvider;
23
31 private const LOCK_TIMEOUT = 3600 * 2;
32 private const FETCH_TRANSLATABLE_SUBPAGES = true;
33 private MovePageFactory $movePageFactory;
34 private ?int $pageMoveLimit;
35 private JobQueueGroup $jobQueue;
36 private LinkBatchFactory $linkBatchFactory;
37 private TranslatableBundleFactory $bundleFactory;
38 private SubpageListBuilder $subpageBuilder;
39 private IConnectionProvider $dbProvider;
40 private BagOStuff $cache;
41 private bool $pageMoveLimitEnabled = true;
42
43 private const REDIRECTABLE_PAGE_TYPES = [
44 'pt-movepage-list-source' => true,
45 'pt-movepage-list-section' => false,
46 'pt-movepage-list-nonmovable' => false,
47 'pt-movepage-list-translatable' => false,
48 'pt-movepage-list-translation' => false,
49 'pt-movepage-list-other' => true
50 ];
51
52 public function __construct(
53 MovePageFactory $movePageFactory,
54 JobQueueGroup $jobQueue,
55 LinkBatchFactory $linkBatchFactory,
56 TranslatableBundleFactory $bundleFactory,
57 SubpageListBuilder $subpageBuilder,
58 IConnectionProvider $dbProvider,
59 ObjectCacheFactory $objectCacheFactory,
60 ?int $pageMoveLimit
61 ) {
62 $this->movePageFactory = $movePageFactory;
63 $this->jobQueue = $jobQueue;
64 $this->pageMoveLimit = $pageMoveLimit;
65 $this->linkBatchFactory = $linkBatchFactory;
66 $this->bundleFactory = $bundleFactory;
67 $this->subpageBuilder = $subpageBuilder;
68 $this->dbProvider = $dbProvider;
69 $this->cache = $objectCacheFactory->getInstance( CACHE_ANYTHING );
70 }
71
72 public function getPageMoveCollection(
73 Title $source,
74 ?Title $target,
75 User $user,
76 string $reason,
77 bool $moveSubPages,
78 bool $moveTalkPages,
79 bool $leaveRedirect
81 $blockers = new SplObjectStorage();
82
83 if ( !$target ) {
84 $blockers[$source] = Status::newFatal( 'pt-movepage-block-base-invalid' );
85 throw new ImpossiblePageMove( $blockers );
86 }
87
88 if ( $target->inNamespaces( NS_MEDIAWIKI, NS_TRANSLATIONS ) ) {
89 $blockers[$source] = Status::newFatal( 'immobile-target-namespace', $target->getNsText() );
90 throw new ImpossiblePageMove( $blockers );
91 }
92
93 // Set allowTargetEdit = true so Hooks knows you're going through the right system
94 Hooks::$allowTargetEdit = true;
95 $movePage = $this->movePageFactory->newMovePage( $source, $target );
96 $status = $movePage->isValidMove();
97 $status->merge( $movePage->probablyCanMove( $user, $reason ) );
98 Hooks::$allowTargetEdit = false;
99 if ( !$status->isOK() ) {
100 $blockers[$source] = $status;
101 }
102
103 // Don't spam the same errors for all pages if base page fails
104 if ( count( $blockers ) ) {
105 throw new ImpossiblePageMove( $blockers );
106 }
107
108 $pageCollection = $this->getPagesToMove(
109 $source, $target, $moveSubPages, self::FETCH_TRANSLATABLE_SUBPAGES, $moveTalkPages, $leaveRedirect
110 );
111
112 // Collect all the old and new titles for checks
113 $titles = [
114 'tp' => $pageCollection->getTranslationPagesPair(),
115 'subpage' => $pageCollection->getSubpagesPair(),
116 'section' => $pageCollection->getUnitPagesPair()
117 ];
118
119 // Check that all new titles are valid and count them. Add 1 for source page.
120 $moveCount = 1;
121 $lb = $this->linkBatchFactory->newLinkBatch();
122 foreach ( $titles as $type => $list ) {
123 $moveCount += count( $list );
124 // Give grep a chance to find the usages:
125 // pt-movepage-block-tp-invalid, pt-movepage-block-section-invalid,
126 // pt-movepage-block-subpage-invalid
127 foreach ( $list as $pair ) {
128 $old = $pair->getOldTitle();
129 $new = $pair->getNewTitle();
130
131 if ( $new === null ) {
132 $blockers[$old] = $this->getRenameMoveBlocker( $old, $type, $pair->getRenameErrorCode() );
133 continue;
134 }
135 $lb->addObj( $old );
136 $lb->addObj( $new );
137 }
138 }
139
140 if ( $this->pageMoveLimitEnabled ) {
141 if ( $this->pageMoveLimit !== null && $moveCount > $this->pageMoveLimit ) {
142 $blockers[$source] = Status::newFatal(
143 'pt-movepage-page-count-limit',
144 Message::numParam( $this->pageMoveLimit )
145 );
146 }
147 }
148
149 // Stop further validation if there are blockers already.
150 if ( count( $blockers ) ) {
151 throw new ImpossiblePageMove( $blockers );
152 }
153
154 // Check that there are no move blockers
155 $lb->setCaller( __METHOD__ )->execute();
156 foreach ( $titles as $type => $list ) {
157 foreach ( $list as $pair ) {
158 $old = $pair->getOldTitle();
159 $new = $pair->getNewTitle();
160
161 /* This method has terrible performance:
162 * - 2 queries by core
163 * - 3 queries by lqt
164 * - and no obvious way to preload the data! */
165 Hooks::$allowTargetEdit = true;
166 $movePage = $this->movePageFactory->newMovePage( $old, $new );
167 $status = $movePage->isValidMove();
168 Hooks::$allowTargetEdit = false;
169
170 // Do not check for permissions here, as these pages are not editable/movable
171 // in regular use
172 if ( !$status->isOK() ) {
173 $blockers[$old] = $status;
174 }
175
176 /* Because of the poor performance, check only one of the possibly thousands
177 * of section pages and assume rest are fine. This assumes section pages are
178 * listed last in the array. */
179 if ( $type === 'section' ) {
180 break;
181 }
182 }
183 }
184
185 if ( count( $blockers ) ) {
186 throw new ImpossiblePageMove( $blockers );
187 }
188
189 return $pageCollection;
190 }
191
192 public function moveAsynchronously(
193 Title $source,
194 Title $target,
195 bool $moveSubPages,
196 User $user,
197 string $moveReason,
198 bool $moveTalkPages,
199 bool $leaveRedirect,
200 array $userSessionInfo
201 ): void {
202 $pageCollection = $this->getPagesToMove(
203 $source, $target, $moveSubPages, !self::FETCH_TRANSLATABLE_SUBPAGES, $moveTalkPages, $leaveRedirect
204 );
205 $pagesToMove = $pageCollection->getListOfPages();
206 $pagesToLeaveRedirect = $pageCollection->getListOfPagesToRedirect();
207
209 $source,
210 $target,
211 $pagesToMove,
212 $pagesToLeaveRedirect,
213 $moveReason,
214 $user,
215 $userSessionInfo
216 );
217
218 $this->lock( array_keys( $pagesToMove ) );
219 $this->lock( array_values( $pagesToMove ) );
220
221 $this->jobQueue->push( $job );
222 }
223
233 public function moveSynchronously(
234 Title $source,
235 Title $target,
236 array $pagesToMove,
237 array $pagesToRedirect,
238 User $performer,
239 string $moveReason,
240 ?callable $progressCallback = null
241 ): void {
242 $sourceBundle = $this->bundleFactory->getValidBundle( $source );
243
244 $this->move( $sourceBundle, $performer, $pagesToMove, $pagesToRedirect, $moveReason, $progressCallback );
245
246 $this->bundleFactory->getStore( $sourceBundle )->move( $source, $target );
247
248 $this->bundleFactory->getPageMoveLogger( $sourceBundle )
249 ->logSuccess( $performer, $target, $moveReason );
250 }
251
252 public function disablePageMoveLimit(): void {
253 $this->pageMoveLimitEnabled = false;
254 }
255
256 public function enablePageMoveLimit(): void {
257 $this->pageMoveLimitEnabled = true;
258 }
259
260 public static function shouldLeaveRedirect( string $pageType, bool $leaveRedirect ): bool {
261 return self::REDIRECTABLE_PAGE_TYPES[ $pageType ] && $leaveRedirect;
262 }
263
264 private function getPagesToMove(
265 Title $source,
266 Title $target,
267 bool $moveSubPages,
268 bool $fetchTranslatableSubpages,
269 bool $moveTalkPages,
270 bool $leaveRedirect
271 ): PageMoveCollection {
272 $sourceBundle = $this->bundleFactory->getValidBundle( $source );
273
274 $classifiedSubpages = $this->subpageBuilder->getSubpagesPerType( $sourceBundle, $moveTalkPages );
275
276 $talkPages = $moveTalkPages ? $classifiedSubpages['talkPages'] : [];
277 $subpages = $moveSubPages ? $classifiedSubpages['normalSubpages'] : [];
278 $nonMovableSubpages = [];
279
280 $relatedTranslatablePageList = [];
281 if ( $fetchTranslatableSubpages ) {
282 $relatedTranslatablePageList = array_merge(
283 $classifiedSubpages['translatableSubpages'],
284 $classifiedSubpages['translatableTalkPages']
285 );
286 }
287
288 $movePageFactory = $this->movePageFactory;
289 $pageTitleRenamer = new PageTitleRenamer( $source, $target );
290 $createOps = function ( array $pages, string $pageType )
291 use ( $pageTitleRenamer, $talkPages, &$nonMovableSubpages, $leaveRedirect, $movePageFactory ) {
292 $leaveRedirect = self::shouldLeaveRedirect( $pageType, $leaveRedirect );
293 $ops = [];
294 foreach ( $pages as $from ) {
295 $to = $pageTitleRenamer->getNewTitle( $from );
296 $op = new PageMoveOperation( $from, $to );
297
298 if ( $pageType === 'pt-movepage-list-other' ) {
299 // TODO: In the future, think of moving all checks regarding whether a page
300 // is movable into this method. Currently its' being checked in two places
301 // making things slow.
302 $movePage = $movePageFactory->newMovePage( $from, $to );
303 $status = $movePage->isValidMove();
304
305 // Remove non movable subpages
306 if ( !$status->isOK() ) {
307 $nonMovableSubpages[ $from->getPrefixedText() ] = $status;
308 continue;
309 }
310 }
311
312 $op->setLeaveRedirect( $leaveRedirect );
313
314 $talkPage = $talkPages[ $from->getPrefixedDBkey() ] ?? null;
315 if ( $talkPage ) {
316 $op->setTalkpage( $talkPage, $pageTitleRenamer->getNewTitle( $talkPage ) );
317 }
318 $ops[] = $op;
319 }
320
321 return $ops;
322 };
323
324 return new PageMoveCollection(
325 $createOps( [ $source ], 'pt-movepage-list-source' )[0],
326 $createOps( $classifiedSubpages['translationPages'], 'pt-movepage-list-translation' ),
327 $createOps( $classifiedSubpages['translationUnitPages'], 'pt-movepage-list-section' ),
328 $createOps( $subpages, 'pt-movepage-list-other' ),
329 $nonMovableSubpages,
330 $relatedTranslatablePageList
331 );
332 }
333
335 private function lock( array $titles ): void {
336 $data = [];
337 foreach ( $titles as $title ) {
338 $data[$this->cache->makeKey( 'pt-lock', sha1( $title ) )] = 'locked';
339 }
340
341 // Do not lock pages indefinitely during translatable page moves since
342 // they can fail. Add a timeout so that the locks expire by themselves.
343 // Timeout value has been chosen by a gut feeling
344 $this->cache->setMulti( $data, self::LOCK_TIMEOUT );
345 }
346
348 private function unlock( array $titles ): void {
349 foreach ( $titles as $title ) {
350 $this->cache->delete( $this->cache->makeKey( 'pt-lock', sha1( $title ) ) );
351 }
352 }
353
362 private function move(
363 TranslatableBundle $sourceBundle,
364 User $performer,
365 array $pagesToMove,
366 array $pagesToRedirect,
367 string $reason,
368 ?callable $progressCallback = null
369 ): void {
370 $fuzzyBot = FuzzyBot::getUser();
371
372 Hooks::$allowTargetEdit = true;
373
374 $processed = 0;
375
376 $this->dbProvider->getPrimaryDatabase()->startAtomic( __METHOD__ );
377 foreach ( $pagesToMove as $source => $target ) {
378 $sourceTitle = Title::newFromText( $source );
379 $targetTitle = Title::newFromText( $target );
380
381 if ( $source === $sourceBundle->getTitle()->getPrefixedText() ) {
382 $user = $performer;
383 $moveSummary = $reason;
384 } else {
385 $user = $fuzzyBot;
386 $moveSummary = wfMessage(
387 'pt-movepage-logreason', $sourceBundle->getTitle()->getPrefixedText()
388 )->text();
389 }
390
391 $mover = $this->movePageFactory->newMovePage( $sourceTitle, $targetTitle );
392 $status = $mover->move( $user, $moveSummary, $pagesToRedirect[$source] ?? false );
393 $processed++;
394
395 if ( $progressCallback ) {
396 $progressCallback(
397 $sourceTitle,
398 $targetTitle,
399 $status,
400 count( $pagesToMove ),
401 $processed
402 );
403 }
404
405 if ( !$status->isOK() ) {
406 $this->bundleFactory->getPageMoveLogger( $sourceBundle )
407 ->logError( $performer, $sourceTitle, $targetTitle, $status );
408 }
409
410 $this->unlock( [ $source, $target ] );
411 }
412 $this->dbProvider->getPrimaryDatabase()->endAtomic( __METHOD__ );
413
414 Hooks::$allowTargetEdit = false;
415 }
416
417 private function getRenameMoveBlocker( Title $old, string $pageType, int $renameError ): Status {
418 if ( $renameError === PageTitleRenamer::NO_ERROR ) {
419 throw new LogicException(
420 'Trying to fetch MoveBlocker when there was no error during rename. Title: ' .
421 $old->getPrefixedText() . ', page type: ' . $pageType
422 );
423 }
424
425 if ( $renameError === PageTitleRenamer::UNKNOWN_PAGE ) {
426 $status = Status::newFatal( 'pt-movepage-block-unknown-page', $old->getPrefixedText() );
427 } elseif ( $renameError === PageTitleRenamer::NS_TALK_UNSUPPORTED ) {
428 $status = Status::newFatal( 'pt-movepage-block-ns-talk-unsupported', $old->getPrefixedText() );
429 } elseif ( $renameError === PageTitleRenamer::RENAME_FAILED ) {
430 $status = Status::newFatal( 'pt-movepage-block-rename-failed', $old->getPrefixedText() );
431 } else {
432 return Status::newFatal( "pt-movepage-block-$pageType-invalid", $old->getPrefixedText() );
433 }
434
435 return $status;
436 }
437}
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
static newJob(Title $source, Title $target, array $moves, array $redirects, string $reason, User $performer, array $session)
Generates list of subpages for the translatable bundle that can be moved or deleted.
Create instances of various classes based on the type of TranslatableBundle.
Translatable bundle represents a message group where its translatable content is defined on a wiki pa...
Exception thrown when a translatable page move is not possible.
Collection of pages potentially affected by a page move operation.
Contains the core logic to validate and move translatable bundles.
moveSynchronously(Title $source, Title $target, array $pagesToMove, array $pagesToRedirect, User $performer, string $moveReason, ?callable $progressCallback=null)
FuzzyBot - the misunderstood workhorse.
Definition FuzzyBot.php:15