MediaWiki master
mergeMessageFileList.php
Go to the documentation of this file.
1<?php
11# Start from scratch
16
17// @codeCoverageIgnoreStart
18define( 'MW_NO_EXTENSION_MESSAGES', 1 );
19
20require_once __DIR__ . '/Maintenance.php';
21// @codeCoverageIgnoreEnd
22
30 public function __construct() {
31 parent::__construct();
32 $this->addOption(
33 'list-file',
34 'A file containing a list of extension setup files, one per line.',
35 false,
36 true
37 );
38 $this->addOption( 'extensions-dir', 'Path where extensions can be found.', false, true );
39 $this->addOption( 'output', 'Send output to this file (omit for stdout)', false, true );
40 $this->addDescription( 'Merge $wgExtensionMessagesFiles and $wgMessagesDirs from ' .
41 ' various extensions to produce a single file listing all message files and dirs.'
42 );
43 }
44
45 public function execute() {
46 $config = $this->getConfig();
47 $extensionEntryPointListFiles = $config->get( MainConfigNames::ExtensionEntryPointListFiles );
48
49 if (
50 !count( $extensionEntryPointListFiles )
51 && !$this->hasOption( 'list-file' )
52 && !$this->hasOption( 'extensions-dir' )
53 ) {
54 $this->fatalError( "Either --list-file or --extensions-dir must be provided if " .
55 "\$wgExtensionEntryPointListFiles is not set" );
56 }
57
58 $setupFiles = [];
59
60 # Add setup files contained in the file passed to --list-file
61 if ( $this->hasOption( 'list-file' ) ) {
62 $extensionPaths = $this->readFile( $this->getOption( 'list-file' ) );
63 $setupFiles = array_merge( $setupFiles, $extensionPaths );
64 }
65
66 # Now find out files in a directory
67 if ( $this->hasOption( 'extensions-dir' ) ) {
68 # Allow multiple directories to be passed with ":" as a delimiter
69 $extDirs = explode( ':', $this->getOption( 'extensions-dir' ) );
70 foreach ( $extDirs as $extDir ) {
71 $entries = scandir( $extDir );
72 foreach ( $entries as $extName ) {
73 if ( $extName === '.' || $extName === '..' || !is_dir( "$extDir/$extName" ) ) {
74 continue;
75 }
76 $possibilities = [
77 "$extDir/$extName/extension.json",
78 "$extDir/$extName/skin.json",
79 ];
80 $found = false;
81 foreach ( $possibilities as $extFile ) {
82 if ( file_exists( $extFile ) ) {
83 $setupFiles[] = $extFile;
84 $found = true;
85 break;
86 }
87 }
88
89 if ( !$found ) {
90 $this->error( "Extension {$extName} in {$extDir} lacks expected entry point: " .
91 "extension.json or skin.json " .
92 "(PHP entry points are no longer supported by this script)." );
93 }
94 }
95 }
96 }
97
98 # Add setup files defined via configuration
99 foreach ( $extensionEntryPointListFiles as $points ) {
100 $extensionPaths = $this->readFile( $points );
101 $setupFiles = array_merge( $setupFiles, $extensionPaths );
102 }
103
104 $this->generateMessageFileList( $setupFiles );
105 }
106
107 public function finalSetup( SettingsBuilder $settingsBuilder ) {
108 # This script commonly needs to be run before the l10n cache. But if
109 # LanguageCode is not 'en', it won't be able to run because there is
110 # no l10n cache. Break the cycle by forcing the LanguageCode setting to 'en'.
111 $settingsBuilder->putConfigValue( MainConfigNames::LanguageCode, 'en' );
112 parent::finalSetup( $settingsBuilder );
113 }
114
120 public function getDbType() {
121 return Maintenance::DB_NONE;
122 }
123
128 private function readFile( string $fileName ): array {
129 $files = [];
130 $fileLines = file( $fileName );
131 if ( $fileLines === false ) {
132 $this->error( "Unable to open list file $fileName." );
133
134 return $files;
135 }
136 # Strip comments, discard empty lines, and trim leading and trailing
137 # whitespace. Comments start with '#' and extend to the end of the line.
138 foreach ( $fileLines as $extension ) {
139 $extension = trim( preg_replace( '/#.*/', '', $extension ) );
140 if ( $extension !== '' ) {
141 # Paths may use a $IP placeholder string which we substitute here
142 $extension = str_replace( '$IP', MW_INSTALL_PATH, $extension );
143 if ( !str_ends_with( $extension, '.json' ) ) {
144 $this->error( "Extension {$extension} does not end with .json " .
145 "(PHP entry points are no longer supported by this script)" );
146 } elseif ( file_exists( $extension ) ) {
147 $files[] = $extension;
148 } else {
149 $this->error( "Extension {$extension} doesn't exist" );
150 }
151 }
152 }
153
154 return $files;
155 }
156
157 private function generateMessageFileList( array $setupFiles ): void {
158 $outputFile = $this->getOption( 'output' );
159 $quiet = $this->hasOption( 'quiet' );
160
161 $queue = [];
162 foreach ( $setupFiles as $fileName ) {
163 if ( (string)$fileName === '' ) {
164 continue;
165 }
166 if ( !$quiet ) {
167 fwrite( STDERR, "Loading data from $fileName\n" );
168 }
169 $queue[$fileName] = 1;
170 }
171
172 $config = $this->getConfig();
173 $vars = [
174 'wgExtensionMessagesFiles' => $config->get( MainConfigNames::ExtensionMessagesFiles ),
175 'wgMessagesDirs' => $config->get( MainConfigNames::MessagesDirs ),
176 ];
177
178 if ( $queue ) {
179 $registry = new ExtensionRegistry();
180 $data = $registry->readFromQueue( $queue );
181 foreach ( [ 'wgExtensionMessagesFiles', 'wgMessagesDirs' ] as $var ) {
182 if ( isset( $data['globals'][$var] ) ) {
183 $vars[$var] = array_merge( $data['globals'][$var], $vars[$var] );
184 }
185 }
186 }
187
188 if ( !$quiet ) {
189 fwrite( STDERR, "\n" );
190 }
191 $s =
192 "<?php\n" .
193 "## This file is generated by mergeMessageFileList.php. Do not edit it directly.\n\n" .
194 "if ( defined( 'MW_NO_EXTENSION_MESSAGES' ) ) return;\n\n" .
195 '$wgExtensionMessagesFiles = ' . var_export( $vars['wgExtensionMessagesFiles'], true ) . ";\n\n" .
196 '$wgMessagesDirs = ' . var_export( $vars['wgMessagesDirs'], true ) . ";\n\n";
197
198 $dirs = [
199 MW_INSTALL_PATH,
200 dirname( __DIR__ ),
201 realpath( MW_INSTALL_PATH )
202 ];
203 foreach ( $dirs as $dir ) {
204 $s = preg_replace( "/'" . preg_quote( $dir, '/' ) . "([^']*)'/", '"$IP\1"', $s );
205 }
206
207 if ( $outputFile !== null ) {
208 $res = file_put_contents( $outputFile, $s );
209 if ( $res === false ) {
210 fwrite( STDERR, "Failed to write to $outputFile\n" );
211 exit( 1 );
212 }
213 } else {
214 echo $s;
215 }
216 }
217}
218
219// @codeCoverageIgnoreStart
220$maintClass = MergeMessageFileList::class;
221require_once RUN_MAINTENANCE_IF_MAIN;
222// @codeCoverageIgnoreEnd
A class containing constants representing the names of configuration variables.
Abstract maintenance class for quickly writing and churning out maintenance scripts with minimal effo...
fatalError( $msg, $exitCode=1)
Output a message and terminate the current script.
addOption( $name, $description, $required=false, $withArg=false, $shortName=false, $multiOccurrence=false)
Add a parameter to the script.
hasOption( $name)
Checks to see if a particular option was set.
getOption( $name, $default=null)
Get an option, or return the default.
error( $err, $die=0)
Throw an error to the user.
addDescription( $text)
Set the description text.
Load JSON files, and uses a Processor to extract information.
Builder class for constructing a Config object from a set of sources during bootstrap.
putConfigValue(string $key, $value)
Puts a value into a config variable.
Maintenance script that merges $wgExtensionMessagesFiles from various extensions to produce a single ...
getDbType()
Database access is not needed.
execute()
Do the actual work.
finalSetup(SettingsBuilder $settingsBuilder)
Handle some last-minute setup here.
__construct()
Default constructor.