MediaWiki master
LogEventsList.php
Go to the documentation of this file.
1<?php
26namespace MediaWiki\Logging;
27
28use InvalidArgumentException;
29use MapCacheLRU;
59use stdClass;
60
62 public const NO_ACTION_LINK = 1;
63 public const NO_EXTRA_USER_LINKS = 2;
64 public const USE_CHECKBOXES = 4;
65
67 public $flags;
68
72 protected $showTagEditUI;
73
77 private $linkRenderer;
78
80 private $hookRunner;
81
82 private LogFormatterFactory $logFormatterFactory;
83
85 private $tagsCache;
86
93 public function __construct( $context, $linkRenderer = null, $flags = 0 ) {
94 $this->setContext( $context );
95 $this->flags = $flags;
96 $this->showTagEditUI = ChangeTags::showTagEditingUI( $this->getAuthority() );
97 if ( $linkRenderer instanceof LinkRenderer ) {
98 $this->linkRenderer = $linkRenderer;
99 }
100 $services = MediaWikiServices::getInstance();
101 $this->hookRunner = new HookRunner( $services->getHookContainer() );
102 $this->logFormatterFactory = $services->getLogFormatterFactory();
103 $this->tagsCache = new MapCacheLRU( 50 );
104 }
105
110 protected function getLinkRenderer() {
111 if ( $this->linkRenderer !== null ) {
112 return $this->linkRenderer;
113 } else {
114 return MediaWikiServices::getInstance()->getLinkRenderer();
115 }
116 }
117
129 public function showOptions( $type = '', $year = 0, $month = 0, $day = 0 ) {
130 $formDescriptor = [];
131
132 / Basic selectors
133 $formDescriptor['type'] = $this->getTypeMenuDesc();
134 $formDescriptor['user'] = [
135 'class' => HTMLUserTextField::class,
136 'label-message' => 'specialloguserlabel',
137 'name' => 'user',
138 'ipallowed' => true,
139 'iprange' => true,
140 'external' => true,
141 ];
142 $formDescriptor['page'] = [
143 'class' => HTMLTitleTextField::class,
144 'label-message' => 'speciallogtitlelabel',
145 'name' => 'page',
146 'required' => false,
147 ];
148
149 / Title pattern, if allowed
150 if ( !$this->getConfig()->get( MainConfigNames::MiserMode ) ) {
151 $formDescriptor['pattern'] = [
152 'type' => 'check',
153 'label-message' => 'log-title-wildcard',
154 'name' => 'pattern',
155 ];
156 }
157
158 / Add extra inputs if any
159 $extraInputsDescriptor = $this->getExtraInputsDesc( $type );
160 if ( $extraInputsDescriptor ) {
161 $formDescriptor[ 'extra' ] = $extraInputsDescriptor;
162 }
163
164 / Date menu
165 $formDescriptor['date'] = [
166 'type' => 'date',
167 'label-message' => 'date',
168 'default' => $year && $month && $day ? sprintf( "%04d-%02d-%02d", $year, $month, $day ) : '',
169 ];
170
171 / Tag filter
172 $formDescriptor['tagfilter'] = [
173 'type' => 'tagfilter',
174 'name' => 'tagfilter',
175 'label-message' => 'tag-filter',
176 ];
177 $formDescriptor['tagInvert'] = [
178 'type' => 'check',
179 'name' => 'tagInvert',
180 'label-message' => 'invert',
181 'hide-if' => [ '===', 'tagfilter', '' ],
182 ];
183
184 / Filter checkboxes, when work on all logs
185 if ( $type === '' ) {
186 $formDescriptor['filters'] = $this->getFiltersDesc();
187 }
188
189 / Action filter
190 $allowedActions = $this->getConfig()->get( MainConfigNames::ActionFilteredLogs );
191 if ( isset( $allowedActions[$type] ) ) {
192 $formDescriptor['subtype'] = $this->getActionSelectorDesc( $type, $allowedActions[$type] );
193 }
194
195 $htmlForm = HTMLForm::factory( 'ooui', $formDescriptor, $this->getContext() );
196 $htmlForm
197 ->setTitle( SpecialPage::getTitleFor( 'Log' ) ) / Remove subpage
198 ->setSubmitTextMsg( 'logeventslist-submit' )
199 ->setMethod( 'GET' )
200 ->setWrapperLegendMsg( 'log' )
201 ->setFormIdentifier( 'logeventslist', true ) / T321154
202 / Set callback for data validation and log type description.
203 ->setSubmitCallback( static function ( $formData, $form ) {
204 $form->addPreHtml(
205 ( new LogPage( $formData['type'] ) )->getDescription()
206 ->setContext( $form->getContext() )->parseAsBlock()
207 );
208 return true;
209 } );
210
211 $result = $htmlForm->prepareForm()->trySubmit();
212 $htmlForm->displayForm( $result );
213 return $result === true || ( $result instanceof Status && $result->isGood() );
214 }
215
219 private function getFiltersDesc() {
220 $optionsMsg = [];
221 $filters = $this->getConfig()->get( MainConfigNames::FilterLogTypes );
222 foreach ( $filters as $type => $val ) {
223 $optionsMsg["logeventslist-{$type}-log"] = $type;
224 }
225 return [
226 'class' => HTMLMultiSelectField::class,
227 'label-message' => 'logeventslist-more-filters',
228 'flatlist' => true,
229 'options-messages' => $optionsMsg,
230 'default' => array_keys( array_intersect( $filters, [ false ] ) ),
231 ];
232 }
233
237 private function getTypeMenuDesc() {
238 $typesByName = [];
239 / Load the log names
240 foreach ( LogPage::validTypes() as $type ) {
241 $page = new LogPage( $type );
242 $pageText = $page->getName()->text();
243 if ( in_array( $pageText, $typesByName ) ) {
244 LoggerFactory::getInstance( 'translation-problem' )->error(
245 'The log type {log_type_one} has the same translation as {log_type_two} for {lang}. ' .
246 '{log_type_one} will not be displayed in the drop down menu on Special:Log.',
247 [
248 'log_type_one' => $type,
249 'log_type_two' => array_search( $pageText, $typesByName ),
250 'lang' => $this->getLanguage()->getCode(),
251 ]
252 );
253 continue;
254 }
255 if ( $this->getAuthority()->isAllowed( $page->getRestriction() ) ) {
256 $typesByName[$type] = $pageText;
257 }
258 }
259
260 asort( $typesByName );
261
262 / Always put "All public logs" on top
263 $public = $typesByName[''];
264 unset( $typesByName[''] );
265 $typesByName = [ '' => $public ] + $typesByName;
266
267 return [
268 'class' => HTMLSelectField::class,
269 'name' => 'type',
270 'options' => array_flip( $typesByName ),
271 ];
272 }
273
278 private function getExtraInputsDesc( $type ) {
279 if ( $type === 'suppress' ) {
280 return [
281 'type' => 'text',
282 'label-message' => 'revdelete-offender',
283 'name' => 'offender',
284 ];
285 } else {
286 / Allow extensions to add an extra input into the descriptor array.
287 $unused = ''; / Deprecated since 1.32, removed in 1.41
288 $formDescriptor = [];
289 $this->hookRunner->onLogEventsListGetExtraInputs( $type, $this, $unused, $formDescriptor );
290
291 return $formDescriptor;
292 }
293 }
294
301 private function getActionSelectorDesc( $type, $actions ) {
302 $actionOptions = [ 'log-action-filter-all' => '' ];
303
304 foreach ( $actions as $value => $_ ) {
305 $msgKey = "log-action-filter-$type-$value";
306 $actionOptions[ $msgKey ] = $value;
307 }
308
309 return [
310 'class' => HTMLSelectField::class,
311 'name' => 'subtype',
312 'options-messages' => $actionOptions,
313 'label-message' => 'log-action-filter-' . $type,
314 ];
315 }
316
320 public function beginLogEventsList() {
321 return "<ul class='mw-logevent-loglines'>\n";
322 }
323
327 public function endLogEventsList() {
328 return "</ul>\n";
329 }
330
335 public function logLine( $row ) {
336 $entry = DatabaseLogEntry::newFromRow( $row );
337 $formatter = $this->logFormatterFactory->newFromEntry( $entry );
338 $formatter->setContext( $this->getContext() );
339 $formatter->setShowUserToolLinks( !( $this->flags & self::NO_EXTRA_USER_LINKS ) );
340
341 $time = $this->getLanguage()->userTimeAndDate(
342 $entry->getTimestamp(),
343 $this->getUser()
344 );
345 / Link the time text to the specific log entry, see T207562
346 $timeLink = $this->getLinkRenderer()->makeKnownLink(
348 $time,
349 [],
350 [ 'logid' => $entry->getId() ]
351 );
352
353 $action = $formatter->getActionText();
354
355 if ( $this->flags & self::NO_ACTION_LINK ) {
356 $revert = '';
357 } else {
358 $revert = $formatter->getActionLinks();
359 if ( $revert != '' ) {
360 $revert = '<span class="mw-logevent-actionlink">' . $revert . '</span>';
361 }
362 }
363
364 $comment = $formatter->getComment();
365
366 / Some user can hide log items and have review links
367 $del = $this->getShowHideLinks( $row );
368
369 / Any tags...
370 [ $tagDisplay, $newClasses ] = $this->tagsCache->getWithSetCallback(
371 $this->tagsCache->makeKey(
372 $row->ts_tags ?? '',
373 $this->getUser()->getName(),
374 $this->getLanguage()->getCode()
375 ),
376 fn () => ChangeTags::formatSummaryRow(
377 $row->ts_tags,
378 'logevent',
379 $this->getContext()
380 )
381 );
382 $classes = array_merge(
383 [ 'mw-logline-' . $entry->getType() ],
384 $newClasses
385 );
386 $attribs = [
387 'data-mw-logid' => $entry->getId(),
388 'data-mw-logaction' => $entry->getFullType(),
389 ];
390 $ret = "$del $timeLink $action $comment $revert $tagDisplay";
391
392 / Let extensions add data
393 $ret .= Html::openElement( 'span', [ 'class' => 'mw-logevent-tool' ] );
394 / FIXME: this hook assumes that callers will only append to $ret value.
395 / In future this hook should be replaced with a new hook: LogTools that has a
396 / hook interface consistent with DiffTools and HistoryTools.
397 $this->hookRunner->onLogEventsListLineEnding( $this, $ret, $entry, $classes, $attribs );
398 $attribs = array_filter( $attribs,
399 [ Sanitizer::class, 'isReservedDataAttribute' ],
400 ARRAY_FILTER_USE_KEY
401 );
402 $ret .= Html::closeElement( 'span' );
403 $attribs['class'] = $classes;
404
405 return Html::rawElement( 'li', $attribs, $ret ) . "\n";
406 }
407
412 private function getShowHideLinks( $row ) {
413 / We don't want to see the links and
414 if ( $this->flags == self::NO_ACTION_LINK ) {
415 return '';
416 }
417
418 / If change tag editing is available to this user, return the checkbox
419 if ( $this->flags & self::USE_CHECKBOXES && $this->showTagEditUI ) {
420 return Xml::check(
421 'showhiderevisions',
422 false,
423 [ 'name' => 'ids[' . $row->log_id . ']' ]
424 );
425 }
426
427 / no one can hide items from the suppress log.
428 if ( $row->log_type == 'suppress' ) {
429 return '';
430 }
431
432 $del = '';
433 $authority = $this->getAuthority();
434 / Don't show useless checkbox to people who cannot hide log entries
435 if ( $authority->isAllowed( 'deletedhistory' ) ) {
436 $canHide = $authority->isAllowed( 'deletelogentry' );
437 $canViewSuppressedOnly = $authority->isAllowed( 'viewsuppressed' ) &&
438 !$authority->isAllowed( 'suppressrevision' );
439 $entryIsSuppressed = self::isDeleted( $row, LogPage::DELETED_RESTRICTED );
440 $canViewThisSuppressedEntry = $canViewSuppressedOnly && $entryIsSuppressed;
441 if ( $row->log_deleted || $canHide ) {
442 / Show checkboxes instead of links.
443 if ( $canHide && $this->flags & self::USE_CHECKBOXES && !$canViewThisSuppressedEntry ) {
444 / If event was hidden from sysops
445 if ( !self::userCan( $row, LogPage::DELETED_RESTRICTED, $authority ) ) {
446 $del = Xml::check( 'deleterevisions', false, [ 'disabled' => 'disabled' ] );
447 } else {
448 $del = Xml::check(
449 'showhiderevisions',
450 false,
451 [ 'name' => 'ids[' . $row->log_id . ']' ]
452 );
453 }
454 } else {
455 / If event was hidden from sysops
456 if ( !self::userCan( $row, LogPage::DELETED_RESTRICTED, $authority ) ) {
457 $del = Linker::revDeleteLinkDisabled( $canHide );
458 } else {
459 $query = [
460 'target' => SpecialPage::getTitleFor( 'Log', $row->log_type )->getPrefixedDBkey(),
461 'type' => 'logging',
462 'ids' => $row->log_id,
463 ];
464 $del = Linker::revDeleteLink(
465 $query,
466 $entryIsSuppressed,
467 $canHide && !$canViewThisSuppressedEntry
468 );
469 }
470 }
471 }
472 }
473
474 return $del;
475 }
476
483 public static function typeAction( $row, $type, $action ) {
484 $match = is_array( $type ) ?
485 in_array( $row->log_type, $type ) : $row->log_type == $type;
486 if ( $match ) {
487 $match = is_array( $action ) ?
488 in_array( $row->log_action, $action ) : $row->log_action == $action;
489 }
490
491 return $match;
492 }
493
503 public static function userCan( $row, $field, Authority $performer ) {
504 return self::userCanBitfield( $row->log_deleted, $field, $performer ) &&
505 self::userCanViewLogType( $row->log_type, $performer );
506 }
507
517 public static function userCanBitfield( $bitfield, $field, Authority $performer ) {
518 if ( $bitfield & $field ) {
519 if ( $bitfield & LogPage::DELETED_RESTRICTED ) {
520 return $performer->isAllowedAny( 'suppressrevision', 'viewsuppressed' );
521 } else {
522 return $performer->isAllowed( 'deletedhistory' );
523 }
524 }
525 return true;
526 }
527
536 public static function userCanViewLogType( $type, Authority $performer ) {
537 $logRestrictions = MediaWikiServices::getInstance()->getMainConfig()->get( MainConfigNames::LogRestrictions );
538 if ( isset( $logRestrictions[$type] ) && !$performer->isAllowed( $logRestrictions[$type] ) ) {
539 return false;
540 }
541 return true;
542 }
543
549 public static function isDeleted( $row, $field ) {
550 return ( $row->log_deleted & $field ) == $field;
551 }
552
580 public static function showLogExtract(
581 &$out, $types = [], $page = '', $user = '', $param = []
582 ) {
583 $defaultParameters = [
584 'lim' => 25,
585 'conds' => [],
586 'showIfEmpty' => true,
587 'msgKey' => [ '' ],
588 'wrap' => "$1",
589 'flags' => 0,
590 'useRequestParams' => false,
591 'useMaster' => false,
592 'extraUrlParams' => false,
593 'footerHtmlItems' => []
594 ];
595 # The + operator appends elements of remaining keys from the right
596 # handed array to the left handed, whereas duplicated keys are NOT overwritten.
597 $param += $defaultParameters;
598 # Convert $param array to individual variables
599 $lim = $param['lim'];
600 $conds = $param['conds'];
601 $showIfEmpty = $param['showIfEmpty'];
602 $msgKey = $param['msgKey'];
603 $wrap = $param['wrap'];
604 $flags = $param['flags'];
605 $extraUrlParams = $param['extraUrlParams'];
606
607 $useRequestParams = $param['useRequestParams'];
608 / @phan-suppress-next-line PhanRedundantCondition
609 if ( !is_array( $msgKey ) ) {
610 $msgKey = [ $msgKey ];
611 }
612
613 / ???
614 / @phan-suppress-next-line PhanRedundantCondition
615 if ( $out instanceof OutputPage ) {
616 $context = $out->getContext();
617 } else {
618 $context = RequestContext::getMain();
619 }
620
621 $services = MediaWikiServices::getInstance();
622 / FIXME: Figure out how to inject this
623 $linkRenderer = $services->getLinkRenderer();
624
625 # Insert list of top 50 (or top $lim) items
626 $loglist = new LogEventsList( $context, $linkRenderer, $flags );
627 $pager = new LogPager(
628 $loglist,
629 $types,
630 $user,
631 $page,
632 false,
633 $conds,
634 false,
635 false,
636 false,
637 '',
638 '',
639 0,
640 $services->getLinkBatchFactory(),
641 $services->getActorNormalization(),
642 $services->getLogFormatterFactory()
643 );
644 if ( !$useRequestParams ) {
645 # Reset vars that may have been taken from the request
646 $pager->mLimit = 50;
647 $pager->mDefaultLimit = 50;
648 $pager->mOffset = "";
649 $pager->mIsBackwards = false;
650 }
651
652 if ( $param['useMaster'] ) {
653 $pager->mDb = $services->getConnectionProvider()->getPrimaryDatabase();
654 }
655 / @phan-suppress-next-line PhanImpossibleCondition
656 if ( isset( $param['offset'] ) ) { # Tell pager to ignore WebRequest offset
657 $pager->setOffset( $param['offset'] );
658 }
659
660 / @phan-suppress-next-line PhanSuspiciousValueComparison
661 if ( $lim > 0 ) {
662 $pager->mLimit = $lim;
663 }
664 / Fetch the log rows and build the HTML if needed
665 $logBody = $pager->getBody();
666 $numRows = $pager->getNumRows();
667
668 $s = '';
669 $footerHtmlItems = [];
670
671 if ( $logBody ) {
672 if ( $msgKey[0] ) {
673 / @phan-suppress-next-line PhanParamTooFewUnpack Non-emptiness checked above
674 $msg = $context->msg( ...$msgKey );
675 if ( $page instanceof PageReference ) {
676 $msg->page( $page );
677 }
678 $s .= $msg->parseAsBlock();
679 }
680 $s .= $loglist->beginLogEventsList() .
681 $logBody .
682 $loglist->endLogEventsList();
683 / add styles for change tags
684 $context->getOutput()->addModuleStyles( 'mediawiki.interface.helpers.styles' );
685 } elseif ( $showIfEmpty ) {
686 $s = Html::rawElement( 'div', [ 'class' => 'mw-warning-logempty' ],
687 $context->msg( 'logempty' )->parse() );
688 }
689
690 if ( $page instanceof PageReference ) {
691 $titleFormatter = MediaWikiServices::getInstance()->getTitleFormatter();
692 $pageName = $titleFormatter->getPrefixedDBkey( $page );
693 } elseif ( $page != '' ) {
694 $pageName = $page;
695 } else {
696 $pageName = null;
697 }
698
699 if ( $numRows > $pager->mLimit ) { # Show "Full log" link
700 $urlParam = [];
701 if ( $pageName ) {
702 $urlParam['page'] = $pageName;
703 }
704
705 if ( $user != '' ) {
706 $urlParam['user'] = $user;
707 }
708
709 if ( !is_array( $types ) ) { # Make it an array, if it isn't
710 $types = [ $types ];
711 }
712
713 # If there is exactly one log type, we can link to Special:Log?type=foo
714 if ( count( $types ) == 1 ) {
715 $urlParam['type'] = $types[0];
716 }
717
718 / @phan-suppress-next-line PhanSuspiciousValueComparison
719 if ( $extraUrlParams !== false ) {
720 $urlParam = array_merge( $urlParam, $extraUrlParams );
721 }
722
723 $footerHtmlItems[] = $linkRenderer->makeKnownLink(
724 SpecialPage::getTitleFor( 'Log' ),
725 $context->msg( 'log-fulllog' )->text(),
726 [],
727 $urlParam
728 );
729 }
730 if ( $param['footerHtmlItems'] ) {
731 $footerHtmlItems = array_merge( $footerHtmlItems, $param['footerHtmlItems'] );
732 }
733 if ( $logBody && $footerHtmlItems ) {
734 $s .= '<ul class="mw-logevent-footer">';
735 foreach ( $footerHtmlItems as $item ) {
736 $s .= Html::rawElement( 'li', [], $item );
737 }
738 $s .= '</ul>';
739 }
740
741 if ( $logBody && $msgKey[0] ) {
742 / TODO: The condition above is weird. Should this be done in any other cases?
743 / Or is it always true in practice?
744
745 / Mark as interface language (T60685)
746 $dir = $context->getLanguage()->getDir();
747 $lang = $context->getLanguage()->getHtmlCode();
748 $s = Html::rawElement( 'div', [
749 'class' => "mw-content-$dir",
750 'dir' => $dir,
751 'lang' => $lang,
752 ], $s );
753
754 / Wrap in warning box
755 $s = Html::warningBox(
756 $s,
757 'mw-warning-with-logexcerpt'
758 );
759 / Add styles for warning box
760 $context->getOutput()->addModuleStyles( 'mediawiki.codex.messagebox.styles' );
761 }
762
763 / @phan-suppress-next-line PhanSuspiciousValueComparison
764 if ( $wrap != '' ) { / Wrap message in html
765 $s = str_replace( '$1', $s, $wrap );
766 }
767
768 /* hook can return false, if we don't want the message to be emitted (Wikia BugId:7093) */
769 $hookRunner = new HookRunner( $services->getHookContainer() );
770 if ( $hookRunner->onLogEventsListShowLogExtract( $s, $types, $pageName, $user, $param ) ) {
771 / $out can be either an OutputPage object or a String-by-reference
772 if ( $out instanceof OutputPage ) {
773 $out->addHTML( $s );
774 } else {
775 $out = $s;
776 }
777 }
778
779 return $numRows;
780 }
781
791 public static function getExcludeClause( $db, $audience = 'public', ?Authority $performer = null ) {
792 $logRestrictions = MediaWikiServices::getInstance()->getMainConfig()->get( MainConfigNames::LogRestrictions );
793
794 if ( $audience != 'public' && $performer === null ) {
795 throw new InvalidArgumentException(
796 'A User object must be given when checking for a user audience.'
797 );
798 }
799
800 / Reset the array, clears extra "where" clauses when $par is used
801 $hiddenLogs = [];
802
803 / Don't show private logs to unprivileged users
804 foreach ( $logRestrictions as $logType => $right ) {
805 if ( $audience == 'public' || !$performer->isAllowed( $right ) ) {
806 $hiddenLogs[] = $logType;
807 }
808 }
809 if ( count( $hiddenLogs ) == 1 ) {
810 return 'log_type != ' . $db->addQuotes( $hiddenLogs[0] );
811 } elseif ( $hiddenLogs ) {
812 return 'log_type NOT IN (' . $db->makeList( $hiddenLogs ) . ')';
813 }
814
815 return false;
816 }
817
832 public static function getBlockLogWarningBox(
833 DatabaseBlockStore $blockStore,
834 NamespaceInfo $namespaceInfo,
835 MessageLocalizer $localizer,
836 LinkRenderer $linkRenderer,
837 $user,
838 Title $title
839 ) {
840 if ( !$user ) {
841 return null;
842 }
843 $appliesToTitle = false;
844 $logTargetPage = '';
845 $blockTargetName = '';
846 $blocks = $blockStore->newListFromTarget( $user, $user, false,
847 DatabaseBlockStore::AUTO_NONE );
848 foreach ( $blocks as $block ) {
849 if ( $block->appliesToTitle( $title ) ) {
850 $appliesToTitle = true;
851 }
852 $blockTargetName = $block->getTargetName();
853 $logTargetPage = $namespaceInfo->getCanonicalName( NS_USER ) .
854 ':' . $blockTargetName;
855 }
856
857 / Show log extract if the user is sitewide blocked or is partially
858 / blocked and not allowed to edit their user page or user talk page
859 if ( !count( $blocks ) || !$appliesToTitle ) {
860 return null;
861 }
862 $msgKey = count( $blocks ) === 1
863 ? 'blocked-notice-logextract' : 'blocked-notice-logextract-multi';
864 $params = [
865 'lim' => 1,
866 'showIfEmpty' => false,
867 'msgKey' => [
868 $msgKey,
869 $user->getName(), # Support GENDER in notice
870 count( $blocks )
871 ],
872 ];
873 if ( count( $blocks ) > 1 ) {
874 $params['footerHtmlItems'] = [
875 $linkRenderer->makeKnownLink(
876 SpecialPage::getTitleFor( 'BlockList' ),
877 $localizer->msg( 'blocked-notice-list-link' )->text(),
878 [],
879 [ 'wpTarget' => $blockTargetName ]
880 ),
881 ];
882 }
883
884 $outString = '';
885 self::showLogExtract( $outString, 'block', $logTargetPage, '', $params );
886 return $outString ?: null;
887 }
888}
889
891class_alias( LogEventsList::class, 'LogEventsList' );
const NS_USER
Definition Defines.php:67
Store key-value entries in a size-limited in-memory LRU cache.
newListFromTarget( $specificTarget, $vagueTarget=null, $fromPrimary=false, $auto=self::AUTO_ALL)
This is similar to DatabaseBlockStore::newFromTarget, but it returns all the relevant blocks.
Recent changes tagging.
The simplest way of implementing IContextSource is to hold a RequestContext as a member variable and ...
setContext(IContextSource $context)
getContext()
Get the base IContextSource object.
Group all the pieces relevant to the context of a request into one instance.
Implements a text input field for page titles.
Implements a text input field for user names.
Object handling generic submission, CSRF protection, layout and other logic for UI forms in a reusabl...
Definition HTMLForm.php:209
This class provides an implementation of the core hook interfaces, forwarding hook calls to HookConta...
onLogEventsListShowLogExtract(&$s, $types, $page, $user, $param)
This hook is called before the string is added to OutputPage.
This class is a collection of static functions that serve two purposes:
Definition Html.php:56
static check( $name, $checked=false, array $attribs=[])
Convenience function to produce a checkbox (input element with type=checkbox)
Definition Html.php:672
Class that generates HTML for internal links.
makeKnownLink( $target, $text=null, array $extraAttribs=[], array $query=[])
Make a link that's styled as if the target page exists (usually a "blue link", although the styling m...
Some internal bits split of from Skin.php.
Definition Linker.php:62
Create PSR-3 logger objects.
static newFromRow( $row)
Constructs new LogEntry from database result row.
static isDeleted( $row, $field)
static getExcludeClause( $db, $audience='public', ?Authority $performer=null)
SQL clause to skip forbidden log types for this user.
showOptions( $type='', $year=0, $month=0, $day=0)
Show options for the log list.
__construct( $context, $linkRenderer=null, $flags=0)
static showLogExtract(&$out, $types=[], $page='', $user='', $param=[])
Show log extract.
static typeAction( $row, $type, $action)
static getBlockLogWarningBox(DatabaseBlockStore $blockStore, NamespaceInfo $namespaceInfo, MessageLocalizer $localizer, LinkRenderer $linkRenderer, $user, Title $title)
static userCanBitfield( $bitfield, $field, Authority $performer)
Determine if the current user is allowed to view a particular field of this log row,...
static userCanViewLogType( $type, Authority $performer)
Determine if the current user is allowed to view a particular field of this log row,...
static userCan( $row, $field, Authority $performer)
Determine if the current user is allowed to view a particular field of this log row,...
Class to simplify the use of log pages.
Definition LogPage.php:50
static validTypes()
Get the list of valid log types.
Definition LogPage.php:221
A class containing constants representing the names of configuration variables.
const LogRestrictions
Name constant for the LogRestrictions setting, for use with Config::get()
const ActionFilteredLogs
Name constant for the ActionFilteredLogs setting, for use with Config::get()
const FilterLogTypes
Name constant for the FilterLogTypes setting, for use with Config::get()
const MiserMode
Name constant for the MiserMode setting, for use with Config::get()
Service locator for MediaWiki core services.
static getInstance()
Returns the global default instance of the top level service locator.
This is one of the Core classes and should be read at least once by any new developers.
HTML sanitizer for MediaWiki.
Definition Sanitizer.php:46
The WebRequest class encapsulates getting at data passed in the URL or via a POSTed form,...
Parent class for all special pages.
static getTitleFor( $name, $subpage=false, $fragment='')
Get a localised Title object for a specified special page name If you don't need a full Title object,...
static getTitleValueFor( $name, $subpage=false, $fragment='')
Get a localised TitleValue object for a specified special page name.
Generic operation result class Has warning/error list, boolean status and arbitrary value.
Definition Status.php:54
This is a utility class for dealing with namespaces that encodes all the "magic" behaviors of them ba...
getCanonicalName( $index)
Returns the canonical (English) name for a given index.
Represents a title within MediaWiki.
Definition Title.php:78
Module of static functions for generating XML.
Definition Xml.php:37
isGood()
Returns whether the operation completed and didn't have any error or warnings.
Interface for objects which can provide a MediaWiki context on request.
Interface for objects (potentially) representing a page that can be viewable and linked to on a wiki.
This interface represents the authority associated with the current execution context,...
Definition Authority.php:37
isAllowed(string $permission, ?PermissionStatus $status=null)
Checks whether this authority has the given permission in general.
isAllowedAny(... $permissions)
Checks whether this authority has any of the given permissions in general.
Interface for objects representing user identity.
Interface for localizing messages in MediaWiki.
msg( $key,... $params)
This is the method for getting translated interface messages.
This program is free software; you can redistribute it and/or modify it under the terms of the GNU Ge...

Follow Lee on X/Twitter - Father, Husband, Serial builder creating AI, crypto, games & web tools. We are friends :) AI Will Come To Life!

Check out: eBank.nz (Art Generator) | Netwrck.com (AI Tools) | Text-Generator.io (AI API) | BitBank.nz (Crypto AI) | ReadingTime (Kids Reading) | RewordGame | BigMultiplayerChess | WebFiddle | How.nz | Helix AI Assistant