cd98c52b0fd2b04bf71d1fc4554d3a9431d10db9
[lhc/web/wiklou.git] / includes / api / ApiQueryRedirects.php
1 <?php
2 /**
3 * API module to return redirects to a page
4 *
5 * Created on Dec 30, 2013
6 *
7 * Copyright © 2013 Brad Jorsch <bjorsch@wikimedia.org>
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
22 * http://www.gnu.org/copyleft/gpl.html
23 *
24 * @file
25 * @since 1.23
26 */
27
28 /**
29 * This query lists redirects to the given pages.
30 *
31 * @ingroup API
32 */
33 class ApiQueryRedirects extends ApiQueryGeneratorBase {
34
35 public function __construct( $query, $moduleName ) {
36 parent::__construct( $query, $moduleName, 'rd' );
37 }
38
39 public function execute() {
40 $this->run();
41 }
42
43 public function executeGenerator( $resultPageSet ) {
44 $this->run( $resultPageSet );
45 }
46
47 /**
48 * @param ApiPageSet $resultPageSet
49 */
50 private function run( ApiPageSet $resultPageSet = null ) {
51 $db = $this->getDB();
52 $params = $this->extractRequestParams();
53 $emptyString = $db->addQuotes( '' );
54
55 $pageSet = $this->getPageSet();
56 $titles = $pageSet->getGoodTitles() + $pageSet->getMissingTitles();
57
58 if ( !is_null( $params['continue'] ) ) {
59 $cont = explode( '|', $params['continue'] );
60 $this->dieContinueUsageIf( count( $cont ) != 3 );
61 $rd_namespace = (int)$cont[0];
62 $this->dieContinueUsageIf( $rd_namespace != $cont[0] );
63 $rd_title = $db->addQuotes( $cont[1] );
64 $rd_from = (int)$cont[2];
65 $this->dieContinueUsageIf( $rd_from != $cont[2] );
66 $this->addWhere(
67 "rd_namespace > $rd_namespace OR " .
68 "(rd_namespace = $rd_namespace AND " .
69 "(rd_title > $rd_title OR " .
70 "(rd_title = $rd_title AND " .
71 "rd_from >= $rd_from)))"
72 );
73
74 // Remove titles that we're past already
75 $titles = array_filter( $titles, function ( $t ) use ( $rd_namespace, $rd_title ) {
76 $ns = $t->getNamespace();
77 return ( $ns > $rd_namespace ||
78 $ns == $rd_namespace && $t->getDBKey() >= $rd_title
79 );
80 } );
81 }
82
83 if ( !$titles ) {
84 return; // nothing to do
85 }
86
87 $this->addTables( array( 'redirect', 'page' ) );
88 $this->addFields( array(
89 'rd_from',
90 'rd_namespace',
91 'rd_title',
92 ) );
93
94 if ( is_null( $resultPageSet ) ) {
95 $prop = array_flip( $params['prop'] );
96 $fld_pageid = isset( $prop['pageid'] );
97 $fld_title = isset( $prop['title'] );
98 $fld_fragment = isset( $prop['fragment'] );
99
100 $this->addFieldsIf( 'rd_fragment', $fld_fragment );
101 $this->addFieldsIf( array( 'page_namespace', 'page_title' ), $fld_title );
102 } else {
103 $this->addFields( array( 'page_namespace', 'page_title' ) );
104 }
105
106 $lb = new LinkBatch( $titles );
107 $this->addWhere( array(
108 'rd_from = page_id',
109 "rd_interwiki = $emptyString OR rd_interwiki IS NULL",
110 $lb->constructSet( 'rd', $db ),
111 ) );
112
113 if ( $params['show'] !== null ) {
114 $show = array_flip( $params['show'] );
115 if ( isset( $show['fragment'] ) && isset( $show['!fragment'] ) ) {
116 $this->dieUsageMsg( 'show' );
117 }
118 $this->addWhereIf( "rd_fragment != $emptyString", isset( $show['fragment'] ) );
119 $this->addWhereIf( "rd_fragment = $emptyString OR rd_fragment IS NULL", isset( $show['!fragment'] ) );
120 }
121
122 $map = $pageSet->getAllTitlesByNamespace();
123
124 // Why, MySQL? Why do you do this to us?
125 $sortby = array();
126 if ( count( $map ) > 1 ) {
127 $sortby[] = 'rd_namespace';
128 }
129 $theTitle = null;
130 foreach ( $map as $nsTitles ) {
131 reset( $nsTitles );
132 $key = key( $nsTitles );
133 if ( $theTitle === null ) {
134 $theTitle = $key;
135 }
136 if ( count( $nsTitles ) > 1 || $key !== $theTitle ) {
137 $sortby[] = 'rd_title';
138 break;
139 }
140 }
141 $sortby[] = 'rd_from';
142 $this->addOption( 'ORDER BY', $sortby );
143
144 $this->addOption( 'LIMIT', $params['limit'] + 1 );
145
146 $res = $this->select( __METHOD__ );
147
148 if ( is_null( $resultPageSet ) ) {
149 $count = 0;
150 foreach ( $res as $row ) {
151 if ( ++$count > $params['limit'] ) {
152 // We've reached the one extra which shows that
153 // there are additional pages to be had. Stop here...
154 $this->setContinueEnumParameter( 'continue',
155 "$row->rd_namespace|$row->rd_title|$row->rd_from"
156 );
157 break;
158 }
159
160 # Get the ID of the current page
161 $id = $map[$row->rd_namespace][$row->rd_title];
162
163 $vals = array();
164 if ( $fld_pageid ) {
165 $vals['pageid'] = $row->rd_from;
166 }
167 if ( $fld_title ) {
168 ApiQueryBase::addTitleInfo( $vals,
169 Title::makeTitle( $row->page_namespace, $row->page_title )
170 );
171 }
172 if ( $fld_fragment && $row->rd_fragment !== null && $row->rd_fragment !== '' ) {
173 $vals['fragment'] = $row->rd_fragment;
174 }
175 $fit = $this->addPageSubItem( $id, $vals );
176 if ( !$fit ) {
177 $this->setContinueEnumParameter( 'continue',
178 "$row->rd_namespace|$row->rd_title|$row->rd_from"
179 );
180 break;
181 }
182 }
183 } else {
184 $titles = array();
185 $count = 0;
186 foreach ( $res as $row ) {
187 if ( ++$count > $params['limit'] ) {
188 // We've reached the one extra which shows that
189 // there are additional pages to be had. Stop here...
190 $this->setContinueEnumParameter( 'continue',
191 "$row->rd_namespace|$row->rd_title|$row->rd_from"
192 );
193 break;
194 }
195 $titles[] = Title::makeTitle( $row->page_namespace, $row->page_title );
196 }
197 $resultPageSet->populateFromTitles( $titles );
198 }
199 }
200
201 public function getCacheMode( $params ) {
202 return 'public';
203 }
204
205 public function getAllowedParams() {
206 return array(
207 'prop' => array(
208 ApiBase::PARAM_TYPE => array(
209 'pageid',
210 'title',
211 'fragment',
212 ),
213 ApiBase::PARAM_ISMULTI => true,
214 ApiBase::PARAM_DFLT => 'pageid|title',
215 ),
216 'show' => array(
217 ApiBase::PARAM_TYPE => array(
218 'fragment', '!fragment',
219 ),
220 ApiBase::PARAM_ISMULTI => true,
221 ),
222 'limit' => array(
223 ApiBase::PARAM_DFLT => 10,
224 ApiBase::PARAM_TYPE => 'limit',
225 ApiBase::PARAM_MIN => 1,
226 ApiBase::PARAM_MAX => ApiBase::LIMIT_BIG1,
227 ApiBase::PARAM_MAX2 => ApiBase::LIMIT_BIG2
228 ),
229 'continue' => null,
230 );
231 }
232
233 public function getParamDescription() {
234 return array(
235 'prop' => array(
236 'Which properties to get:',
237 ' pageid - Page id of each redirect',
238 ' title - Title of each redirect',
239 ' fragment - Fragment of each redirect, if any',
240 ),
241 'show' => array(
242 'Show only items that meet this criteria.',
243 ' fragment - Only show redirects with a fragment',
244 ' !fragment - Only show redirects without a fragment',
245 ),
246 'limit' => 'How many redirects to return',
247 'continue' => 'When more results are available, use this to continue',
248 );
249 }
250
251 public function getDescription() {
252 return 'Returns all redirects to the given page(s).';
253 }
254
255 public function getExamples() {
256 return array(
257 'api.php?action=query&prop=redirects&titles=Main%20Page'
258 => 'Get a list of redirects to the [[Main Page]]',
259 'api.php?action=query&generator=redirects&titles=Main%20Page&prop=info'
260 => 'Get information about all redirects to the [[Main Page]]',
261 );
262 }
263
264 public function getHelpUrls() {
265 return 'https://www.mediawiki.org/wiki/API:Properties#redirects_.2F_rd';
266 }
267 }