Merge "Expose SearchEngine specific profiles"
[lhc/web/wiklou.git] / maintenance / convertExtensionToRegistration.php
1 <?php
2
3 require_once __DIR__ . '/Maintenance.php';
4
5 class ConvertExtensionToRegistration extends Maintenance {
6
7 protected $custom = [
8 'MessagesDirs' => 'handleMessagesDirs',
9 'ExtensionMessagesFiles' => 'handleExtensionMessagesFiles',
10 'AutoloadClasses' => 'removeAbsolutePath',
11 'ExtensionCredits' => 'handleCredits',
12 'ResourceModules' => 'handleResourceModules',
13 'ResourceModuleSkinStyles' => 'handleResourceModules',
14 'Hooks' => 'handleHooks',
15 'ExtensionFunctions' => 'handleExtensionFunctions',
16 'ParserTestFiles' => 'removeAbsolutePath',
17 ];
18
19 /**
20 * Things that were formerly globals and should still be converted
21 *
22 * @var array
23 */
24 protected $formerGlobals = [
25 'TrackingCategories',
26 ];
27
28 /**
29 * No longer supported globals (with reason) should not be converted and emit a warning
30 *
31 * @var array
32 */
33 protected $noLongerSupportedGlobals = [
34 'SpecialPageGroups' => 'deprecated', // Deprecated 1.21, removed in 1.26
35 ];
36
37 /**
38 * Keys that should be put at the top of the generated JSON file (T86608)
39 *
40 * @var array
41 */
42 protected $promote = [
43 'name',
44 'namemsg',
45 'version',
46 'author',
47 'url',
48 'description',
49 'descriptionmsg',
50 'license-name',
51 'type',
52 ];
53
54 private $json, $dir, $hasWarning = false;
55
56 public function __construct() {
57 parent::__construct();
58 $this->addDescription( 'Converts extension entry points to the new JSON registration format' );
59 $this->addArg( 'path', 'Location to the PHP entry point you wish to convert',
60 /* $required = */ true );
61 $this->addOption( 'skin', 'Whether to write to skin.json', false, false );
62 $this->addOption( 'config-prefix', 'Custom prefix for configuration settings', false, true );
63 }
64
65 protected function getAllGlobals() {
66 $processor = new ReflectionClass( 'ExtensionProcessor' );
67 $settings = $processor->getProperty( 'globalSettings' );
68 $settings->setAccessible( true );
69 return $settings->getValue() + $this->formerGlobals;
70 }
71
72 public function execute() {
73 // Extensions will do stuff like $wgResourceModules += array(...) which is a
74 // fatal unless an array is already set. So set an empty value.
75 // And use the weird $__settings name to avoid any conflicts
76 // with real poorly named settings.
77 $__settings = array_merge( $this->getAllGlobals(), array_keys( $this->custom ) );
78 foreach ( $__settings as $var ) {
79 $var = 'wg' . $var;
80 $$var = [];
81 }
82 unset( $var );
83 $arg = $this->getArg( 0 );
84 if ( !is_file( $arg ) ) {
85 $this->error( "$arg is not a file.", true );
86 }
87 require $arg;
88 unset( $arg );
89 // Try not to create any local variables before this line
90 $vars = get_defined_vars();
91 unset( $vars['this'] );
92 unset( $vars['__settings'] );
93 $this->dir = dirname( realpath( $this->getArg( 0 ) ) );
94 $this->json = [];
95 $globalSettings = $this->getAllGlobals();
96 $configPrefix = $this->getOption( 'config-prefix', 'wg' );
97 if ( $configPrefix !== 'wg' ) {
98 $this->json['config']['_prefix'] = $configPrefix;
99 }
100 foreach ( $vars as $name => $value ) {
101 $realName = substr( $name, 2 ); // Strip 'wg'
102 if ( $realName === false ) {
103 continue;
104 }
105
106 // If it's an empty array that we likely set, skip it
107 if ( is_array( $value ) && count( $value ) === 0 && in_array( $realName, $__settings ) ) {
108 continue;
109 }
110
111 if ( isset( $this->custom[$realName] ) ) {
112 call_user_func_array( [ $this, $this->custom[$realName] ],
113 [ $realName, $value, $vars ] );
114 } elseif ( in_array( $realName, $globalSettings ) ) {
115 $this->json[$realName] = $value;
116 } elseif ( array_key_exists( $realName, $this->noLongerSupportedGlobals ) ) {
117 $this->output( 'Warning: Skipped global "' . $name . '" (' .
118 $this->noLongerSupportedGlobals[$realName] . '). ' .
119 "Please update the entry point before convert to registration.\n" );
120 $this->hasWarning = true;
121 } elseif ( strpos( $name, $configPrefix ) === 0 ) {
122 // Most likely a config setting
123 $this->json['config'][substr( $name, strlen( $configPrefix ) )] = $value;
124 } elseif ( $configPrefix !== 'wg' && strpos( $name, 'wg' ) === 0 ) {
125 // Warn about this
126 $this->output( 'Warning: Skipped global "' . $name . '" (' .
127 'config prefix is "' . $configPrefix . '"). ' .
128 "Please check that this setting isn't needed.\n" );
129 }
130 }
131
132 // check, if the extension requires composer libraries
133 if ( $this->needsComposerAutoloader( dirname( $this->getArg( 0 ) ) ) ) {
134 // set the load composer autoloader automatically property
135 $this->output( "Detected composer dependencies, setting 'load_composer_autoloader' to true.\n" );
136 $this->json['load_composer_autoloader'] = true;
137 }
138
139 // Move some keys to the top
140 $out = [];
141 foreach ( $this->promote as $key ) {
142 if ( isset( $this->json[$key] ) ) {
143 $out[$key] = $this->json[$key];
144 unset( $this->json[$key] );
145 }
146 }
147 $out += $this->json;
148 // Put this at the bottom
149 $out['manifest_version'] = ExtensionRegistry::MANIFEST_VERSION;
150 $type = $this->hasOption( 'skin' ) ? 'skin' : 'extension';
151 $fname = "{$this->dir}/$type.json";
152 $prettyJSON = FormatJson::encode( $out, "\t", FormatJson::ALL_OK );
153 file_put_contents( $fname, $prettyJSON . "\n" );
154 $this->output( "Wrote output to $fname.\n" );
155 if ( $this->hasWarning ) {
156 $this->output( "Found warnings! Please resolve the warnings and rerun this script.\n" );
157 }
158 }
159
160 protected function handleExtensionFunctions( $realName, $value ) {
161 foreach ( $value as $func ) {
162 if ( $func instanceof Closure ) {
163 $this->error( "Error: Closures cannot be converted to JSON. " .
164 "Please move your extension function somewhere else.", 1
165 );
166 }
167 // check if $func exists in the global scope
168 if ( function_exists( $func ) ) {
169 $this->error( "Error: Global functions cannot be converted to JSON. " .
170 "Please move your extension function ($func) into a class.", 1
171 );
172 }
173 }
174
175 $this->json[$realName] = $value;
176 }
177
178 protected function handleMessagesDirs( $realName, $value ) {
179 foreach ( $value as $key => $dirs ) {
180 foreach ( (array)$dirs as $dir ) {
181 $this->json[$realName][$key][] = $this->stripPath( $dir, $this->dir );
182 }
183 }
184 }
185
186 protected function handleExtensionMessagesFiles( $realName, $value, $vars ) {
187 foreach ( $value as $key => $file ) {
188 $strippedFile = $this->stripPath( $file, $this->dir );
189 if ( isset( $vars['wgMessagesDirs'][$key] ) ) {
190 $this->output(
191 "Note: Ignoring PHP shim $strippedFile. " .
192 "If your extension no longer supports versions of MediaWiki " .
193 "older than 1.23.0, you can safely delete it.\n"
194 );
195 } else {
196 $this->json[$realName][$key] = $strippedFile;
197 }
198 }
199 }
200
201 private function stripPath( $val, $dir ) {
202 if ( $val === $dir ) {
203 $val = '';
204 } elseif ( strpos( $val, $dir ) === 0 ) {
205 // +1 is for the trailing / that won't be in $this->dir
206 $val = substr( $val, strlen( $dir ) + 1 );
207 }
208
209 return $val;
210 }
211
212 protected function removeAbsolutePath( $realName, $value ) {
213 $out = [];
214 foreach ( $value as $key => $val ) {
215 $out[$key] = $this->stripPath( $val, $this->dir );
216 }
217 $this->json[$realName] = $out;
218 }
219
220 protected function handleCredits( $realName, $value ) {
221 $keys = array_keys( $value );
222 $this->json['type'] = $keys[0];
223 $values = array_values( $value );
224 foreach ( $values[0][0] as $name => $val ) {
225 if ( $name !== 'path' ) {
226 $this->json[$name] = $val;
227 }
228 }
229 }
230
231 public function handleHooks( $realName, $value ) {
232 foreach ( $value as $hookName => &$handlers ) {
233 foreach ( $handlers as $func ) {
234 if ( $func instanceof Closure ) {
235 $this->error( "Error: Closures cannot be converted to JSON. " .
236 "Please move the handler for $hookName somewhere else.", 1
237 );
238 }
239 // Check if $func exists in the global scope
240 if ( function_exists( $func ) ) {
241 $this->error( "Error: Global functions cannot be converted to JSON. " .
242 "Please move the handler for $hookName inside a class.", 1
243 );
244 }
245 }
246 if ( count( $handlers ) === 1 ) {
247 $handlers = $handlers[0];
248 }
249 }
250 $this->json[$realName] = $value;
251 }
252
253 protected function handleResourceModules( $realName, $value ) {
254 $defaults = [];
255 $remote = $this->hasOption( 'skin' ) ? 'remoteSkinPath' : 'remoteExtPath';
256 foreach ( $value as $name => $data ) {
257 if ( isset( $data['localBasePath'] ) ) {
258 $data['localBasePath'] = $this->stripPath( $data['localBasePath'], $this->dir );
259 if ( !$defaults ) {
260 $defaults['localBasePath'] = $data['localBasePath'];
261 unset( $data['localBasePath'] );
262 if ( isset( $data[$remote] ) ) {
263 $defaults[$remote] = $data[$remote];
264 unset( $data[$remote] );
265 }
266 } else {
267 if ( $data['localBasePath'] === $defaults['localBasePath'] ) {
268 unset( $data['localBasePath'] );
269 }
270 if ( isset( $data[$remote] ) && isset( $defaults[$remote] )
271 && $data[$remote] === $defaults[$remote]
272 ) {
273 unset( $data[$remote] );
274 }
275 }
276 }
277
278 $this->json[$realName][$name] = $data;
279 }
280 if ( $defaults ) {
281 $this->json['ResourceFileModulePaths'] = $defaults;
282 }
283 }
284
285 protected function needsComposerAutoloader( $path ) {
286 $path .= '/composer.json';
287 if ( file_exists( $path ) ) {
288 // assume, that the composer.json file is in the root of the extension path
289 $composerJson = new ComposerJson( $path );
290 // check, if there are some dependencies in the require section
291 if ( $composerJson->getRequiredDependencies() ) {
292 return true;
293 }
294 }
295 return false;
296 }
297 }
298
299 $maintClass = 'ConvertExtensionToRegistration';
300 require_once RUN_MAINTENANCE_IF_MAIN;