mirror of
https://github.com/macintoshplus/mongo-php-driver.git
synced 2026-03-24 00:42:15 +01:00
* Remove PHP 8.0 and 7.4 from CI matrix * Bump PHP requirement in config.m4 * Remove conditional dependencies in config.w32 * Bump minimum PHP version in package.xml * Remove conditional macro definitions * Remove conditional arginfo definitions * Directly implement Stringable interface * Remove conditional code * Remove skipped tests and obsolete SKIPIF statements in tests * Clarify removal requirement for return type checks * Remove obsolete serialisation macro * Remove unused macro for final classes * Make enum code unconditional
493 lines
13 KiB
PHP
493 lines
13 KiB
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 (! 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');
|
|
}
|
|
}
|