2008-11-30 02:01:58 +01:00
|
|
|
// dbwebserver.cpp
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include "../util/miniwebserver.h"
|
|
|
|
#include "db.h"
|
2008-12-01 03:00:54 +01:00
|
|
|
#include "repl.h"
|
|
|
|
#include "replset.h"
|
2008-11-30 02:01:58 +01:00
|
|
|
|
|
|
|
extern int port;
|
2008-12-04 20:33:18 +01:00
|
|
|
extern const char *replInfo;
|
2008-11-30 02:01:58 +01:00
|
|
|
|
2008-12-02 20:24:45 +01:00
|
|
|
time_t started = time(0);
|
|
|
|
|
2008-12-05 22:45:10 +01:00
|
|
|
/*
|
|
|
|
string toString() {
|
|
|
|
stringstream ss;
|
|
|
|
unsigned long long dt = last - start;
|
|
|
|
ss << dt/1000;
|
|
|
|
ss << '\t';
|
|
|
|
ss << timeLocked/1000 << '\t';
|
|
|
|
if( dt )
|
|
|
|
ss << (timeLocked*100)/dt << '%';
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct Timing {
|
|
|
|
Timing() { last = start = timeLocked = 0; }
|
|
|
|
unsigned long long last, start, timeLocked;
|
|
|
|
};
|
|
|
|
Timing tlast;
|
|
|
|
const int NStats = 32;
|
|
|
|
string lockStats[NStats];
|
|
|
|
unsigned q = 0;
|
|
|
|
|
|
|
|
void statsThread() {
|
|
|
|
while( 1 ) {
|
|
|
|
sleepsecs(4);
|
|
|
|
dblock lk;
|
|
|
|
q = (q+1)%NStats;
|
|
|
|
Timing timing;
|
|
|
|
dbMutexInfo.timingInfo(timing.start, timing.last, timing.timeLocked);
|
|
|
|
if( tlast.start ) {
|
|
|
|
unsigned long long elapsed = timing.last - tlast.last;
|
|
|
|
unsigned long long locked = timing.timeLocked - tlast.timeLocked;
|
|
|
|
{
|
|
|
|
stringstream ss;
|
|
|
|
ss << elapsed / 1000 << '\t';
|
|
|
|
ss << locked / 1000 << '\t';
|
|
|
|
if( elapsed )
|
|
|
|
ss << (locked*100)/elapsed << '%';
|
|
|
|
lockStats[q] = ss.str();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tlast = timing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-30 02:01:58 +01:00
|
|
|
class DbWebServer : public MiniWebServer {
|
|
|
|
public:
|
2008-12-05 22:45:10 +01:00
|
|
|
// caller locks
|
2008-12-01 03:00:54 +01:00
|
|
|
void doLockedStuff(stringstream& ss) {
|
2008-12-02 20:24:45 +01:00
|
|
|
ss << "# databases: " << databases.size() << '\n';
|
|
|
|
if( database ) {
|
|
|
|
ss << "curclient: " << database->name;
|
2008-12-01 03:00:54 +01:00
|
|
|
ss << '\n';
|
|
|
|
}
|
2008-12-01 20:55:36 +01:00
|
|
|
ss << "\n<b>replication</b>\n";
|
2008-12-01 03:00:54 +01:00
|
|
|
ss << "master: " << master << '\n';
|
|
|
|
ss << "slave: " << slave << '\n';
|
|
|
|
if( replPair ) {
|
|
|
|
ss << "replpair:\n";
|
|
|
|
ss << replPair->getInfo();
|
|
|
|
}
|
2008-12-05 22:45:10 +01:00
|
|
|
|
|
|
|
ss << "\n<b>dt\ttlocked</b>\n";
|
|
|
|
unsigned i = q;
|
|
|
|
while( 1 ) {
|
|
|
|
ss << lockStats[i] << '\n';
|
|
|
|
i = (i-1)%NStats;
|
|
|
|
if( i == q )
|
|
|
|
break;
|
|
|
|
}
|
2008-12-01 03:00:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void doUnlockedStuff(stringstream& ss) {
|
|
|
|
ss << "port: " << port << '\n';
|
2008-12-05 22:45:10 +01:00
|
|
|
ss << "dblocked: " << dbMutexInfo.isLocked() << " (initial)\n";
|
2008-12-02 20:24:45 +01:00
|
|
|
ss << "uptime: " << time(0)-started << " seconds\n";
|
2008-12-01 20:55:36 +01:00
|
|
|
if( allDead )
|
|
|
|
ss << "<b>replication allDead=" << allDead << "</b>\n";
|
2008-12-05 00:11:25 +01:00
|
|
|
ss << "\nassertions:\n";
|
|
|
|
for( int i = 0; i < 4; i++ ) {
|
|
|
|
if( lastAssert[i].isSet() ) {
|
|
|
|
ss << "<b>";
|
|
|
|
if( i == 3 ) ss << "usererr";
|
|
|
|
else ss << i;
|
|
|
|
ss << "</b>" << ' ' << lastAssert[i].toString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ss << "\nreplInfo: " << replInfo << '\n';
|
2008-12-01 03:00:54 +01:00
|
|
|
}
|
2008-11-30 02:01:58 +01:00
|
|
|
|
|
|
|
virtual void doRequest(
|
|
|
|
const char *rq, // the full request
|
|
|
|
string url,
|
|
|
|
// set these and return them:
|
|
|
|
string& responseMsg,
|
|
|
|
int& responseCode,
|
|
|
|
vector<string>& headers // if completely empty, content-type: text/html will be added
|
|
|
|
)
|
|
|
|
{
|
|
|
|
responseCode = 200;
|
|
|
|
stringstream ss;
|
2008-12-04 20:33:18 +01:00
|
|
|
ss << "<html><head><title>";
|
|
|
|
|
|
|
|
string dbname;
|
|
|
|
{
|
|
|
|
stringstream z;
|
|
|
|
z << "db " << getHostName() << ':' << port << ' ';
|
|
|
|
dbname = z.str();
|
|
|
|
}
|
|
|
|
ss << dbname << "</title></head><body><h2>" << dbname << "</h2><p>\n<pre>";
|
2008-11-30 02:01:58 +01:00
|
|
|
|
2008-12-01 03:00:54 +01:00
|
|
|
doUnlockedStuff(ss);
|
2008-11-30 02:01:58 +01:00
|
|
|
|
|
|
|
int n = 2000;
|
2008-12-05 22:45:10 +01:00
|
|
|
Timer t;
|
2008-11-30 02:01:58 +01:00
|
|
|
while( 1 ) {
|
2008-12-05 22:45:10 +01:00
|
|
|
if( !dbMutexInfo.isLocked() ) {
|
|
|
|
{
|
|
|
|
dblock lk;
|
|
|
|
ss << "time to get dblock: " << t.millis() << "ms\n";
|
|
|
|
doLockedStuff(ss);
|
|
|
|
}
|
2008-11-30 02:01:58 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
sleepmillis(1);
|
|
|
|
if( --n < 0 ) {
|
2008-12-05 22:45:10 +01:00
|
|
|
ss << "\n<b>timed out getting dblock</b>\n";
|
2008-11-30 02:01:58 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ss << "</pre></body></html>";
|
|
|
|
responseMsg = ss.str();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void webServerThread() {
|
2008-12-05 22:45:10 +01:00
|
|
|
boost::thread thr(statsThread);
|
2008-11-30 02:01:58 +01:00
|
|
|
DbWebServer mini;
|
|
|
|
if( mini.init(port+1000) )
|
|
|
|
mini.run();
|
|
|
|
}
|