/** * Copyright (C) 2014 MongoDB Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the GNU Affero General Public License in all respects for * all of the code used other than as permitted herein. If you modify file(s) * with this exception, you may extend this exception to your version of the * file(s), but you are not obligated to do so. If you do not wish to do so, * delete this exception statement from your version. If you delete this * exception statement from all source files in the program, then also delete * it in the license file. */ #define MONGO_LOG_DEFAULT_COMPONENT ::mongo::logger::LogComponent::kDefault #include "mongo/db/dbdirectclient.h" #include "mongo/db/client.h" #include "mongo/db/instance.h" #include "mongo/db/operation_context.h" #include "mongo/util/log.h" namespace mongo { // Called from scripting/engine.cpp and scripting/v8_db.cpp. DBClientBase* createDirectClient(OperationContext* txn) { return new DBDirectClient(txn); } namespace { class GodScope { MONGO_DISALLOW_COPYING(GodScope); public: GodScope(OperationContext* txn) : _txn(txn) { _prev = _txn->getClient()->setGod(true); } ~GodScope() { _txn->getClient()->setGod(_prev); } private: bool _prev; OperationContext* _txn; }; } // namespace DBDirectClient::DBDirectClient(OperationContext* txn) : _txn(txn) { } bool DBDirectClient::isFailed() const { return false; } bool DBDirectClient::isStillConnected() { return true; } std::string DBDirectClient::toString() const { return "DBDirectClient"; } std::string DBDirectClient::getServerAddress() const { return "localhost"; // TODO: should this have the port? } void DBDirectClient::sayPiggyBack(Message& toSend) { // don't need to piggy back when connected locally return say(toSend); } bool DBDirectClient::callRead(Message& toSend, Message& response) { return call(toSend, response); } ConnectionString::ConnectionType DBDirectClient::type() const { return ConnectionString::MASTER; } double DBDirectClient::getSoTimeout() const { return 0; } bool DBDirectClient::lazySupported() const { return true; } void DBDirectClient::setOpCtx(OperationContext* txn) { _txn = txn; } QueryOptions DBDirectClient::_lookupAvailableOptions() { // Exhaust mode is not available in DBDirectClient. return QueryOptions(DBClientBase::_lookupAvailableOptions() & ~QueryOption_Exhaust); } bool DBDirectClient::call(Message& toSend, Message& response, bool assertOk, string* actualServer) { GodScope gs(_txn); if (lastError._get()) { lastError.startRequest(toSend, lastError._get()); } DbResponse dbResponse; assembleResponse(_txn, toSend, dbResponse, dummyHost, true); verify(dbResponse.response); // can get rid of this if we make response handling smarter dbResponse.response->concat(); response = *dbResponse.response; return true; } void DBDirectClient::say(Message& toSend, bool isRetry, string* actualServer) { GodScope gs(_txn); if (lastError._get()) { lastError.startRequest(toSend, lastError._get()); } DbResponse dbResponse; assembleResponse(_txn, toSend, dbResponse, dummyHost, true); } auto_ptr DBDirectClient::query(const string& ns, Query query, int nToReturn, int nToSkip, const BSONObj* fieldsToReturn, int queryOptions, int batchSize) { return DBClientBase::query(ns, query, nToReturn, nToSkip, fieldsToReturn, queryOptions, batchSize); } void DBDirectClient::killCursor(long long id) { // The killCursor command on the DB client is only used by sharding, // so no need to have it for MongoD. verify(!"killCursor should not be used in MongoD"); } const HostAndPort DBDirectClient::dummyHost("0.0.0.0", 0); extern long long runCount(OperationContext* txn, const string& ns, const BSONObj &cmd, string &err, int &errCode); unsigned long long DBDirectClient::count(const string& ns, const BSONObj& query, int options, int limit, int skip) { if (skip < 0) { warning() << "setting negative skip value: " << skip << " to zero in query: " << query << endl; skip = 0; } string errmsg; int errCode; long long res = runCount(_txn, ns, _countCmd(ns, query, options, limit, skip), errmsg, errCode); if (res == -1) { // namespace doesn't exist return 0; } massert(errCode, str::stream() << "count failed in DBDirectClient: " << errmsg, res >= 0); return (unsigned long long)res; } } // namespace mongo