From 25e94e788df723986225975c6a509a4501fa5bb7 Mon Sep 17 00:00:00 2001 From: Laurent David Date: Tue, 24 May 2022 15:42:41 +0200 Subject: [PATCH] CONTRIB-8920 moodle-cs: Deprecation alerts for capabilities * Add a simple deprecation warning for capabilities we might want to get rid of. --- .../Access/DeprecatedCapabilitySniff.php | 111 ++++++++++++++++++ moodle/ruleset.xml | 9 ++ moodle/tests/fixtures/moodle_access.php | 8 ++ moodle/tests/moodlestandard_test.php | 25 ++++ 4 files changed, 153 insertions(+) create mode 100644 moodle/Sniffs/Access/DeprecatedCapabilitySniff.php create mode 100644 moodle/tests/fixtures/moodle_access.php diff --git a/moodle/Sniffs/Access/DeprecatedCapabilitySniff.php b/moodle/Sniffs/Access/DeprecatedCapabilitySniff.php new file mode 100644 index 0000000..f9a158d --- /dev/null +++ b/moodle/Sniffs/Access/DeprecatedCapabilitySniff.php @@ -0,0 +1,111 @@ +. + +/** + * Checks that each file contains necessary login checks. + * + * @package local_codechecker + * @copyright 2022 Laurent David + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace MoodleCodeSniffer\moodle\Sniffs\Access; + +use PHP_CodeSniffer\Files\File; +use PHP_CodeSniffer\Sniffs\Sniff; + +class DeprecatedCapabilitySniff implements Sniff { + + // phpcs:disable moodle.NamingConventions.ValidVariableName.MemberNameUnderscore + /** + * If we try to check this capability, a warning will be shown. + * + * @var array + */ + public $capabilitiesWarningList = []; + + /** + * If we try to check this capability, an error will be shown. + * + * @var array + */ + public $capabilitiesErrorList = []; + + // phpcs:enable + + /** + * Access function type + * + * @var string[] + */ + public $accessfunctions = ['has_access', 'has_capability']; + + /** + * Register for open tag (only process once per file). + */ + public function register() { + return array(T_OPEN_PARENTHESIS); + } + + /** + * Processes php files and for required login checks if includeing config.php + * + * @param File $file The file being scanned. + * @param int $stackptr The position in the stack. + */ + public function process(File $file, $stackptr) { + $functionnamepos = $file->findPrevious(T_STRING, $stackptr - 1); + if ($functionnamepos === false) { + return; + } + $tokens = $file->getTokens(); + if (empty($tokens[$functionnamepos]) || !$this->is_an_access_function($tokens[$functionnamepos])) { + return; + } + + $alldeprecated = array_merge($this->capabilitiesErrorList, $this->capabilitiesWarningList); + $closeparenthesispos = $file->findNext(T_CLOSE_PARENTHESIS, $stackptr + 1); + if ($closeparenthesispos === false) { + // Live coding, parse error or not a function call. + return; + } + $closeparenthesistoken = $tokens[$closeparenthesispos]; + $starttoken = $closeparenthesistoken['parenthesis_opener'] + 1; + $endtoken = $closeparenthesistoken['parenthesis_closer']; + $values = $file->getTokensAsString($starttoken, $endtoken - $starttoken); + foreach ($alldeprecated as $capability) { + if (strpos($values, $capability) !== false) { + if (in_array($capability, $this->capabilitiesWarningList)) { + $file->addWarning("The following capability '$capability' will be deprecated soon.", $starttoken, + 'DeprecatedCapability'); + } else { + $file->addError("The following capability '$capability' has been deprecated.", $starttoken, + 'DeprecatedCapability'); + } + } + } + } + + /** + * Is the current name an access function + * + * @param array $token current token + * @return bool true if the current methodname is access function. + */ + protected function is_an_access_function(array $token) { + return in_array($token['content'], $this->accessfunctions); + } +} diff --git a/moodle/ruleset.xml b/moodle/ruleset.xml index 5112fc1..468a5f3 100644 --- a/moodle/ruleset.xml +++ b/moodle/ruleset.xml @@ -87,6 +87,15 @@ 0 + + + + + + + + + diff --git a/moodle/tests/fixtures/moodle_access.php b/moodle/tests/fixtures/moodle_access.php new file mode 100644 index 0000000..54c5f7d --- /dev/null +++ b/moodle/tests/fixtures/moodle_access.php @@ -0,0 +1,8 @@ +verify_cs_results(); } + + /** + * Test the moodle.Access.DeprecatedCapabilitySniff sniff. + * + * + * @covers \MoodleCodeSniffer\moodle\Sniffs\Access\DeprecatedCapabilitySniff + */ + public function test_moodle_deprecated_access() { + + // Define the standard, sniff and fixture to use. + $this->set_standard('moodle'); + $this->set_sniff('moodle.Access.DeprecatedCapability'); + $this->set_fixture(__DIR__ . '/fixtures/moodle_access.php'); + // Define expected results (errors and warnings). Format, array of: + // - line => problem. + $this->set_errors([ + 7 => 'The following capability \'moodle/site:useremail\' has been deprecated', + ]); + $this->set_warnings([ + 6 => 'The following capability \'moodle/course:useremail\' will be deprecated soon', + ]); + + // Let's do all the hard work! + $this->verify_cs_results(); + } }