Why: * Maintenance scripts in core have bolierplate code that is added before and after the class to allow directly running the maintenance script. * Running the maintenance script directly has been deprecated since 1.40, so this boilerplate code is only to support a now deprecated method of running maintenance scripts. * This code cannot also be marked as covered, due to PHPUnit not recognising code coverage for files. * Therefore, it is best to ignore this boilerplate code in code coverage reports as it cannot be marked as covered and also is for deprecated code. What: * Wrap the boilerplate code (requiring Maintenance.php and then later defining the maintenance script class and running if the maintenance script was called directly) with @codeCoverageIgnore comments. * Some files use a different boilerplate code, however, these should also be marked as ignored for coverage for the same reason that coverage is not properly reported for files. Bug: T371167 Change-Id: I32f5c6362dfb354149a48ce9c28da9a7fc494f7c
222 lines
6 KiB
PHP
222 lines
6 KiB
PHP
<?php
|
|
/**
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
* http://www.gnu.org/copyleft/gpl.html
|
|
*
|
|
* @file
|
|
*/
|
|
|
|
// @codeCoverageIgnoreStart
|
|
require_once __DIR__ . '/Maintenance.php';
|
|
// @codeCoverageIgnoreEnd
|
|
|
|
use MediaWiki\MainConfigNames;
|
|
|
|
/**
|
|
* Diagnostic tool for interacting with memcached.
|
|
*
|
|
* @ingroup Maintenance
|
|
*/
|
|
class Mcc extends Maintenance {
|
|
public function __construct() {
|
|
parent::__construct();
|
|
$this->addDescription(
|
|
'MemCached Command (mcc) is an interactive CLI that lets you interact ' .
|
|
'with the MediaWiki memcached backends.'
|
|
);
|
|
$this->addOption( 'cache', 'Cache type to use (a key in $wgObjectCaches)', false, true );
|
|
$this->addOption( 'debug', 'Set debug mode on the memcached connection' );
|
|
}
|
|
|
|
protected function showHelp() {
|
|
parent::showHelp();
|
|
$this->output( "Interactive commands:\n " );
|
|
$this->output( str_replace( "\n", "\n ", $this->mccGetHelp( false ) ) );
|
|
$this->output( "\n" );
|
|
}
|
|
|
|
public function execute() {
|
|
$mcc = new MemcachedClient( [
|
|
'persistent' => true,
|
|
'debug' => $this->hasOption( 'debug' ),
|
|
] );
|
|
|
|
$config = $this->getConfig();
|
|
$objectCaches = $config->get( MainConfigNames::ObjectCaches );
|
|
$mainCacheType = $config->get( MainConfigNames::MainCacheType );
|
|
if ( $this->hasOption( 'cache' ) ) {
|
|
$cache = $this->getOption( 'cache' );
|
|
if ( !isset( $objectCaches[$cache] ) ) {
|
|
$this->fatalError( "MediaWiki isn't configured with a cache named '$cache'" );
|
|
}
|
|
$servers = $objectCaches[$cache]['servers'];
|
|
} elseif ( $mainCacheType === CACHE_MEMCACHED ) {
|
|
$mcc->set_servers( $config->get( MainConfigNames::MemCachedServers ) );
|
|
} elseif ( isset( $objectCaches[$mainCacheType]['servers'] ) ) {
|
|
$mcc->set_servers( $objectCaches[$mainCacheType]['servers'] );
|
|
} else {
|
|
$this->fatalError( "MediaWiki isn't configured for Memcached usage" );
|
|
}
|
|
|
|
do {
|
|
$bad = false;
|
|
$quit = false;
|
|
|
|
$line = self::readconsole();
|
|
if ( $line === false ) {
|
|
break;
|
|
}
|
|
|
|
$args = explode( ' ', $line );
|
|
$command = array_shift( $args );
|
|
|
|
// process command
|
|
switch ( $command ) {
|
|
case 'help':
|
|
// show a help message
|
|
print $this->mccGetHelp( array_shift( $args ) );
|
|
break;
|
|
|
|
case 'get':
|
|
$sub = '';
|
|
if ( array_key_exists( 1, $args ) ) {
|
|
$sub = $args[1];
|
|
}
|
|
print "Getting {$args[0]}[$sub]\n";
|
|
$res = $mcc->get( $args[0] );
|
|
if ( array_key_exists( 1, $args ) ) {
|
|
$res = $res[$args[1]];
|
|
}
|
|
if ( $res === false ) {
|
|
# print 'Error: ' . $mcc->error_string() . "\n";
|
|
print "MemCached error\n";
|
|
} elseif ( is_string( $res ) ) {
|
|
print "$res\n";
|
|
} else {
|
|
var_dump( $res );
|
|
}
|
|
break;
|
|
|
|
case 'getsock':
|
|
$res = $mcc->get( $args[0] );
|
|
$sock = $mcc->get_sock( $args[0] );
|
|
var_dump( $sock );
|
|
break;
|
|
|
|
case 'server':
|
|
if ( $mcc->_single_sock !== null ) {
|
|
print $mcc->_single_sock . "\n";
|
|
break;
|
|
}
|
|
$res = $mcc->get( $args[0] );
|
|
$hv = $mcc->_hashfunc( $args[0] );
|
|
for ( $i = 0; $i < 3; $i++ ) {
|
|
print $mcc->_buckets[$hv % $mcc->_bucketcount] . "\n";
|
|
$hv += $mcc->_hashfunc( $i . $args[0] );
|
|
}
|
|
break;
|
|
|
|
case 'set':
|
|
$key = array_shift( $args );
|
|
if ( $args[0] == "#" && is_numeric( $args[1] ) ) {
|
|
$value = str_repeat( '*', (int)$args[1] );
|
|
} else {
|
|
$value = implode( ' ', $args );
|
|
}
|
|
if ( !$mcc->set( $key, $value, 0 ) ) {
|
|
# print 'Error: ' . $mcc->error_string() . "\n";
|
|
print "MemCached error\n";
|
|
}
|
|
break;
|
|
|
|
case 'delete':
|
|
$key = implode( ' ', $args );
|
|
if ( !$mcc->delete( $key ) ) {
|
|
# print 'Error: ' . $mcc->error_string() . "\n";
|
|
print "MemCached error\n";
|
|
}
|
|
break;
|
|
|
|
case 'history':
|
|
if ( function_exists( 'readline_list_history' ) ) {
|
|
foreach ( readline_list_history() as $num => $line ) {
|
|
print "$num: $line\n";
|
|
}
|
|
} else {
|
|
print "readline_list_history() not available\n";
|
|
}
|
|
break;
|
|
|
|
case 'dumpmcc':
|
|
var_dump( $mcc );
|
|
break;
|
|
|
|
case 'quit':
|
|
case 'exit':
|
|
$quit = true;
|
|
break;
|
|
|
|
default:
|
|
$bad = true;
|
|
}
|
|
|
|
if ( $bad ) {
|
|
if ( $command ) {
|
|
print "Bad command\n";
|
|
}
|
|
} else {
|
|
if ( function_exists( 'readline_add_history' ) ) {
|
|
readline_add_history( $line );
|
|
}
|
|
}
|
|
} while ( !$quit );
|
|
}
|
|
|
|
private function mccGetHelp( $command ) {
|
|
$output = '';
|
|
$commandList = [
|
|
'get' => 'grabs something',
|
|
'getsock' => 'lists sockets',
|
|
'set' => 'changes something',
|
|
'delete' => 'deletes something',
|
|
'history' => 'show command line history',
|
|
'server' => 'show current memcached server',
|
|
'dumpmcc' => 'shows the whole thing',
|
|
'exit' => 'exit mcc',
|
|
'quit' => 'exit mcc',
|
|
'help' => 'help about a command',
|
|
];
|
|
if ( !$command ) {
|
|
$command = 'fullhelp';
|
|
}
|
|
if ( $command === 'fullhelp' ) {
|
|
$max_cmd_len = max( array_map( 'strlen', array_keys( $commandList ) ) );
|
|
foreach ( $commandList as $cmd => $desc ) {
|
|
$output .= sprintf( "%-{$max_cmd_len}s: %s\n", $cmd, $desc );
|
|
}
|
|
} elseif ( isset( $commandList[$command] ) ) {
|
|
$output .= "$command: $commandList[$command]\n";
|
|
} else {
|
|
$output .= "$command: command does not exist or no help for it\n";
|
|
}
|
|
|
|
return $output;
|
|
}
|
|
}
|
|
|
|
// @codeCoverageIgnoreStart
|
|
$maintClass = Mcc::class;
|
|
require_once RUN_MAINTENANCE_IF_MAIN;
|
|
// @codeCoverageIgnoreEnd
|