true], array $driverOptions = []) { if ($server === 'default') { $server = 'mongodb://' . self::DEFAULT_HOST . ':' . self::DEFAULT_PORT; } if (isset($options['readPreferenceTags'])) { $options['readPreferenceTags'] = [$this->getReadPreferenceTags($options['readPreferenceTags'])]; } $this->applyConnectionOptions($server, $options); $this->server = $server; if (false === strpos($this->server, '://')) { $this->server = 'mongodb://' . $this->server; } $this->client = new Client($this->server, $options, $driverOptions); $info = $this->client->__debugInfo(); $this->manager = $info['manager']; if (isset($options['connect']) && $options['connect']) { $this->connect(); } } /** * Closes this database connection * * @link http://www.php.net/manual/en/mongoclient.close.php * @param boolean|string $connection * @return boolean If the connection was successfully closed. */ public function close($connection = null) { $this->connected = false; return false; } /** * Connects to a database server * * @link http://www.php.net/manual/en/mongoclient.connect.php * * @throws MongoConnectionException * @return boolean If the connection was successful. */ public function connect() { $this->connected = true; return true; } /** * Drops a database * * @link http://www.php.net/manual/en/mongoclient.dropdb.php * @param mixed $db The database to drop. Can be a MongoDB object or the name of the database. * @return array The database response. * @deprecated Use MongoDB::drop() instead. */ public function dropDB($db) { return $this->selectDB($db)->drop(); } /** * Gets a database * * @link http://php.net/manual/en/mongoclient.get.php * @param string $dbname The database name. * @return MongoDB The database name. */ public function __get($dbname) { return $this->selectDB($dbname); } /** * Gets the client for this object * * @internal This part is not of the ext-mongo API and should not be used * @return Client */ public function getClient() { return $this->client; } /** * Get connections * * Returns an array of all open connections, and information about each of the servers * * @return array */ public static function getConnections() { return []; } /** * Get hosts * * This method is only useful with a connection to a replica set. It returns the status of all of the hosts in the * set. Without a replica set, it will just return an array with one element containing the host that you are * connected to. * * @return array */ public function getHosts() { $this->forceConnect(); $results = []; try { $servers = $this->manager->getServers(); } catch (\MongoDB\Driver\Exception\Exception $e) { throw ExceptionConverter::toLegacy($e); } foreach ($servers as $server) { $key = sprintf('%s:%d;-;.;%d', $server->getHost(), $server->getPort(), getmypid()); $info = $server->getInfo(); switch ($server->getType()) { case \MongoDB\Driver\Server::TYPE_RS_PRIMARY: $state = 1; break; case \MongoDB\Driver\Server::TYPE_RS_SECONDARY: $state = 2; break; default: $state = 0; } $results[$key] = [ 'host' => $server->getHost(), 'port' => $server->getPort(), 'health' => 1, 'state' => $state, 'ping' => $server->getLatency(), 'lastPing' => null, ]; } return $results; } /** * Kills a specific cursor on the server * * @link http://www.php.net/manual/en/mongoclient.killcursor.php * @param string $server_hash The server hash that has the cursor. * @param int|MongoInt64 $id The ID of the cursor to kill. * @return bool */ public function killCursor($server_hash, $id) { $this->notImplemented(); } /** * Lists all of the databases available * * @link http://php.net/manual/en/mongoclient.listdbs.php * @return array Returns an associative array containing three fields. The first field is databases, which in turn contains an array. Each element of the array is an associative array corresponding to a database, giving the database's name, size, and if it's empty. The other two fields are totalSize (in bytes) and ok, which is 1 if this method ran successfully. */ public function listDBs() { try { $databaseInfoIterator = $this->client->listDatabases(); } catch (\MongoDB\Driver\Exception\Exception $e) { throw ExceptionConverter::toLegacy($e); } $databases = [ 'databases' => [], 'totalSize' => 0, 'ok' => 1.0, ]; foreach ($databaseInfoIterator as $databaseInfo) { $databases['databases'][] = [ 'name' => $databaseInfo->getName(), 'empty' => $databaseInfo->isEmpty(), 'sizeOnDisk' => $databaseInfo->getSizeOnDisk(), ]; $databases['totalSize'] += $databaseInfo->getSizeOnDisk(); } return $databases; } /** * Gets a database collection * * @link http://www.php.net/manual/en/mongoclient.selectcollection.php * @param string $db The database name. * @param string $collection The collection name. * @return MongoCollection Returns a new collection object. * @throws Exception Throws Exception if the database or collection name is invalid. */ public function selectCollection($db, $collection) { return new MongoCollection($this->selectDB($db), $collection); } /** * Gets a database * * @link http://www.php.net/manual/en/mongo.selectdb.php * @param string $name The database name. * @return MongoDB Returns a new db object. * @throws InvalidArgumentException */ public function selectDB($name) { return new MongoDB($this, $name); } /** * {@inheritdoc} */ public function setReadPreference($readPreference, $tags = null) { return $this->setReadPreferenceFromParameters($readPreference, $tags); } /** * {@inheritdoc} */ public function setWriteConcern($wstring, $wtimeout = 0) { return $this->setWriteConcernFromParameters($wstring, $wtimeout); } /** * String representation of this connection * * @link http://www.php.net/manual/en/mongoclient.tostring.php * @return string Returns hostname and port for this connection. */ public function __toString() { return $this->server; } /** * Forces a connection by executing the ping command */ private function forceConnect() { try { $command = new \MongoDB\Driver\Command(['ping' => 1]); $this->manager->executeCommand('db', $command); } catch (\MongoDB\Driver\Exception\Exception $e) { throw ExceptionConverter::toLegacy($e); } } private function notImplemented() { throw new \Exception('Not implemented'); } /** * @return array */ public function __sleep() { return [ 'connected', 'status', 'server', 'persistent' ]; } /** * @param $server * @return array */ private function extractUrlOptions($server) { $queryOptions = parse_url($server, PHP_URL_QUERY); if (!$queryOptions) { return []; } $queryOptions = explode('&', $queryOptions); $options = []; foreach ($queryOptions as $option) { if (strpos($option, '=') === false) { continue; } $keyValue = explode('=', $option); if ($keyValue[0] === 'readPreferenceTags') { $options[$keyValue[0]][] = $this->getReadPreferenceTags($keyValue[1]); } elseif (ctype_digit($keyValue[1])) { $options[$keyValue[0]] = (int) $keyValue[1]; } else { $options[$keyValue[0]] = $keyValue[1]; } } return $options; } /** * @param $readPreferenceTagString * @return array */ private function getReadPreferenceTags($readPreferenceTagString) { $tagSets = []; foreach (explode(',', $readPreferenceTagString) as $index => $tagSet) { $tags = explode(':', $tagSet); $tagSets[$tags[0]] = $tags[1]; } return $tagSets; } /** * @param string $server * @param array $options */ private function applyConnectionOptions($server, array $options) { $urlOptions = $this->extractUrlOptions($server); if (isset($urlOptions['wTimeout'])) { $urlOptions['wTimeoutMS'] = $urlOptions['wTimeout']; unset($urlOptions['wTimeout']); } if (isset($options['wTimeout'])) { $options['wTimeoutMS'] = $options['wTimeout']; unset($options['wTimeout']); } // Special handling for readPreferenceTags which are merged if (isset($options['readPreferenceTags']) && isset($urlOptions['readPreferenceTags'])) { $options['readPreferenceTags'] = array_merge($urlOptions['readPreferenceTags'], $options['readPreferenceTags']); unset($urlOptions['readPreferenceTags']); } $urlOptions = array_merge($urlOptions, $options); if (isset($urlOptions['slaveOkay'])) { $this->setReadPreferenceFromSlaveOkay($urlOptions['slaveOkay']); } elseif (isset($urlOptions['readPreference']) || isset($urlOptions['readPreferenceTags'])) { $readPreference = isset($urlOptions['readPreference']) ? $urlOptions['readPreference'] : null; $tags = isset($urlOptions['readPreferenceTags']) ? $urlOptions['readPreferenceTags'] : null; $this->setReadPreferenceFromParameters($readPreference, $tags); } if (isset($urlOptions['w']) || isset($urlOptions['wTimeoutMs'])) { $writeConcern = (isset($urlOptions['w'])) ? $urlOptions['w'] : 1; $wTimeout = (isset($urlOptions['wTimeoutMs'])) ? $urlOptions['wTimeoutMs'] : null; $this->setWriteConcern($writeConcern, $wTimeout); } } }