관리-도구
편집 파일: skipif.php
<?php use MongoDB\Driver\Command; use MongoDB\Driver\Manager; use MongoDB\Driver\ReadPreference; use MongoDB\Driver\Server; use MongoDB\Driver\Exception\ConnectionException; use MongoDB\Driver\Exception\RuntimeException; require_once __DIR__ . '/basic.inc'; require_once __DIR__ . '/tools.php'; /** * Disables SKIPIF caching (PHP 8.1+). */ function disable_skipif_caching() { static $skipifCachingDisabled; if (PHP_VERSION_ID < 80100) { return; } if (! isset($skipifCachingDisabled)) { $skipifCachingDisabled = true; register_shutdown_function(function() { echo "nocache\n"; }); } } /** * Skips the test if the topology is load balanced. */ function skip_if_load_balanced() { is_load_balanced(URI) and exit('skip topology is load balanced'); } /** * Skips the test if the topology is not load balanced. */ function skip_if_not_load_balanced() { is_load_balanced(URI) or exit('skip topology is not load balanced'); } /** * Skips the test if the topology is a sharded cluster. */ function skip_if_mongos() { is_mongos(URI) and exit('skip topology is a sharded cluster'); } /** * Skips the test if the topology contains multiple mongos nodes. * * This is particularly useful for tests that rely on configureFailPoint, since * randomized server selection can interfere with testing. */ function skip_if_multiple_mongos() { $manager = create_test_manager(); // Ensure SDAM is initialized before calling Manager::getServers() $manager->selectServer(new ReadPreference('nearest')); $mongosNodes = array_filter($manager->getServers(), function(Server $server) { return $server->getType() === Server::TYPE_MONGOS; }); if (count($mongosNodes) > 1) { exit('skip topology contains multiple mongos nodes'); } } /** * Skips the test if the topology is not a shard cluster. */ function skip_if_not_mongos() { is_mongos(URI) or exit('skip topology is not a sharded cluster'); } function skip_if_not_sharded_cluster_with_replica_set() { is_sharded_cluster_with_replica_set(URI) or exit('skip topology is not a sharded cluster with replica set'); } /** * Skips the test if the topology is a replica set. */ function skip_if_replica_set() { is_replica_set(URI) and exit('skip topology is a replica set'); } /** * Skips the test if the topology is not a replica set. */ function skip_if_not_replica_set() { is_replica_set(URI) or exit('skip topology is not a replica set'); } /** * Skips the test if the topology is not a replica set or sharded cluster backed by replica sets */ function skip_if_not_replica_set_or_sharded_cluster_with_replica_set() { is_replica_set(URI) or is_sharded_cluster_with_replica_set(URI) or exit('skip topology is not a replica set or sharded cluster with replica set'); } function skip_if_no_transactions() { if (is_sharded_cluster_with_replica_set(URI)) { skip_if_server_version('<', '4.2'); } elseif (is_replica_set(URI)) { skip_if_server_version('<', '4.0'); } else { exit('skip topology does not support transactions'); } } /** * Skips the test if the topology has no arbiter. */ function skip_if_no_arbiter() { try { $primary = get_primary_server(URI); } catch (ConnectionException $e) { exit('skip primary server is not accessible: ' . $e->getMessage()); } $info = $primary->getInfo(); if (!isset($info['arbiters']) || count($info['arbiters']) < 1) { exit('skip no arbiters available'); } } /** * Skips the test if the topology has no secondary. */ function skip_if_no_secondary() { try { $primary = get_primary_server(URI); } catch (ConnectionException $e) { exit('skip primary server is not accessible: ' . $e->getMessage()); } $info = $primary->getInfo(); if (!isset($info['hosts']) || count($info['hosts']) < 2) { exit('skip no secondaries available'); } } /** * Skips the test if the topology does not have enough data carrying nodes */ function skip_if_not_enough_data_nodes($requiredNodes, $maxNodeCount = null) { try { $primary = get_primary_server(URI); } catch (ConnectionException $e) { exit('skip primary server is not accessible: ' . $e->getMessage()); } $info = $primary->getInfo(); $dataNodeCount = isset($info['hosts']) ? count($info['hosts']) : 0; if ($dataNodeCount < $requiredNodes) { exit("skip not enough nodes available (wanted: {$requiredNodes}, available: " . count($info['hosts']) . ')'); } if ($maxNodeCount !== null && $dataNodeCount > $requiredNodes) { exit("skip too many nodes available (wanted: {$requiredNodes}, available: " . count($info['hosts']) . ')'); } } /** * Skips the test if the topology does not have enough nodes */ function skip_if_not_enough_nodes($requiredNodes, $maxNodeCount = null) { try { $primary = get_primary_server(URI); } catch (ConnectionException $e) { exit('skip primary server is not accessible: ' . $e->getMessage()); } $info = $primary->getInfo(); $nodeCount = (isset($info['hosts']) ? count($info['hosts']) : 0) + (isset($info['arbiters']) ? count($info['arbiters']) : 0); if ($nodeCount < $requiredNodes) { exit("skip not enough nodes available (wanted: {$requiredNodes}, available: " . count($info['hosts']) . ')'); } if ($maxNodeCount !== null && $nodeCount > $requiredNodes) { exit("skip too many nodes available (wanted: {$requiredNodes}, available: " . count($info['hosts']) . ')'); } } /** * Skips the test if the topology is a standalone. */ function skip_if_standalone() { is_standalone(URI) and exit('skip topology is a standalone'); } /** * Skips the test if the topology is not a standalone. */ function skip_if_not_standalone() { is_standalone(URI) or exit('skip topology is not a standalone'); } /** * Skips the test if the connection string uses SSL. */ function skip_if_ssl() { is_ssl(URI) and exit('skip URI is using SSL'); } /** * Skips the test if the connection string uses SSL. */ function skip_if_not_ssl() { is_ssl(URI) or exit('skip URI is not using SSL'); } /** * Skips the test if no SSL directory has been defined. */ function skip_if_no_ssl_dir() { $sslDir = getenv('SSL_DIR'); $sslDir !== false or exit('skip SSL_DIR environment variable not set'); $sslDir = realpath($sslDir); ($sslDir !== false && is_dir($sslDir)) or exit('skip SSL_DIR is not a valid directory'); } /** * Skips the test if the connection string is using auth. */ function skip_if_auth() { is_auth(URI) and exit('skip URI is using auth'); } /** * Skips the test if the connection string is not using auth. */ function skip_if_not_auth() { is_auth(URI) or exit('skip URI is not using auth'); } /** * Skips the test if the connection string is not using a particular * authMechanism. * * @param string $authMechanism */ function skip_if_not_auth_mechanism($authMechanism) { $uriAuthMechanism = get_uri_option(URI, 'authMechanism'); if ($uriAuthMechanism === null && $authMechanism !== null) { exit('skip URI is not using authMechanism'); } if ($uriAuthMechanism !== $authMechanism) { exit("skip URI authMechanism is '$uriAuthMechanism' (needed: '$authMechanism')"); } } /** * Skips the test if the server is not accessible. */ function skip_if_not_live() { try { get_primary_server(URI); } catch (ConnectionException $e) { exit('skip server is not accessible: ' . $e->getMessage()); } } /** * Skips the test if the server version satisfies a comparison. * * @see http://php.net/version_compare * @param string $operator Comparison operator * @param string $version Version to compare against */ function skip_if_server_version($operator, $version) { $serverVersion = get_server_version(URI); if (version_compare($serverVersion, $version, $operator)) { exit("skip Server version '$serverVersion' $operator '$version'"); } } /** * Skips the test if the PHP version satisfies a comparison. * * @see http://php.net/version_compare * @param string $operator Comparison operator * @param string $version Version to compare against */ function skip_if_php_version($operator, $version) { if (version_compare(PHP_VERSION, $version, $operator)) { exit("skip PHP version '" . PHP_VERSION . "' $operator '$version'"); } } /** * Skips the test if the server not using a particular storage engine. * * @param string $storageEngine Storage engine name */ function skip_if_not_server_storage_engine($storageEngine) { $serverStorageEngine = get_server_storage_engine(URI); if ($serverStorageEngine !== $storageEngine) { exit("skip Server storage engine is '$serverStorageEngine' (needed '$storageEngine')"); } } /** * Skips the test if the server does not support the sleep command. */ function skip_if_sleep_command_unavailable() { if (!command_works(URI, ['sleep' => 1, 'secs' => 1, 'w' => false])) { exit('skip sleep command not available'); } } /** * Skips the test if the server does not support test commands. */ function skip_if_test_commands_disabled() { if (!get_server_parameter(URI, 'enableTestCommands')) { exit('skip test commands are disabled'); } } /** * Skips the test if libmongoc does not support crypto. * * If one or more libaries are provided, additionally check that the reported * library is in that array. Possible values are "libcrypto", "Common Crypto", * and "CNG". * * @param array $libs Optional list of crypto libraries to require */ function skip_if_not_libmongoc_crypto(array $libs = []) { $lib = get_module_info('libmongoc crypto library'); if ($lib === null) { exit('skip libmongoc crypto is not enabled'); } if (!empty($libs) && !in_array($lib, $libs)) { exit('skip Needs libmongoc crypto library ' . implode(', ', $libs) . ', but found ' . $lib); } } /** * Skips the test if libmongoc does not support SSL. * * If one or more libaries are provided, additionally check that the reported * library is in that array. Possible values are "OpenSSL", "LibreSSL", * "Secure Transport", and "Secure Channel". * * @param array $libs Optional list of SSL libraries to require */ function skip_if_not_libmongoc_ssl(array $libs = []) { $lib = get_module_info('libmongoc SSL library'); if ($lib === null) { exit('skip libmongoc SSL is not enabled'); } if (!empty($libs) && !in_array($lib, $libs)) { exit('skip Needs libmongoc SSL library ' . implode(', ', $libs) . ', but found ' . $lib); } } /** * Skips the test if the driver was not compiled with support for FLE */ function skip_if_not_libmongocrypt() { $lib = get_module_info('libmongocrypt'); if ($lib === 'disabled') { exit('skip libmongocrypt is not enabled'); } } /** * Skips the test if the driver was compiled with support for FLE */ function skip_if_libmongocrypt() { $lib = get_module_info('libmongocrypt'); if ($lib !== 'disabled') { exit('skip libmongocrypt is enabled'); } } /** * Skips the test if the collection cannot be dropped. * * @param string $databaseName Database name * @param string $collectionName Collection name */ function skip_if_not_clean($databaseName = DATABASE_NAME, $collectionName = COLLECTION_NAME) { try { drop_collection(URI, $databaseName, $collectionName); } catch (RuntimeException $e) { exit("skip Could not drop '$databaseName.$collectionName': " . $e->getMessage()); } /* Since this function modifies the state of the database, we need it to run * each time before a test. */ disable_skipif_caching(); } function skip_if_no_getmore_failpoint() { $serverVersion = get_server_version(URI); if (version_compare($serverVersion, '4.0', '<')) { exit("skip Server version '$serverVersion' does not support a getMore failpoint'"); } } function skip_if_no_failcommand_failpoint() { skip_if_test_commands_disabled(); $serverVersion = get_server_version(URI); if (is_mongos(URI) && version_compare($serverVersion, '4.1.8', '<')) { exit("skip mongos version '$serverVersion' does not support 'failCommand' failpoint'"); } elseif (version_compare($serverVersion, '4.0', '<')) { exit("skip mongod version '$serverVersion' does not support 'failCommand' failpoint'"); } } function skip_if_no_mongo_orchestration() { $ctx = stream_context_create(['http' => ['timeout' => 0.5]]); $result = @file_get_contents(MONGO_ORCHESTRATION_URI, false, $ctx); /* Note: file_get_contents emits an E_WARNING on failure, which will be * caught by the error handler in basic-skipif.inc. In that case, this may * never be reached. */ if ($result === false) { exit("skip mongo-orchestration is not accessible: '" . MONGO_ORCHESTRATION_URI . "'"); } } function skip_if_crypt_shared() { // Intentionally ignore empty values for CRYPT_SHARED_LIB_PATH if (getenv('CRYPT_SHARED_LIB_PATH')) { exit('skip crypt_shared is available'); } } function skip_if_no_crypt_shared() { // Intentionally consider empty values for CRYPT_SHARED_LIB_PATH if ( ! getenv('CRYPT_SHARED_LIB_PATH')) { exit('skip crypt_shared is not available'); } }