/**
* 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/commands.h"
#include "mongo/db/instance.h"
#include "mongo/db/lasterror.h"
#include "mongo/db/operation_context.h"
#include "mongo/util/log.h"
namespace mongo {
using std::unique_ptr;
using std::endl;
using std::string;
// Called from scripting/engine.cpp and scripting/v8_db.cpp.
DBClientBase* createDirectClient(OperationContext* txn) {
return new DBDirectClient(txn);
}
namespace {
class DirectClientScope {
MONGO_DISALLOW_COPYING(DirectClientScope);
public:
explicit DirectClientScope(OperationContext* txn)
: _txn(txn), _prev(_txn->getClient()->isInDirectClient()) {
_txn->getClient()->setInDirectClient(true);
}
~DirectClientScope() {
_txn->getClient()->setInDirectClient(_prev);
}
private:
OperationContext* const _txn;
const bool _prev;
};
} // 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?
}
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) {
DirectClientScope directClientScope(_txn);
LastError::get(_txn->getClient()).startRequest();
DbResponse dbResponse;
CurOp curOp(_txn);
assembleResponse(_txn, toSend, dbResponse, dummyHost);
verify(dbResponse.response);
// can get rid of this if we make response handling smarter
dbResponse.response->concat();
response = std::move(*dbResponse.response);
return true;
}
void DBDirectClient::say(Message& toSend, bool isRetry, string* actualServer) {
DirectClientScope directClientScope(_txn);
LastError::get(_txn->getClient()).startRequest();
DbResponse dbResponse;
CurOp curOp(_txn);
assembleResponse(_txn, toSend, dbResponse, dummyHost);
}
unique_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);
}
const HostAndPort DBDirectClient::dummyHost("0.0.0.0", 0);
unsigned long long DBDirectClient::count(
const string& ns, const BSONObj& query, int options, int limit, int skip) {
BSONObj cmdObj = _countCmd(ns, query, options, limit, skip);
NamespaceString nsString(ns);
std::string dbname = nsString.db().toString();
Command* countCmd = Command::findCommand("count");
invariant(countCmd);
std::string errmsg;
BSONObjBuilder result;
bool runRetval = countCmd->run(_txn, dbname, cmdObj, options, errmsg, result);
if (!runRetval) {
Command::appendCommandStatus(result, runRetval, errmsg);
Status commandStatus = Command::getStatusFromCommandResult(result.obj());
invariant(!commandStatus.isOK());
uassertStatusOK(commandStatus);
}
BSONObj resultObj = result.obj();
return static_cast(resultObj["n"].numberLong());
}
} // namespace mongo