Merge "API: Avoid duplicate IDs in API documentation"
[lhc/web/wiklou.git] / includes / jobqueue / jobs / AssembleUploadChunksJob.php
1 <?php
2 /**
3 * Assemble the segments of a chunked upload.
4 *
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.
9 *
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.
14 *
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
19 *
20 * @file
21 * @ingroup Upload
22 */
23
24 /**
25 * Assemble the segments of a chunked upload.
26 *
27 * @ingroup Upload
28 */
29 class AssembleUploadChunksJob extends Job {
30 public function __construct( Title $title, array $params ) {
31 parent::__construct( 'AssembleUploadChunks', $title, $params );
32 $this->removeDuplicates = true;
33 }
34
35 public function run() {
36 $scope = RequestContext::importScopedSession( $this->params['session'] );
37 $this->addTeardownCallback( function () use ( &$scope ) {
38 ScopedCallback::consume( $scope ); // T126450
39 } );
40
41 $context = RequestContext::getMain();
42 $user = $context->getUser();
43 try {
44 if ( !$user->isLoggedIn() ) {
45 $this->setLastError( "Could not load the author user from session." );
46
47 return false;
48 }
49
50 UploadBase::setSessionStatus(
51 $user,
52 $this->params['filekey'],
53 [ 'result' => 'Poll', 'stage' => 'assembling', 'status' => Status::newGood() ]
54 );
55
56 $upload = new UploadFromChunks( $user );
57 $upload->continueChunks(
58 $this->params['filename'],
59 $this->params['filekey'],
60 new WebRequestUpload( $context->getRequest(), 'null' )
61 );
62
63 // Combine all of the chunks into a local file and upload that to a new stash file
64 $status = $upload->concatenateChunks();
65 if ( !$status->isGood() ) {
66 UploadBase::setSessionStatus(
67 $user,
68 $this->params['filekey'],
69 [ 'result' => 'Failure', 'stage' => 'assembling', 'status' => $status ]
70 );
71 $this->setLastError( $status->getWikiText( false, false, 'en' ) );
72
73 return false;
74 }
75
76 // We have a new filekey for the fully concatenated file
77 $newFileKey = $upload->getLocalFile()->getFileKey();
78
79 // Remove the old stash file row and first chunk file
80 $upload->stash->removeFileNoAuth( $this->params['filekey'] );
81
82 // Build the image info array while we have the local reference handy
83 $apiMain = new ApiMain(); // dummy object (XXX)
84 $imageInfo = $upload->getImageInfo( $apiMain->getResult() );
85
86 // Cleanup any temporary local file
87 $upload->cleanupTempFile();
88
89 // Cache the info so the user doesn't have to wait forever to get the final info
90 UploadBase::setSessionStatus(
91 $user,
92 $this->params['filekey'],
93 [
94 'result' => 'Success',
95 'stage' => 'assembling',
96 'filekey' => $newFileKey,
97 'imageinfo' => $imageInfo,
98 'status' => Status::newGood()
99 ]
100 );
101 } catch ( Exception $e ) {
102 UploadBase::setSessionStatus(
103 $user,
104 $this->params['filekey'],
105 [
106 'result' => 'Failure',
107 'stage' => 'assembling',
108 'status' => Status::newFatal( 'api-error-stashfailed' )
109 ]
110 );
111 $this->setLastError( get_class( $e ) . ": " . $e->getMessage() );
112 // To be extra robust.
113 MWExceptionHandler::rollbackMasterChangesAndLog( $e );
114
115 return false;
116 }
117
118 return true;
119 }
120
121 public function getDeduplicationInfo() {
122 $info = parent::getDeduplicationInfo();
123 if ( is_array( $info['params'] ) ) {
124 $info['params'] = [ 'filekey' => $info['params']['filekey'] ];
125 }
126
127 return $info;
128 }
129
130 public function allowRetries() {
131 return false;
132 }
133 }