Update IPSet use statements
[lhc/web/wiklou.git] / maintenance / mergeMessageFileList.php
1 <?php
2 /**
3 * Merge $wgExtensionMessagesFiles from various extensions to produce a
4 * single array containing all message files.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 * http://www.gnu.org/copyleft/gpl.html
20 *
21 * @file
22 * @ingroup Maintenance
23 */
24
25 # Start from scratch
26 define( 'MW_NO_EXTENSION_MESSAGES', 1 );
27
28 require_once __DIR__ . '/Maintenance.php';
29 $maintClass = 'MergeMessageFileList';
30 $mmfl = false;
31
32 /**
33 * Maintenance script that merges $wgExtensionMessagesFiles from various
34 * extensions to produce a single array containing all message files.
35 *
36 * @ingroup Maintenance
37 */
38 class MergeMessageFileList extends Maintenance {
39 function __construct() {
40 parent::__construct();
41 $this->addOption(
42 'list-file',
43 'A file containing a list of extension setup files, one per line.',
44 false,
45 true
46 );
47 $this->addOption( 'extensions-dir', 'Path where extensions can be found.', false, true );
48 $this->addOption( 'output', 'Send output to this file (omit for stdout)', false, true );
49 $this->addDescription( 'Merge $wgExtensionMessagesFiles and $wgMessagesDirs from ' .
50 ' various extensions to produce a single file listing all message files and dirs.'
51 );
52 }
53
54 public function execute() {
55 // phpcs:ignore MediaWiki.NamingConventions.ValidGlobalName.wgPrefix
56 global $mmfl;
57 global $wgExtensionEntryPointListFiles;
58
59 if ( !count( $wgExtensionEntryPointListFiles )
60 && !$this->hasOption( 'list-file' )
61 && !$this->hasOption( 'extensions-dir' )
62 ) {
63 $this->fatalError( "Either --list-file or --extensions-dir must be provided if " .
64 "\$wgExtensionEntryPointListFiles is not set" );
65 }
66
67 $mmfl = [ 'setupFiles' => [] ];
68
69 # Add setup files contained in file passed to --list-file
70 if ( $this->hasOption( 'list-file' ) ) {
71 $extensionPaths = $this->readFile( $this->getOption( 'list-file' ) );
72 $mmfl['setupFiles'] = array_merge( $mmfl['setupFiles'], $extensionPaths );
73 }
74
75 # Now find out files in a directory
76 if ( $this->hasOption( 'extensions-dir' ) ) {
77 $extdir = $this->getOption( 'extensions-dir' );
78 # Allow multiple directories to be passed with ":" as delimiter
79 $extdirs = explode( ':', $extdir );
80 $entries = [];
81 foreach ( $extdirs as $extdir ) {
82 $entries = scandir( $extdir );
83 foreach ( $entries as $extname ) {
84 if ( $extname == '.' || $extname == '..' || !is_dir( "$extdir/$extname" ) ) {
85 continue;
86 }
87 $possibilities = [
88 "$extdir/$extname/extension.json",
89 "$extdir/$extname/skin.json",
90 "$extdir/$extname/$extname.php"
91 ];
92 $found = false;
93 foreach ( $possibilities as $extfile ) {
94 if ( file_exists( $extfile ) ) {
95 $mmfl['setupFiles'][] = $extfile;
96 $found = true;
97 break;
98 }
99 }
100
101 if ( !$found ) {
102 $this->error( "Extension {$extname} in {$extdir} lacks expected entry point: " .
103 "extension.json, skin.json, or {$extname}.php." );
104 }
105 }
106 }
107 }
108
109 # Add setup files defined via configuration
110 foreach ( $wgExtensionEntryPointListFiles as $points ) {
111 $extensionPaths = $this->readFile( $points );
112 $mmfl['setupFiles'] = array_merge( $mmfl['setupFiles'], $extensionPaths );
113 }
114
115 if ( $this->hasOption( 'output' ) ) {
116 $mmfl['output'] = $this->getOption( 'output' );
117 }
118 if ( $this->hasOption( 'quiet' ) ) {
119 $mmfl['quiet'] = true;
120 }
121 }
122
123 /**
124 * @param string $fileName
125 * @return array List of absolute extension paths
126 */
127 private function readFile( $fileName ) {
128 global $IP;
129
130 $files = [];
131 $fileLines = file( $fileName );
132 if ( $fileLines === false ) {
133 $this->hasError = true;
134 $this->error( "Unable to open list file $fileName." );
135
136 return $files;
137 }
138 # Strip comments, discard empty lines, and trim leading and trailing
139 # whitespace. Comments start with '#' and extend to the end of the line.
140 foreach ( $fileLines as $extension ) {
141 $extension = trim( preg_replace( '/#.*/', '', $extension ) );
142 if ( $extension !== '' ) {
143 # Paths may use the string $IP to be substituted by the actual value
144 $extension = str_replace( '$IP', $IP, $extension );
145 if ( file_exists( $extension ) ) {
146 $files[] = $extension;
147 } else {
148 $this->hasError = true;
149 $this->error( "Extension {$extension} doesn't exist" );
150 }
151 }
152 }
153
154 return $files;
155 }
156 }
157
158 require_once RUN_MAINTENANCE_IF_MAIN;
159
160 $queue = [];
161 foreach ( $mmfl['setupFiles'] as $fileName ) {
162 if ( strval( $fileName ) === '' ) {
163 continue;
164 }
165 if ( empty( $mmfl['quiet'] ) ) {
166 fwrite( STDERR, "Loading data from $fileName\n" );
167 }
168 // Using extension.json or skin.json
169 if ( substr( $fileName, -strlen( '.json' ) ) === '.json' ) {
170 $queue[$fileName] = 1;
171 } else {
172 require_once $fileName;
173 }
174 }
175
176 if ( $queue ) {
177 $registry = new ExtensionRegistry();
178 $data = $registry->readFromQueue( $queue );
179 foreach ( [ 'wgExtensionMessagesFiles', 'wgMessagesDirs' ] as $var ) {
180 if ( isset( $data['globals'][$var] ) ) {
181 $GLOBALS[$var] = array_merge( $data['globals'][$var], $GLOBALS[$var] );
182 }
183 }
184 }
185
186 fwrite( STDERR, "\n" );
187 $s =
188 "<" . "?php\n" .
189 "## This file is generated by mergeMessageFileList.php. Do not edit it directly.\n\n" .
190 "if ( defined( 'MW_NO_EXTENSION_MESSAGES' ) ) return;\n\n" .
191 '$wgExtensionMessagesFiles = ' . var_export( $wgExtensionMessagesFiles, true ) . ";\n\n" .
192 '$wgMessagesDirs = ' . var_export( $wgMessagesDirs, true ) . ";\n\n";
193
194 $dirs = [
195 $IP,
196 dirname( __DIR__ ),
197 realpath( $IP )
198 ];
199
200 foreach ( $dirs as $dir ) {
201 $s = preg_replace( "/'" . preg_quote( $dir, '/' ) . "([^']*)'/", '"$IP\1"', $s );
202 }
203
204 if ( isset( $mmfl['output'] ) ) {
205 file_put_contents( $mmfl['output'], $s );
206 } else {
207 echo $s;
208 }