2008-09-08 20:37:59 -04:00
|
|
|
/* commands.cpp
|
|
|
|
|
db "commands" (sent via db.$cmd.findOne(...))
|
|
|
|
|
*/
|
|
|
|
|
|
2009-10-27 15:58:27 -04:00
|
|
|
/* Copyright 2009 10gen Inc.
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
2008-09-08 20:37:59 -04:00
|
|
|
|
2010-04-27 15:27:52 -04:00
|
|
|
#include "pch.h"
|
2008-09-08 20:37:59 -04:00
|
|
|
#include "jsobj.h"
|
2008-09-10 15:31:05 -04:00
|
|
|
#include "commands.h"
|
2010-02-12 13:04:20 -05:00
|
|
|
#include "client.h"
|
2011-03-29 16:56:21 -07:00
|
|
|
#include "replutil.h"
|
2008-09-08 20:37:59 -04:00
|
|
|
|
2009-01-14 17:09:51 -05:00
|
|
|
namespace mongo {
|
|
|
|
|
|
2010-04-23 15:50:49 -04:00
|
|
|
map<string,Command*> * Command::_commandsByBestName;
|
2010-04-19 12:43:45 -04:00
|
|
|
map<string,Command*> * Command::_webCommands;
|
2010-01-20 14:02:14 -05:00
|
|
|
map<string,Command*> * Command::_commands;
|
2008-09-12 11:26:51 -04:00
|
|
|
|
2010-04-23 15:50:49 -04:00
|
|
|
void Command::htmlHelp(stringstream& ss) const {
|
|
|
|
|
string helpStr;
|
|
|
|
|
{
|
|
|
|
|
stringstream h;
|
|
|
|
|
help(h);
|
|
|
|
|
helpStr = h.str();
|
|
|
|
|
}
|
|
|
|
|
ss << "\n<tr><td>";
|
|
|
|
|
bool web = _webCommands->count(name) != 0;
|
2010-08-25 14:52:36 -04:00
|
|
|
if( web ) ss << "<a href=\"/" << name << "?text=1\">";
|
2010-04-23 15:50:49 -04:00
|
|
|
ss << name;
|
|
|
|
|
if( web ) ss << "</a>";
|
|
|
|
|
ss << "</td>\n";
|
|
|
|
|
ss << "<td>";
|
|
|
|
|
int l = locktype();
|
|
|
|
|
//if( l == NONE ) ss << "N ";
|
|
|
|
|
if( l == READ ) ss << "R ";
|
|
|
|
|
else if( l == WRITE ) ss << "W ";
|
|
|
|
|
if( slaveOk() )
|
|
|
|
|
ss << "S ";
|
2010-04-23 16:41:56 -04:00
|
|
|
if( adminOnly() )
|
|
|
|
|
ss << "A";
|
2010-04-23 15:50:49 -04:00
|
|
|
ss << "</td>";
|
|
|
|
|
ss << "<td>";
|
|
|
|
|
if( helpStr != "no help defined" ) {
|
|
|
|
|
const char *p = helpStr.c_str();
|
2011-01-04 00:40:41 -05:00
|
|
|
while( *p ) {
|
2010-04-23 16:41:56 -04:00
|
|
|
if( *p == '<' ) {
|
|
|
|
|
ss << "<";
|
|
|
|
|
p++; continue;
|
|
|
|
|
}
|
|
|
|
|
else if( *p == '{' )
|
2010-04-23 15:50:49 -04:00
|
|
|
ss << "<code>";
|
|
|
|
|
else if( *p == '}' ) {
|
|
|
|
|
ss << "}</code>";
|
|
|
|
|
p++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2011-01-04 00:40:41 -05:00
|
|
|
if( strncmp(p, "http:", 5) == 0 ) {
|
2010-04-23 15:50:49 -04:00
|
|
|
ss << "<a href=\"";
|
|
|
|
|
const char *q = p;
|
|
|
|
|
while( *q && *q != ' ' && *q != '\n' )
|
|
|
|
|
ss << *q++;
|
|
|
|
|
ss << "\">";
|
|
|
|
|
q = p;
|
|
|
|
|
if( startsWith(q, "http://www.mongodb.org/display/") )
|
|
|
|
|
q += 31;
|
|
|
|
|
while( *q && *q != ' ' && *q != '\n' ) {
|
|
|
|
|
ss << (*q == '+' ? ' ' : *q);
|
|
|
|
|
q++;
|
2011-01-04 00:40:41 -05:00
|
|
|
if( *q == '#' )
|
2010-04-23 16:41:56 -04:00
|
|
|
while( *q && *q != ' ' && *q != '\n' ) q++;
|
2010-04-23 15:50:49 -04:00
|
|
|
}
|
|
|
|
|
ss << "</a>";
|
|
|
|
|
p = q;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if( *p == '\n' ) ss << "<br>";
|
|
|
|
|
else ss << *p;
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ss << "</td>";
|
|
|
|
|
ss << "</tr>\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Command::Command(const char *_name, bool web, const char *oldName) : name(_name) {
|
2009-01-15 10:17:11 -05:00
|
|
|
// register ourself.
|
2010-04-23 15:50:49 -04:00
|
|
|
if ( _commands == 0 )
|
2010-01-20 14:02:14 -05:00
|
|
|
_commands = new map<string,Command*>;
|
2010-04-23 15:50:49 -04:00
|
|
|
if( _commandsByBestName == 0 )
|
|
|
|
|
_commandsByBestName = new map<string,Command*>;
|
2010-04-03 21:08:04 -04:00
|
|
|
Command*& c = (*_commands)[name];
|
|
|
|
|
if ( c )
|
|
|
|
|
log() << "warning: 2 commands with name: " << _name << endl;
|
|
|
|
|
c = this;
|
2010-04-23 15:50:49 -04:00
|
|
|
(*_commandsByBestName)[name] = this;
|
2010-04-19 12:43:45 -04:00
|
|
|
|
|
|
|
|
if( web ) {
|
|
|
|
|
if( _webCommands == 0 )
|
|
|
|
|
_webCommands = new map<string,Command*>;
|
|
|
|
|
(*_webCommands)[name] = this;
|
|
|
|
|
}
|
2010-04-23 15:50:49 -04:00
|
|
|
|
|
|
|
|
if( oldName )
|
|
|
|
|
(*_commands)[oldName] = this;
|
2009-01-15 10:17:11 -05:00
|
|
|
}
|
2008-09-10 15:31:05 -04:00
|
|
|
|
2009-02-13 09:44:07 -05:00
|
|
|
void Command::help( stringstream& help ) const {
|
|
|
|
|
help << "no help defined";
|
|
|
|
|
}
|
2011-01-04 00:40:41 -05:00
|
|
|
|
|
|
|
|
Command* Command::findCommand( const string& name ) {
|
2010-01-20 14:02:14 -05:00
|
|
|
map<string,Command*>::iterator i = _commands->find( name );
|
|
|
|
|
if ( i == _commands->end() )
|
|
|
|
|
return 0;
|
|
|
|
|
return i->second;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2011-01-04 00:40:41 -05:00
|
|
|
Command::LockType Command::locktype( const string& name ) {
|
2010-01-20 14:02:14 -05:00
|
|
|
Command * c = findCommand( name );
|
|
|
|
|
if ( ! c )
|
2010-02-26 14:38:51 -05:00
|
|
|
return WRITE;
|
|
|
|
|
return c->locktype();
|
2010-01-20 14:02:14 -05:00
|
|
|
}
|
2010-02-12 13:04:20 -05:00
|
|
|
|
2010-07-03 14:20:18 -04:00
|
|
|
void Command::logIfSlow( const Timer& timer, const string& msg ) {
|
|
|
|
|
int ms = timer.millis();
|
2011-01-04 00:40:41 -05:00
|
|
|
if ( ms > cmdLine.slowMS ) {
|
2010-07-02 11:43:26 -04:00
|
|
|
out() << msg << " took " << ms << " ms." << endl;
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-01-04 00:40:41 -05:00
|
|
|
|
2011-09-14 22:12:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#include "../client/connpool.h"
|
|
|
|
|
|
|
|
|
|
namespace mongo {
|
|
|
|
|
|
|
|
|
|
extern DBConnectionPool pool;
|
|
|
|
|
|
|
|
|
|
class PoolFlushCmd : public Command {
|
|
|
|
|
public:
|
|
|
|
|
PoolFlushCmd() : Command( "connPoolSync" , false , "connpoolsync" ) {}
|
|
|
|
|
virtual void help( stringstream &help ) const { help<<"internal"; }
|
|
|
|
|
virtual LockType locktype() const { return NONE; }
|
|
|
|
|
virtual bool run(const string&, mongo::BSONObj&, int, std::string&, mongo::BSONObjBuilder& result, bool) {
|
|
|
|
|
pool.flush();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
virtual bool slaveOk() const {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} poolFlushCmd;
|
|
|
|
|
|
|
|
|
|
class PoolStats : public Command {
|
|
|
|
|
public:
|
|
|
|
|
PoolStats() : Command( "connPoolStats" ) {}
|
|
|
|
|
virtual void help( stringstream &help ) const { help<<"stats about connection pool"; }
|
|
|
|
|
virtual LockType locktype() const { return NONE; }
|
|
|
|
|
virtual bool run(const string&, mongo::BSONObj&, int, std::string&, mongo::BSONObjBuilder& result, bool) {
|
|
|
|
|
pool.appendInfo( result );
|
|
|
|
|
result.append( "numDBClientConnection" , DBClientConnection::getNumConnections() );
|
|
|
|
|
result.append( "numAScopedConnection" , AScopedConnection::getNumConnections() );
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
virtual bool slaveOk() const {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} poolStatsCmd;
|
2011-01-04 00:40:41 -05:00
|
|
|
|
2009-01-14 17:09:51 -05:00
|
|
|
} // namespace mongo
|