Go to the documentation of this file.
24 require_once __DIR__ .
'/Maintenance.php';
35 private $mIgnorePaths =
array(), $mNoStyleCheckPaths =
array();
38 parent::__construct();
39 $this->mDescription =
"Check syntax for all PHP files in MediaWiki";
40 $this->
addOption(
'with-extensions',
'Also recurse the extensions folder' );
41 $this->
addOption(
'path',
'Specific path (file or directory) to check, either with absolute path or relative to the root of this MediaWiki installation',
43 $this->
addOption(
'list-file',
'Text file containing list of files or directories to check',
false,
true );
44 $this->
addOption(
'modified',
'Check only files that were modified (requires Git command-line client)' );
45 $this->
addOption(
'syntax-only',
'Check for syntax validity only, skip code style warnings' );
53 $this->buildFileList();
56 $useParseKit = function_exists(
'parsekit_compile_file' ) && version_compare( PHP_VERSION,
'5.3',
'<' );
58 $str =
'Checking syntax (using ' . ( $useParseKit ?
59 'parsekit' :
' php -l, this can take a long time' ) .
")\n";
61 foreach ( $this->mFiles
as $f ) {
63 $this->checkFileWithParsekit(
$f );
65 $this->checkFileWithCli(
$f );
67 if ( !$this->
hasOption(
'syntax-only' ) ) {
68 $this->checkForMistakes(
$f );
71 $this->
output(
"\nDone! " . count( $this->mFiles ) .
" files checked, " .
72 count( $this->mFailures ) .
" failures and " . count( $this->mWarnings ) .
73 " warnings found\n" );
79 private function buildFileList() {
82 $this->mIgnorePaths =
array(
84 "includes/NamespaceCompat.php$",
87 $this->mNoStyleCheckPaths =
array(
90 "EmailPage/PHPMailer",
91 "FCKeditor/fckeditor/",
103 if ( !$this->addPath(
$path ) ) {
104 $this->
error(
"Error: can't find file or directory $path\n",
true );
107 } elseif ( $this->
hasOption(
'list-file' ) ) {
110 $f = fopen(
$file,
'r' );
113 $this->
error(
"Can't open file $file\n",
true );
117 $this->addPath(
$path );
121 } elseif ( $this->
hasOption(
'modified' ) ) {
122 $this->
output(
"Retrieving list from Git... " );
123 $files = $this->getGitModifiedFiles(
$IP );
124 $this->
output(
"done\n" );
126 if ( $this->isSuitableFile(
$file ) && !is_dir(
$file ) ) {
127 $this->mFiles[] =
$file;
133 $this->
output(
'Building file list...',
'listfiles' );
141 $IP .
'/maintenance',
144 if ( $this->
hasOption(
'with-extensions' ) ) {
145 $dirs[] = $IP .
'/extensions';
149 $this->addDirectoryContent( $d );
153 if ( file_exists(
"$IP/LocalSettings.php" ) ) {
154 $this->mFiles[] =
"$IP/LocalSettings.php";
157 $this->
output(
'done.',
'listfiles' );
165 private function getGitModifiedFiles(
$path ) {
169 if ( !is_dir(
"$path/.git" ) ) {
170 $this->
error(
"Error: Not a Git repository!\n",
true );
174 $oldMaxShellMemory = $wgMaxShellMemory;
175 if ( $wgMaxShellMemory < 1024000 ) {
176 $wgMaxShellMemory = 1024000;
183 $cmd =
"cd $ePath && git merge-base master HEAD";
187 $this->
error(
"Error retrieving base SHA1 from Git!\n",
true );
192 $cmd =
"cd $ePath && git diff --name-only --diff-filter AM $eBase";
196 $this->
error(
"Error retrieving list from Git!\n",
true );
199 $wgMaxShellMemory = $oldMaxShellMemory;
202 $filename = strtok(
$output,
"\n" );
203 while ( $filename !==
false ) {
204 if ( $filename !==
'' ) {
205 $arr[] =
"$path/$filename";
207 $filename = strtok(
"\n" );
218 private function isSuitableFile(
$file ) {
220 $ext = pathinfo(
$file, PATHINFO_EXTENSION );
221 if (
$ext !=
'php' &&
$ext !=
'inc' &&
$ext !=
'php5' ) {
224 foreach ( $this->mIgnorePaths
as $regex ) {
226 if ( preg_match(
"~{$regex}~",
$file, $m ) ) {
238 private function addPath(
$path ) {
240 return $this->addFileOrDir(
$path ) || $this->addFileOrDir(
"$IP/$path" );
248 private function addFileOrDir(
$path ) {
249 if ( is_dir(
$path ) ) {
250 $this->addDirectoryContent(
$path );
251 } elseif ( file_exists(
$path ) ) {
252 $this->mFiles[] =
$path;
264 private function addDirectoryContent(
$dir ) {
265 $iterator =
new RecursiveIteratorIterator(
266 new RecursiveDirectoryIterator(
$dir ),
267 RecursiveIteratorIterator::SELF_FIRST
269 foreach ( $iterator
as $file ) {
270 if ( $this->isSuitableFile(
$file->getRealPath() ) ) {
271 $this->mFiles[] =
$file->getRealPath();
282 private function checkFileWithParsekit(
$file ) {
283 static $okErrors =
array(
284 'Redefining already defined constructor',
285 'Assigning the return value of new by reference is deprecated',
288 parsekit_compile_file(
$file, $errors, PARSEKIT_SIMPLE );
292 foreach ( $okErrors
as $okError ) {
293 if ( substr(
$error[
'errstr'], 0, strlen( $okError ) ) == $okError ) {
298 $this->
output(
"Error in $file line {$error['lineno']}: {$error['errstr']}\n" );
299 $this->mFailures[
$file] = $errors;
310 private function checkFileWithCli(
$file ) {
312 if ( strpos(
$res,
'No syntax errors detected' ) ===
false ) {
327 private function checkForMistakes(
$file ) {
328 foreach ( $this->mNoStyleCheckPaths
as $regex ) {
330 if ( preg_match(
"~{$regex}~",
$file, $m ) ) {
335 $text = file_get_contents(
$file );
336 $tokens = token_get_all( $text );
338 $this->checkEvilToken(
$file, $tokens,
'@',
'Error supression operator (@)' );
339 $this->checkRegex(
$file, $text,
'/^[\s\r\n]+<\?/',
'leading whitespace' );
340 $this->checkRegex(
$file, $text,
'/\?>[\s\r\n]*$/',
'trailing ?>' );
341 $this->checkRegex(
$file, $text,
'/^[\xFF\xFE\xEF]/',
'byte-order mark' );
344 private function checkRegex(
$file, $text, $regex, $desc ) {
345 if ( !preg_match( $regex, $text ) ) {
349 if ( !isset( $this->mWarnings[
$file] ) ) {
352 $this->mWarnings[
$file][] = $desc;
353 $this->
output(
"Warning in file $file: $desc found.\n" );
356 private function checkEvilToken(
$file, $tokens, $evilToken, $desc ) {
357 if ( !in_array( $evilToken, $tokens ) ) {
361 if ( !isset( $this->mWarnings[
$file] ) ) {
364 $this->mWarnings[
$file][] = $desc;
365 $this->
output(
"Warning in file $file: $desc found.\n" );
wfShellExec( $cmd, &$retval=null, $environ=array(), $limits=array(), $options=array())
Execute a shell command, with time and memory limits mirrored from the PHP configuration if supported...
skin txt MediaWiki includes four core it has been set as the default in MediaWiki since the replacing Monobook it had been been the default skin since before being replaced by Vector largely rewritten in while keeping its appearance Several legacy skins were removed in the as the burden of supporting them became too heavy to bear Those in etc for skin dependent CSS etc for skin dependent JavaScript These can also be customised on a per user by etc This feature has led to a wide variety of user styles becoming that gallery is a good place to ending in php
getDbType()
Does the script need different DB access? By default, we give Maintenance scripts normal rights to th...
addOption( $name, $description, $required=false, $withArg=false, $shortName=false)
Add a parameter to the script.
null means default in associative array with keys and values unescaped Should be merged with default with a value of false meaning to suppress the attribute in associative array with keys and values unescaped noclasses & $ret
wfSuppressWarnings( $end=false)
Reference-counted warning suppression.
require_once RUN_MAINTENANCE_IF_MAIN
Abstract maintenance class for quickly writing and churning out maintenance scripts with minimal effo...
public function __construct()
wfRestoreWarnings()
Restore error level to previous value.
the array() calling protocol came about after MediaWiki 1.4rc1.
List of Api Query prop modules.
when a variable name is used in a it is silently declared as a new masking the global
wfEscapeShellArg()
Windows-compatible version of escapeshellarg() Windows doesn't recognise single-quotes in the shell,...
const DB_NONE
Constants for DB access type.
if(PHP_SAPI !='cli') $file
getOption( $name, $default=null)
Get an option, or return the default.
if(count( $args)==0) $dir
This document is intended to provide useful advice for parties seeking to redistribute MediaWiki to end users It s targeted particularly at maintainers for Linux since it s been observed that distribution packages of MediaWiki often break We ve consistently had to recommend that users seeking support use official tarballs instead of their distribution s and this often solves whatever problem the user is having It would be nice if this could such as
error( $err, $die=0)
Throw an error to the user.
output( $out, $channel=null)
Throw some output to the user.
usually copyright or history_copyright This message must be in HTML not wikitext $subpages will be ignored and the rest of subPageSubtitle() will run. 'SkinTemplateBuildNavUrlsNav_urlsAfterPermalink' whether MediaWiki currently thinks this is a CSS JS page Hooks may change this value to override the return value of Title::isCssOrJsPage(). 'TitleIsAlwaysKnown' whether MediaWiki currently thinks this page is known isMovable() always returns false. $title whether MediaWiki currently thinks this page is movable Hooks may change this value to override the return value of Title::isMovable(). 'TitleIsWikitextPage' whether MediaWiki currently thinks this is a wikitext page Hooks may change this value to override the return value of Title::isWikitextPage() 'TitleMove' use UploadVerification and UploadVerifyFile instead where the first element is the message key and the remaining elements are used as parameters to the message based on mime etc Preferred in most cases over UploadVerification object with all info about the upload string as detected by MediaWiki Handlers will typically only apply for specific mime types object & $error
hasOption( $name)
Checks to see if a particular param exists.
public function execute()
please add to it if you re going to add events to the MediaWiki code where normally authentication against an external auth plugin would be creating a account incomplete not yet checked for validity & $retval