addDescription( 'Perform administrative tasks on a job queue' ); $this->addOption( 'type', 'Job type', true, true ); $this->addOption( 'action', 'Queue operation ("delete", "repush-abandoned")', true, true ); } public function execute() { $type = $this->getOption( 'type' ); $action = $this->getOption( 'action' ); $group = JobQueueGroup::singleton(); $queue = $group->get( $type ); if ( $action === 'delete' ) { $this->delete( $queue ); } elseif ( $action === 'repush-abandoned' ) { $this->repushAbandoned( $queue ); } else { $this->fatalError( "Invalid action '$action'." ); } } private function delete( JobQueue $queue ) { $this->output( "Queue has {$queue->getSize()} job(s); deleting...\n" ); $queue->delete(); $this->output( "Done; current size is {$queue->getSize()} job(s).\n" ); } private function repushAbandoned( JobQueue $queue ) { $cache = ObjectCache::getInstance( CACHE_DB ); $key = $cache->makeGlobalKey( 'last-job-repush', $queue->getWiki(), $queue->getType() ); $now = wfTimestampNow(); $lastRepushTime = $cache->get( $key ); if ( $lastRepushTime === false ) { $lastRepushTime = wfTimestamp( TS_MW, 1 ); // include all jobs } $this->output( "Last re-push time: $lastRepushTime; current time: $now\n" ); $count = 0; $skipped = 0; foreach ( $queue->getAllAbandonedJobs() as $job ) { /** @var Job $job */ if ( $job->getQueuedTimestamp() < wfTimestamp( TS_UNIX, $lastRepushTime ) ) { ++$skipped; continue; // already re-pushed in prior round } $queue->push( $job ); ++$count; if ( ( $count % $this->getBatchSize() ) == 0 ) { $queue->waitForBackups(); } } $cache->set( $key, $now ); // next run will ignore these jobs $this->output( "Re-pushed $count job(s) [$skipped skipped].\n" ); } } $maintClass = ManageJobs::class; require_once RUN_MAINTENANCE_IF_MAIN;