Files
mongo/db/commands.cpp
2010-02-12 13:04:20 -05:00

189 lines
5.8 KiB
C++

/* 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.
*/
#include "stdafx.h"
#include "jsobj.h"
#include "commands.h"
#include "client.h"
#include "replset.h"
namespace mongo {
map<string,Command*> * Command::_commands;
Command::Command(const char *_name) : name(_name) {
// register ourself.
if ( _commands == 0 )
_commands = new map<string,Command*>;
(*_commands)[name] = this;
}
void Command::help( stringstream& help ) const {
help << "no help defined";
}
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;
bool ok = false;
bool valid = false;
BSONElement e;
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.
*/
else if ( (i = _commands->find(e.fieldName())) != _commands->end() ) {
valid = true;
string errmsg;
Command *c = i->second;
if ( c->adminOnly() && strncmp(ns, "admin", 5) != 0 ) {
ok = false;
errmsg = "access denied";
}
else if ( jsobj.getBoolField( "help" ) ){
stringstream help;
help << "help for: " << e.fieldName() << " ";
c->help( help );
anObjBuilder.append( "help" , help.str() );
}
else {
ok = c->run(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 );
if ( !ok && !have_errmsg) {
anObjBuilder.append("errmsg", errmsg);
uassert_nothrow(errmsg.c_str());
}
return true;
}
return false;
}
Command* Command::findCommand( const string& name ){
map<string,Command*>::iterator i = _commands->find( name );
if ( i == _commands->end() )
return 0;
return i->second;
}
bool Command::readOnly( const string& name ){
Command * c = findCommand( name );
if ( ! c )
return false;
return c->readOnly();
}
bool Command::exec( Client& client , int queryOptions ,
const char *ns, BSONObj& cmdObj ,
BSONObjBuilder& result,
bool fromRepl ){
string dbname = nsToDatabase( ns );
AuthenticationInfo *ai = client.getAuthenticationInfo();
if( adminOnly() && localHostOnlyIfNoAuth( cmdObj ) && noauth && !ai->isLocalHost ) {
result.append( "errmsg" ,
"unauthorized: this command must run from localhost when running db without auth" );
log() << "command denied: " << cmdObj.toString() << endl;
return false;
}
if ( adminOnly() && ! fromRepl && dbname != "admin" ) {
result.append( "errmsg" , "access denied" );
log() << "command denied: " << cmdObj.toString() << endl;
return false;
}
if ( cmdObj["help"].trueValue() ){
stringstream ss;
ss << "help for: " << name << " ";
help( ss );
result.append( "help" , ss.str() );
return true;
}
bool canRunHere =
isMaster() ||
slaveOk() ||
( slaveOverrideOk() && ( queryOptions & QueryOption_SlaveOk ) ) ||
fromRepl;
if ( ! canRunHere ){
result.append( "errmsg" , "not master" );
return false;
}
bool needWriteLock = ! readOnly();
if ( ! requiresAuth() &&
( ai->isAuthorizedReads( dbname ) &&
! ai->isAuthorized( dbname ) ) ){
// this means that they can read, but not write
// so only get a read lock
needWriteLock = false;
}
mongolock lk( needWriteLock );
Client::Context ctx( ns , dbpath , &lk , requiresAuth() );
if ( adminOnly() )
log( 2 ) << "command: " << cmdObj << endl;
try {
string errmsg;
if ( ! run(ns, cmdObj, errmsg, result, fromRepl ) ){
result.append( "errmsg" , errmsg );
return false;
}
}
catch ( AssertionException& e ){
stringstream ss;
ss << "assertion: " << e.what();
result.append( "errmsg" , ss.str() );
return false;
}
if ( logTheOp() && ! fromRepl )
logOp("c", ns, cmdObj);
return true;
}
} // namespace mongo