%PDF- %PDF-
Direktori : /www/varak.net/wiki.varak.net/extensions/CirrusSearch/maintenance/ |
Current File : /www/varak.net/wiki.varak.net/extensions/CirrusSearch/maintenance/checkIndexes.php |
<?php namespace CirrusSearch; use CirrusSearch\Maintenance\Maintenance; /** * Check that all Cirrus indexes report OK. * * 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 */ $IP = getenv( 'MW_INSTALL_PATH' ); if( $IP === false ) { $IP = __DIR__ . '/../../..'; } require_once( "$IP/maintenance/Maintenance.php" ); require_once( __DIR__ . '/../includes/Maintenance/Maintenance.php' ); class CheckIndexes extends Maintenance { /** * @var array[] Nested array of arrays containing error strings. Individual * errors are nested based on the keys in self::$path at the time the error * occurred. */ private $errors = []; /** * @var string[] Represents each step of current indentation level */ private $path; /** * @var array Result of querying elasticsearch _cluster/state api endpoint */ private $clusterState; /** * @var array Version info stored in elasticsearch /mw_cirrus_versions/version */ private $cirrusInfo; public function __construct() { parent::__construct(); $this->mDescription = "Check that all Cirrus indexes report OK. This always operates on a single cluster."; $this->addOption( 'nagios', 'Output in nagios format' ); } public function execute() { if ( $this->hasOption( 'nagios' ) ) { // Force silent running mode so we can match Nagios expected output. $this->mQuiet = true; } $this->ensureClusterStateFetched(); $this->ensureCirrusInfoFetched(); // @todo: use MetaStoreIndex $this->checkIndex( 'mw_cirrus_versions', 1 ); $aliases = []; foreach ( $this->clusterState[ 'metadata' ][ 'indices' ] as $indexName => $data ) { foreach ( $data[ 'aliases' ] as $alias ) { $aliases[ $alias ][] = $indexName; } } foreach ( $this->cirrusInfo as $alias => $data ) { foreach ( $aliases[ $alias ] as $indexName ) { $this->checkIndex( $indexName, $data[ 'shard_count'] ); } } $indexCount = count( $this->cirrusInfo ); $errCount = count( $this->errors ); if ( $this->hasOption( 'nagios' ) ) { // Exit silent running mode so we can log Nagios style output $this->mQuiet = false; if ( $errCount > 0 ) { $this->output( "CIRRUSSEARCH CRITICAL - $indexCount indexes report $errCount errors\n" ); } else { $this->output( "CIRRUSSEARCH OK - $indexCount indexes report 0 errors\n" ); } } $this->printErrorRecursive( '', $this->errors ); // If there are error use the nagios error codes to signal them if ( $errCount > 0 ) { die( 2 ); } } /** * @param string $indexName * @param int */ private function checkIndex( $indexName, $expectedShardCount ) { $this->path = []; $metadata = $this->getIndexMetadata( $indexName ); $this->in( $indexName ); if ( $metadata === null ) { $this->err( "does not exist" ); return; } $this->check( 'state', 'open', $metadata[ 'state' ] ); // TODO check aliases $routingTable = $this->getIndexRoutingTable( $indexName ); $this->check( 'shard count', $expectedShardCount, count( $routingTable[ 'shards' ] ) ); foreach ( $routingTable[ 'shards' ] as $shardIndex => $shardRoutingTable ) { $this->in( "shard $shardIndex" ); foreach ( $shardRoutingTable as $replicaIndex => $replica ) { $this->in( "replica $replicaIndex" ); $this->check( 'state', [ 'STARTED', 'RELOCATING' ], $replica[ 'state' ] ); $this->out(); } $this->out(); } $this->out(); } /** * @param string $header */ private function in( $header ) { $this->path[] = $header; $this->output( str_repeat( "\t", count( $this->path ) - 1 ) ); $this->output( "$header...\n" ); } private function out() { array_pop( $this->path ); } /** * @param string $name * @param mixed $expected * @param mixed $actual */ private function check( $name, $expected, $actual ) { $this->output( str_repeat( "\t", count( $this->path ) ) ); $this->output( "$name..."); if ( is_array( $expected ) ) { if ( in_array( $actual, $expected ) ) { $this->output( "ok\n" ); } else { $expectedStr = implode( ', ', $expected ); $this->output( "$actual not in [$expectedStr]\n" ); $this->err( "expected $name to be in [$expectedStr] but was $actual" ); } } else { if ( $expected === $actual ) { $this->output( "ok\n" ); } else { $this->output( "$expected != $actual\n" ); $this->err( "expected $name to be '$expected' but was '$actual'" ); } } } /** * @param string $explanation */ private function err( $explanation ) { $err = $this->path; $err[] = $explanation; $e = &$this->errors; foreach ( $this->path as $element ) { $e = &$e[ $element ]; } $e[] = $explanation; } /** * @param string $indent Prefix to attach before each line of output * @param array $array */ private function printErrorRecursive( $indent, array $array ) { foreach ( $array as $key => $value ) { $line = $indent; if ( !is_numeric( $key ) ) { $line .= "$key..."; } if ( is_array( $value ) ) { $this->error( $line ); $this->printErrorRecursive( "$indent\t", $value ); } else { $line .= $value; if ( $this->hasOption( 'nagios' ) ) { $this->output( "$line\n" ); } else { $this->error( $line ); } } } } /** * @param string $indexName fully qualified name of elasticsearch index * @return array|null Index metadata from elasticsearch cluster state */ private function getIndexMetadata( $indexName ) { if ( isset( $this->clusterState[ 'metadata' ][ 'indices' ][ $indexName ] ) ) { return $this->clusterState[ 'metadata' ][ 'indices' ][ $indexName ]; } return null; } /** * @param string $indexName fully qualified name of elasticsearch index * @return array */ private function getIndexRoutingTable( $indexName ) { return $this->clusterState[ 'routing_table' ][ 'indices' ][ $indexName ]; } private function ensureClusterStateFetched() { if ( $this->clusterState === null ) { $this->clusterState = $this->getConnection()->getClient() ->request( '_cluster/state' )->getData(); } } private function ensureCirrusInfoFetched() { if ( $this->cirrusInfo === null ) { $query = new \Elastica\Query(); $query->setSize( 5000 ); $res = $this->getConnection()->getIndex( 'mw_cirrus_versions' )->getType( 'version' ) ->getIndex()->search( $query ); $this->cirrusInfo = []; foreach( $res as $r ) { $data = $r->getData(); $this->cirrusInfo[ $r->getId() ] = [ 'shard_count' => $data[ 'shard_count' ], ]; } } } } $maintClass = CheckIndexes::class; require_once RUN_MAINTENANCE_IF_MAIN;