Update docs/hooks.txt for ShowSearchHitTitle
[mediawiki.git] / includes / jobqueue / jobs / PublishStashedFileJob.php
blob37e80c24ce14d314131bb6878a8d70e059021a3b
1 <?php
2 /**
3 * Upload a file from the upload stash into the local file repo.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
20 * @file
21 * @ingroup Upload
22 * @ingroup JobQueue
24 use Wikimedia\ScopedCallback;
26 /**
27 * Upload a file from the upload stash into the local file repo.
29 * @ingroup Upload
30 * @ingroup JobQueue
32 class PublishStashedFileJob extends Job {
33 public function __construct( Title $title, array $params ) {
34 parent::__construct( 'PublishStashedFile', $title, $params );
35 $this->removeDuplicates = true;
38 public function run() {
39 $scope = RequestContext::importScopedSession( $this->params['session'] );
40 $this->addTeardownCallback( function () use ( &$scope ) {
41 ScopedCallback::consume( $scope ); // T126450
42 } );
44 $context = RequestContext::getMain();
45 $user = $context->getUser();
46 try {
47 if ( !$user->isLoggedIn() ) {
48 $this->setLastError( "Could not load the author user from session." );
50 return false;
53 UploadBase::setSessionStatus(
54 $user,
55 $this->params['filekey'],
56 [ 'result' => 'Poll', 'stage' => 'publish', 'status' => Status::newGood() ]
59 $upload = new UploadFromStash( $user );
60 // @todo initialize() causes a GET, ideally we could frontload the antivirus
61 // checks and anything else to the stash stage (which includes concatenation and
62 // the local file is thus already there). That way, instead of GET+PUT, there could
63 // just be a COPY operation from the stash to the public zone.
64 $upload->initialize( $this->params['filekey'], $this->params['filename'] );
66 // Check if the local file checks out (this is generally a no-op)
67 $verification = $upload->verifyUpload();
68 if ( $verification['status'] !== UploadBase::OK ) {
69 $status = Status::newFatal( 'verification-error' );
70 $status->value = [ 'verification' => $verification ];
71 UploadBase::setSessionStatus(
72 $user,
73 $this->params['filekey'],
74 [ 'result' => 'Failure', 'stage' => 'publish', 'status' => $status ]
76 $this->setLastError( "Could not verify upload." );
78 return false;
81 // Upload the stashed file to a permanent location
82 $status = $upload->performUpload(
83 $this->params['comment'],
84 $this->params['text'],
85 $this->params['watch'],
86 $user,
87 isset( $this->params['tags'] ) ? $this->params['tags'] : []
89 if ( !$status->isGood() ) {
90 UploadBase::setSessionStatus(
91 $user,
92 $this->params['filekey'],
93 [ 'result' => 'Failure', 'stage' => 'publish', 'status' => $status ]
95 $this->setLastError( $status->getWikiText( false, false, 'en' ) );
97 return false;
100 // Build the image info array while we have the local reference handy
101 $apiMain = new ApiMain(); // dummy object (XXX)
102 $imageInfo = $upload->getImageInfo( $apiMain->getResult() );
104 // Cleanup any temporary local file
105 $upload->cleanupTempFile();
107 // Cache the info so the user doesn't have to wait forever to get the final info
108 UploadBase::setSessionStatus(
109 $user,
110 $this->params['filekey'],
112 'result' => 'Success',
113 'stage' => 'publish',
114 'filename' => $upload->getLocalFile()->getName(),
115 'imageinfo' => $imageInfo,
116 'status' => Status::newGood()
119 } catch ( Exception $e ) {
120 UploadBase::setSessionStatus(
121 $user,
122 $this->params['filekey'],
124 'result' => 'Failure',
125 'stage' => 'publish',
126 'status' => Status::newFatal( 'api-error-publishfailed' )
129 $this->setLastError( get_class( $e ) . ": " . $e->getMessage() );
130 // To prevent potential database referential integrity issues.
131 // See bug 32551.
132 MWExceptionHandler::rollbackMasterChangesAndLog( $e );
134 return false;
137 return true;
140 public function getDeduplicationInfo() {
141 $info = parent::getDeduplicationInfo();
142 if ( is_array( $info['params'] ) ) {
143 $info['params'] = [ 'filekey' => $info['params']['filekey'] ];
146 return $info;
149 public function allowRetries() {
150 return false;