3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 * http://www.gnu.org/copyleft/gpl.html
21 namespace MediaWiki\Specials
;
23 use MediaWiki\Cache\LinkBatchFactory
;
24 use MediaWiki\Content\IContentHandlerFactory
;
25 use MediaWiki\Html\Html
;
26 use MediaWiki\SpecialPage\QueryPage
;
27 use MediaWiki\Title\Title
;
30 use Wikimedia\Rdbms\IConnectionProvider
;
31 use Wikimedia\Rdbms\IDatabase
;
32 use Wikimedia\Rdbms\IResultWrapper
;
35 * List of redirects to another redirecting page.
37 * The software will by default not follow double redirects, to prevent loops.
38 * Editors are encouraged to fix these, and can discover them via this page.
40 * @ingroup SpecialPage
42 class SpecialDoubleRedirects
extends QueryPage
{
44 private IContentHandlerFactory
$contentHandlerFactory;
45 private LinkBatchFactory
$linkBatchFactory;
48 * @param IContentHandlerFactory $contentHandlerFactory
49 * @param LinkBatchFactory $linkBatchFactory
50 * @param IConnectionProvider $dbProvider
52 public function __construct(
53 IContentHandlerFactory
$contentHandlerFactory,
54 LinkBatchFactory
$linkBatchFactory,
55 IConnectionProvider
$dbProvider
57 parent
::__construct( 'DoubleRedirects' );
58 $this->contentHandlerFactory
= $contentHandlerFactory;
59 $this->linkBatchFactory
= $linkBatchFactory;
60 $this->setDatabaseProvider( $dbProvider );
63 public function isExpensive() {
67 public function isSyndicated() {
71 protected function sortDescending() {
75 protected function getPageHeader() {
76 return $this->msg( 'doubleredirectstext' )->parseAsBlock();
79 private function reallyGetQueryInfo( $namespace = null, $title = null ) {
80 $limitToTitle = !( $namespace === null && $title === null );
89 'namespace' => 'pa.page_namespace',
90 'title' => 'pa.page_title',
92 'b_namespace' => 'pb.page_namespace',
93 'b_title' => 'pb.page_title',
94 'b_fragment' => 'ra.rd_fragment',
96 // Select fields from redirect instead of page. Because there may
97 // not actually be a page table row for this target (e.g. for interwiki redirects)
98 'c_namespace' => 'rb.rd_namespace',
99 'c_title' => 'rb.rd_title',
100 'c_fragment' => 'rb.rd_fragment',
101 'c_interwiki' => 'rb.rd_interwiki',
104 'ra.rd_from = pa.page_id',
106 // Filter out redirects where the target goes interwiki (T42353).
107 // This isn't an optimization, it is required for correct results,
108 // otherwise a non-double redirect like Bar -> w:Foo will show up
109 // like "Bar -> Foo -> w:Foo".
110 'ra.rd_interwiki' => '',
112 'pb.page_namespace = ra.rd_namespace',
113 'pb.page_title = ra.rd_title',
115 'rb.rd_from = pb.page_id',
119 if ( $limitToTitle ) {
120 $retval['conds']['pa.page_namespace'] = $namespace;
121 $retval['conds']['pa.page_title'] = $title;
127 public function getQueryInfo() {
128 return $this->reallyGetQueryInfo();
131 protected function getOrderFields() {
132 return [ 'ra.rd_namespace', 'ra.rd_title' ];
137 * @param stdClass $result Result row
140 public function formatResult( $skin, $result ) {
141 // If no Title B or C is in the query, it means this came from
142 // querycache (which only saves the 3 columns for title A).
143 // That does save the bulk of the query cost, but now we need to
144 // get a little more detail about each individual entry quickly
145 // using the filter of reallyGetQueryInfo.
148 if ( isset( $result->b_namespace
) ) {
151 $qi = $this->reallyGetQueryInfo(
155 $deep = $this->getDatabaseProvider()->getReplicaDatabase()->newSelectQueryBuilder()
157 ->caller( __METHOD__
)
162 $titleA = Title
::makeTitle( $result->namespace, $result->title
);
164 $linkRenderer = $this->getLinkRenderer();
166 return '<del>' . $linkRenderer->makeLink( $titleA, null, [], [ 'redirect' => 'no' ] ) . '</del>';
169 // if the page is editable, add an edit link
171 // check user permissions
172 $this->getAuthority()->isAllowed( 'edit' ) &&
173 // check, if the content model is editable through action=edit
174 $this->contentHandlerFactory
->getContentHandler( $titleA->getContentModel() )
175 ->supportsDirectEditing()
177 $edit = $linkRenderer->makeKnownLink(
179 $this->msg( 'parentheses', $this->msg( 'editlink' )->text() )->text(),
181 [ 'action' => 'edit' ]
187 $arrow = $this->getLanguage()->getArrow();
188 $contentLanguage = $this->getContentLanguage();
190 'dir' => $contentLanguage->getDir(),
191 'lang' => $contentLanguage->getHtmlCode(),
193 $linkA = Html
::rawElement( 'bdi', $bdiAttrs, $linkRenderer->makeKnownLink(
197 [ 'redirect' => 'no' ]
200 $titleB = Title
::makeTitle( $deep->b_namespace
, $deep->b_title
);
201 // We show fragment, but don't link to it, as it probably doesn't exist anymore.
202 $titleBFrag = Title
::makeTitle( $deep->b_namespace
, $deep->b_title
, $deep->b_fragment
);
203 $linkB = Html
::rawElement( 'bdi', $bdiAttrs, $linkRenderer->makeKnownLink(
205 $titleBFrag->getFullText(),
207 [ 'redirect' => 'no' ]
210 $titleC = Title
::makeTitle(
216 $linkC = Html
::rawElement( 'bdi', $bdiAttrs,
217 $linkRenderer->makeKnownLink( $titleC, $titleC->getFullText() )
220 return ( "{$linkA} {$edit} {$arrow} {$linkB} {$arrow} {$linkC}" );
223 public function execute( $par ) {
224 $this->addHelpLink( 'Help:Redirects' );
225 parent
::execute( $par );
229 * Cache page content model and gender distinction for performance
231 * @param IDatabase $db
232 * @param IResultWrapper $res
234 public function preprocessResults( $db, $res ) {
235 if ( !$res->numRows() ) {
239 $batch = $this->linkBatchFactory
->newLinkBatch();
240 foreach ( $res as $row ) {
241 $batch->add( $row->namespace, $row->title
);
242 if ( isset( $row->b_namespace
) ) {
243 // lazy loaded when using cached results
244 $batch->add( $row->b_namespace
, $row->b_title
);
246 if ( isset( $row->c_interwiki
) && !$row->c_interwiki
) {
247 // lazy loaded when using cached result, not added when interwiki link
248 $batch->add( $row->c_namespace
, $row->c_title
);
253 // Back to start for display
257 protected function getGroupName() {
258 return 'maintenance';
262 /** @deprecated class alias since 1.41 */
263 class_alias( SpecialDoubleRedirects
::class, 'SpecialDoubleRedirects' );