'ClassName' Load the specified class which subclasses Action * 'foo' => a callable Load the class returned by the callable * 'foo' => true Load the class FooAction which subclasses Action * 'foo' => false The action is disabled; show an error message * 'foo' => an object Use the specified object, which subclasses Action, useful for tests. * 'foo' => an array Slowly being used to replace the first three. The array * is treated as a specification for an ObjectFactory. */ private const CORE_ACTIONS = [ 'delete' => true, 'edit' => true, 'history' => true, 'protect' => true, 'purge' => true, 'render' => true, 'submit' => true, 'unprotect' => true, 'view' => true, // Beginning of actions switched to using DI with an ObjectFactory spec 'credits' => [ 'class' => CreditsAction::class, 'services' => [ 'LinkRenderer', 'UserFactory', ], ], 'editchangetags' => [ 'class' => SpecialPageAction::class, 'services' => [ 'SpecialPageFactory', ], 'args' => [ // SpecialPageAction is used for both 'editchangetags' and 'revisiondelete' // actions, tell it which one this is 'editchangetags', ], ], 'info' => [ 'class' => InfoAction::class, 'services' => [ 'ContentLanguage', 'HookContainer', 'LanguageNameUtils', 'LinkBatchFactory', 'LinkRenderer', 'DBLoadBalancer', 'MagicWordFactory', 'NamespaceInfo', 'PageProps', 'RepoGroup', 'RevisionLookup', 'MainWANObjectCache', 'WatchedItemStore', 'RedirectLookup', 'MainConfig' ], ], 'markpatrolled' => [ 'class' => MarkpatrolledAction::class, 'services' => [ 'LinkRenderer', ], ], 'mcrundo' => [ 'class' => McrUndoAction::class, 'services' => [ 'ReadOnlyMode', 'RevisionLookup', 'RevisionRenderer', 'MainConfig', ], ], 'mcrrestore' => [ 'class' => McrRestoreAction::class, 'services' => [ 'ReadOnlyMode', 'RevisionLookup', 'RevisionRenderer', 'MainConfig', ], ], 'raw' => [ 'class' => RawAction::class, 'services' => [ 'HookContainer', 'Parser', 'PermissionManager', 'RevisionLookup', ], ], 'revert' => [ 'class' => RevertAction::class, 'services' => [ 'ContentLanguage', 'RepoGroup', ], ], 'revisiondelete' => [ 'class' => SpecialPageAction::class, 'services' => [ 'SpecialPageFactory', ], 'args' => [ // SpecialPageAction is used for both 'editchangetags' and 'revisiondelete' // actions, tell it which one this is 'revisiondelete', ], ], 'rollback' => [ 'class' => RollbackAction::class, 'services' => [ 'ContentHandlerFactory', 'RollbackPageFactory', 'UserOptionsLookup', 'WatchlistManager', 'CommentFormatter' ], ], 'unwatch' => [ 'class' => UnwatchAction::class, 'services' => [ 'WatchlistManager', 'WatchedItemStore', ], ], 'watch' => [ 'class' => WatchAction::class, 'services' => [ 'WatchlistManager', 'WatchedItemStore', ], ], ]; /** * @param array $actionsConfig Configured actions (eg those added by extensions to $wgActions) * @param LoggerInterface $logger * @param ObjectFactory $objectFactory * @param HookContainer $hookContainer */ public function __construct( array $actionsConfig, LoggerInterface $logger, ObjectFactory $objectFactory, HookContainer $hookContainer ) { $this->actionsConfig = $actionsConfig; $this->logger = $logger; $this->objectFactory = $objectFactory; $this->hookRunner = new HookRunner( $hookContainer ); } /** * @param string $actionName should already be in all lowercase * @return string|callable|bool|Action|array|null The spec for the action, in any valid form, * based on $this->actionsConfig, or if not included there, CORE_ACTIONS, or null if the * action does not exist. */ private function getActionSpec( string $actionName ) { if ( isset( $this->actionsConfig[ $actionName ] ) ) { $this->logger->debug( '{actionName} is being set in configuration rather than CORE_ACTIONS', [ 'actionName' => $actionName ] ); return $this->actionsConfig[ $actionName ]; } return ( self::CORE_ACTIONS[ $actionName ] ?? null ); } /** * Get an appropriate Action subclass for the given action, * taking into account Article-specific overrides * * @param string $actionName * @param Article $article * @param IContextSource $context * @return Action|bool|null False if the action is disabled, null if not recognized */ public function getAction( string $actionName, Article $article, IContextSource $context ) { // Normalize to lowercase $actionName = strtolower( $actionName ); $spec = $this->getActionSpec( $actionName ); if ( $spec === false ) { // The action is disabled return $spec; } // Check action overrides even for nonexistent actions, so that actions // can exist just for a single content type. For Flow's convenience. $overrides = $article->getActionOverrides(); if ( isset( $overrides[ $actionName ] ) ) { // The Article class wants to override the action $spec = $overrides[ $actionName ]; $this->logger->debug( 'Overriding normal handler for {actionName}', [ 'actionName' => $actionName ] ); } if ( !$spec ) { // Either no such action exists (null) or the action is disabled // based on the article overrides (false) return $spec; } if ( $spec === true ) { // Old-style: use Action subclass based on name $spec = ucfirst( $actionName ) . 'Action'; } // $spec is either a class name, a callable, a specific object to use, or an // ObjectFactory spec. Convert to ObjectFactory spec, or return the specific object. if ( is_string( $spec ) ) { if ( !class_exists( $spec ) ) { $this->logger->info( 'Missing action class {actionClass}, treating as disabled', [ 'actionClass' => $spec ] ); return false; } // Class exists, can be used by ObjectFactory $spec = [ 'class' => $spec ]; } elseif ( is_callable( $spec ) ) { $spec = [ 'factory' => $spec ]; } elseif ( !is_array( $spec ) ) { // $spec is an object to use directly return $spec; } // ObjectFactory::createObject accepts an array, not just a callable (phan bug) // @phan-suppress-next-line PhanTypeInvalidCallableArrayKey $actionObj = $this->objectFactory->createObject( $spec, [ 'extraArgs' => [ $article, $context ], 'assertClass' => Action::class ] ); return $actionObj; } /** * Get the name of the action that will be executed, not necessarily the one * passed through the "action" request parameter. Actions disabled in * $wgActions will be replaced by "nosuchaction". * * @param IContextSource $context * @return string Action name */ public function getActionName( IContextSource $context ): string { // Trying to get a WikiPage for NS_SPECIAL etc. will result // in WikiPageFactory::newFromTitle throwing "Invalid or virtual namespace -1 given." // For SpecialPages et al, default to action=view. if ( !$context->canUseWikiPage() ) { return 'view'; } $request = $context->getRequest(); $actionName = $request->getRawVal( 'action', 'view' ); // Normalize to lowercase $actionName = strtolower( $actionName ); // Check for disabled actions if ( $this->getActionSpec( $actionName ) === false ) { // We could just set the action to 'nosuchaction' here and proceed, // but there should never be an action with the name 'nosuchaction' // and so getAction will return null, and then we would return // 'nosuchaction' anyway, so lets just return now return 'nosuchaction'; } // Workaround for T22966: inability of IE to provide an action dependent // on which submit button is clicked. if ( $actionName === 'historysubmit' ) { if ( $request->getBool( 'revisiondelete' ) ) { $actionName = 'revisiondelete'; } elseif ( $request->getBool( 'editchangetags' ) ) { $actionName = 'editchangetags'; } else { $actionName = 'view'; } } elseif ( $actionName === 'editredlink' ) { $actionName = 'edit'; } $this->hookRunner->onGetActionName( $context, $actionName ); $action = $this->getAction( $actionName, $this->getArticle( $context ), $context ); // Might not be an Action object if the action is not recognized (so $action could // be null) but should never be false because we already handled disabled actions // above. if ( $action instanceof Action ) { return $action->getName(); } return 'nosuchaction'; } /** * @deprecated since 1.38 * @param string $actionName * @return bool */ public function actionExists( string $actionName ): bool { wfDeprecated( __METHOD__, '1.38' ); // Normalize to lowercase $actionName = strtolower( $actionName ); // Null means no such action return ( $this->getActionSpec( $actionName ) !== null ); } /** * Protected to allow overriding with a partial mock in unit tests * * @codeCoverageIgnore * * @param IContextSource $context * @return Article */ protected function getArticle( IContextSource $context ): Article { return Article::newFromWikiPage( $context->getWikiPage(), $context ); } }