/**
* Copyright (C) 2009 10gen 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.
*/
#include "mongo/platform/basic.h"
#include "mongo/bson/bsonobjbuilder.h"
#include "mongo/db/auth/action_type.h"
#include "mongo/db/auth/authorization_session.h"
#include "mongo/db/client.h"
#include "mongo/db/commands.h"
#include "mongo/db/curop.h"
#include "mongo/db/dbwebserver.h"
#include "mongo/db/matcher/expression_parser.h"
#include "mongo/db/matcher/extensions_callback_disallow_extensions.h"
#include "mongo/db/operation_context.h"
#include "mongo/db/service_context.h"
#include "mongo/db/stats/fill_locker_info.h"
#include "mongo/rpc/metadata/client_metadata.h"
#include "mongo/rpc/metadata/client_metadata_ismaster.h"
#include "mongo/util/mongoutils/html.h"
#include "mongo/util/stringutils.h"
namespace mongo {
using std::unique_ptr;
using std::string;
namespace {
class ClientListPlugin : public WebStatusPlugin {
public:
ClientListPlugin() : WebStatusPlugin("clients", 20) {}
virtual void init() {}
virtual void run(OperationContext* txn, std::stringstream& ss) {
using namespace html;
ss << "\n
";
ss << ""
<< th(a("", "Connections to the database, both internal and external.", "Client"))
<< th(a("http://dochub.mongodb.org/core/viewingandterminatingcurrentoperation",
"",
"OpId"))
<< "Locking | "
<< "Waiting | "
<< "SecsRunning | "
<< "Op | "
<< th(a("http://dochub.mongodb.org/core/whatisanamespace", "", "Namespace"))
<< "Query | "
<< "client | "
<< "msg | "
<< "progress | "
<< "
\n";
_processAllClients(txn->getClient()->getServiceContext(), ss);
ss << "
\n";
}
private:
static void _processAllClients(ServiceContext* service, std::stringstream& ss) {
using namespace html;
for (ServiceContext::LockedClientsCursor cursor(service); Client* client = cursor.next();) {
invariant(client);
// Make the client stable
stdx::lock_guard lk(*client);
const OperationContext* txn = client->getOperationContext();
if (!txn)
continue;
CurOp* curOp = CurOp::get(txn);
if (!curOp)
continue;
ss << "" << client->desc() << " | ";
tablecell(ss, txn->getOpID());
tablecell(ss, true);
// LockState
{
Locker::LockerInfo lockerInfo;
txn->lockState()->getLockerInfo(&lockerInfo);
BSONObjBuilder lockerInfoBuilder;
fillLockerInfo(lockerInfo, lockerInfoBuilder);
tablecell(ss, lockerInfoBuilder.obj());
}
tablecell(ss, curOp->elapsedSeconds());
tablecell(ss, curOp->getNetworkOp());
tablecell(ss, html::escape(curOp->getNS()));
if (curOp->haveQuery()) {
tablecell(ss, html::escape(curOp->query().toString()));
} else {
tablecell(ss, "");
}
tablecell(ss, client->clientAddress(true /*includePort*/));
tablecell(ss, curOp->getMessage());
tablecell(ss, curOp->getProgressMeter().toString());
ss << "
\n";
}
}
} clientListPlugin;
class CurrentOpContexts : public Command {
public:
CurrentOpContexts() : Command("currentOpCtx") {}
virtual bool supportsWriteConcern(const BSONObj& cmd) const override {
return false;
}
virtual bool slaveOk() const {
return true;
}
virtual Status checkAuthForCommand(Client* client,
const std::string& dbname,
const BSONObj& cmdObj) {
if (AuthorizationSession::get(client)->isAuthorizedForActionsOnResource(
ResourcePattern::forClusterResource(), ActionType::inprog)) {
return Status::OK();
}
return Status(ErrorCodes::Unauthorized, "unauthorized");
}
bool run(OperationContext* txn,
const string& dbname,
BSONObj& cmdObj,
int,
string& errmsg,
BSONObjBuilder& result) {
unique_ptr filter;
if (cmdObj["filter"].isABSONObj()) {
const CollatorInterface* collator = nullptr;
StatusWithMatchExpression statusWithMatcher = MatchExpressionParser::parse(
cmdObj["filter"].Obj(), ExtensionsCallbackDisallowExtensions(), collator);
if (!statusWithMatcher.isOK()) {
return appendCommandStatus(result, statusWithMatcher.getStatus());
}
filter = std::move(statusWithMatcher.getValue());
}
result.appendArray("operations",
_processAllClients(txn->getClient()->getServiceContext(), filter.get()));
return true;
}
private:
static BSONArray _processAllClients(ServiceContext* service, MatchExpression* matcher) {
BSONArrayBuilder array;
for (ServiceContext::LockedClientsCursor cursor(service); Client* client = cursor.next();) {
invariant(client);
BSONObjBuilder b;
// Make the client stable
stdx::lock_guard lk(*client);
client->reportState(b);
const auto& clientMetadata =
ClientMetadataIsMasterState::get(client).getClientMetadata();
if (clientMetadata) {
auto appName = clientMetadata.get().getApplicationName();
if (!appName.empty()) {
b.append("applicationName", appName);
}
}
const OperationContext* txn = client->getOperationContext();
b.appendBool("active", static_cast(txn));
if (txn) {
b.append("opid", txn->getOpID());
if (txn->isKillPending()) {
b.append("killPending", true);
}
CurOp::get(txn)->reportState(&b);
// LockState
if (txn->lockState()) {
StringBuilder ss;
ss << txn->lockState();
b.append("lockStatePointer", ss.str());
Locker::LockerInfo lockerInfo;
txn->lockState()->getLockerInfo(&lockerInfo);
BSONObjBuilder lockerInfoBuilder;
fillLockerInfo(lockerInfo, lockerInfoBuilder);
b.append("lockState", lockerInfoBuilder.obj());
}
// RecoveryUnit
if (txn->recoveryUnit()) {
txn->recoveryUnit()->reportState(&b);
}
}
const BSONObj obj = b.obj();
if (!matcher || matcher->matchesBSON(obj)) {
array.append(obj);
}
}
return array.arr();
}
} currentOpContexts;
} // namespace
} // namespace mongo