/**
* Copyright (C) 2008 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/db/dbwebserver.h"
#include "mongo/db/operation_context.h"
#include "mongo/db/stats/snapshots.h"
#include "mongo/util/mongoutils/html.h"
namespace mongo {
namespace {
using namespace html;
using std::fixed;
using std::setprecision;
using std::string;
using std::stringstream;
class DBTopStatus : public WebStatusPlugin {
public:
DBTopStatus() : WebStatusPlugin( "dbtop" , 50 , "(occurrences|percent of elapsed)" ) {}
void display( stringstream& ss , double elapsed , const Top::UsageData& usage ) {
ss << "
";
ss << usage.count;
ss << " | ";
double per = 100 * ((double)usage.time)/elapsed;
if( per == (int) per )
ss << (int) per;
else
ss << setprecision(1) << fixed << per;
ss << '%';
ss << " | ";
}
void display( stringstream& ss , double elapsed , const string& ns , const Top::CollectionData& data ) {
if ( ns != "TOTAL" && data.total.count == 0 )
return;
ss << "" << html::escape( ns ) << " | ";
display( ss , elapsed , data.total );
display( ss , elapsed , data.readLock );
display( ss , elapsed , data.writeLock );
display( ss , elapsed , data.queries );
display( ss , elapsed , data.getmore );
display( ss , elapsed , data.insert );
display( ss , elapsed , data.update );
display( ss , elapsed , data.remove );
ss << "
\n";
}
void run(OperationContext* txn, stringstream& ss) {
StatusWith diff = statsSnapshots.computeDelta();
if ( ! diff.isOK() )
return;
ss << "";
ss << "";
ss << a("http://dochub.mongodb.org/core/whatisanamespace", "namespace") <<
"NS | "
"total | "
"Reads | "
"Writes | "
"Queries | "
"GetMores | "
"Inserts | "
"Updates | "
"Removes | ";
ss << "
\n";
const Top::UsageMap& usage = diff.getValue().usageDiff;
unsigned long long elapsed = diff.getValue().timeElapsed;
for ( Top::UsageMap::const_iterator i=usage.begin(); i != usage.end(); ++i ) {
display( ss , (double) elapsed , i->first , i->second );
}
ss << "
";
}
virtual void init() {}
} dbtopStatus;
} // namespace
} // namespace mongo