0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-12-01 09:32:32 +01:00
mongodb/db/commands.cpp

199 lines
6.2 KiB
C++
Raw Normal View History

2008-09-09 02:37:59 +02:00
/* commands.cpp
db "commands" (sent via db.$cmd.findOne(...))
*/
/* 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-09 02:37:59 +02:00
2010-04-27 21:27:52 +02:00
#include "pch.h"
2008-09-09 02:37:59 +02:00
#include "jsobj.h"
#include "commands.h"
2010-02-12 19:04:20 +01:00
#include "client.h"
2010-04-22 00:46:31 +02:00
#include "replpair.h"
2008-09-09 02:37:59 +02:00
2009-01-14 23:09:51 +01:00
namespace mongo {
2010-04-23 21:50:49 +02:00
map<string,Command*> * Command::_commandsByBestName;
2010-04-19 18:43:45 +02:00
map<string,Command*> * Command::_webCommands;
2010-01-20 20:02:14 +01:00
map<string,Command*> * Command::_commands;
2010-04-23 21:50:49 +02: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 20:52:36 +02:00
if( web ) ss << "<a href=\"/" << name << "?text=1\">";
2010-04-23 21:50:49 +02: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 22:41:56 +02:00
if( adminOnly() )
ss << "A";
2010-04-23 21:50:49 +02:00
ss << "</td>";
ss << "<td>";
if( helpStr != "no help defined" ) {
const char *p = helpStr.c_str();
2011-01-04 06:40:41 +01:00
while( *p ) {
2010-04-23 22:41:56 +02:00
if( *p == '<' ) {
ss << "&lt;";
p++; continue;
}
else if( *p == '{' )
2010-04-23 21:50:49 +02:00
ss << "<code>";
else if( *p == '}' ) {
ss << "}</code>";
p++;
continue;
}
2011-01-04 06:40:41 +01:00
if( strncmp(p, "http:", 5) == 0 ) {
2010-04-23 21:50:49 +02: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 06:40:41 +01:00
if( *q == '#' )
2010-04-23 22:41:56 +02:00
while( *q && *q != ' ' && *q != '\n' ) q++;
2010-04-23 21:50:49 +02: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) {
// register ourself.
2010-04-23 21:50:49 +02:00
if ( _commands == 0 )
2010-01-20 20:02:14 +01:00
_commands = new map<string,Command*>;
2010-04-23 21:50:49 +02:00
if( _commandsByBestName == 0 )
_commandsByBestName = new map<string,Command*>;
Command*& c = (*_commands)[name];
if ( c )
log() << "warning: 2 commands with name: " << _name << endl;
c = this;
2010-04-23 21:50:49 +02:00
(*_commandsByBestName)[name] = this;
2010-04-19 18:43:45 +02:00
if( web ) {
if( _webCommands == 0 )
_webCommands = new map<string,Command*>;
(*_webCommands)[name] = this;
}
2010-04-23 21:50:49 +02:00
if( oldName )
(*_commands)[oldName] = this;
}
void Command::help( stringstream& help ) const {
help << "no help defined";
}
2011-01-04 06:40:41 +01:00
2010-01-20 20:02:14 +01:00
bool Command::runAgainstRegistered(const char *ns, BSONObj& jsobj, BSONObjBuilder& anObjBuilder) {
const char *p = strchr(ns, '.');
if ( !p ) return false;
if ( strcmp(p, ".$cmd") != 0 ) return false;
2008-09-09 02:37:59 +02:00
bool ok = false;
bool valid = false;
2008-09-09 02:37:59 +02:00
2010-04-23 23:37:05 +02:00
BSONElement e = jsobj.firstElement();
map<string,Command*>::iterator i;
if ( e.eoo() )
;
/* check for properly registered command objects. Note that all the commands below should be
migrated over to the command object format.
*/
2010-01-20 20:02:14 +01:00
else if ( (i = _commands->find(e.fieldName())) != _commands->end() ) {
valid = true;
string errmsg;
Command *c = i->second;
2010-07-12 15:13:23 +02:00
if ( c->adminOnly() && !startsWith(ns, "admin.") ) {
ok = false;
2010-07-12 15:13:23 +02:00
errmsg = "access denied - use admin db";
}
2011-01-04 06:40:41 +01:00
else if ( jsobj.getBoolField( "help" ) ) {
stringstream help;
help << "help for: " << e.fieldName() << " ";
c->help( help );
anObjBuilder.append( "help" , help.str() );
}
else {
ok = c->run( nsToDatabase( ns ) , jsobj, errmsg, anObjBuilder, false);
}
BSONObj tmp = anObjBuilder.asTempObj();
bool have_ok = tmp.hasField("ok");
bool have_errmsg = tmp.hasField("errmsg");
if (!have_ok)
anObjBuilder.append( "ok" , ok ? 1.0 : 0.0 );
2011-01-04 06:40:41 +01:00
if ( !ok && !have_errmsg) {
anObjBuilder.append("errmsg", errmsg);
uassert_nothrow(errmsg.c_str());
}
return true;
}
2011-01-04 06:40:41 +01:00
return false;
}
2009-01-14 23:09:51 +01:00
2011-01-04 06:40:41 +01:00
Command* Command::findCommand( const string& name ) {
2010-01-20 20:02:14 +01:00
map<string,Command*>::iterator i = _commands->find( name );
if ( i == _commands->end() )
return 0;
return i->second;
}
2011-01-04 06:40:41 +01:00
Command::LockType Command::locktype( const string& name ) {
2010-01-20 20:02:14 +01:00
Command * c = findCommand( name );
if ( ! c )
2010-02-26 20:38:51 +01:00
return WRITE;
return c->locktype();
2010-01-20 20:02:14 +01:00
}
2010-02-12 19:04:20 +01:00
void Command::logIfSlow( const Timer& timer, const string& msg ) {
int ms = timer.millis();
2011-01-04 06:40:41 +01:00
if ( ms > cmdLine.slowMS ) {
2010-07-02 17:43:26 +02:00
out() << msg << " took " << ms << " ms." << endl;
}
}
2011-01-04 06:40:41 +01:00
2009-01-14 23:09:51 +01:00
} // namespace mongo