/**
* Copyright (C) 2013 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/base/init.h"
#include "mongo/db/auth/action_type.h"
#include "mongo/db/auth/authorization_session.h"
#include "mongo/db/auth/privilege.h"
#include "mongo/db/commands.h"
#include "mongo/db/field_parser.h"
#include "mongo/db/namespace_string.h"
#include "mongo/s/d_logic.h"
#include "mongo/s/d_merge.h"
namespace mongo {
/**
* Mongod-side command for merging chunks.
*/
class MergeChunksCommand : public Command {
public:
MergeChunksCommand() : Command("mergeChunks") {}
virtual void help(stringstream& h) const {
h << "Merge Chunks command\n"
<< "usage: { mergeChunks : , bounds : [ , ],"
<< " (opt) epoch : , (opt) config : ,"
<< " (opt) shardName : }";
}
virtual Status checkAuthForCommand(ClientBasic* client,
const std::string& dbname,
const BSONObj& cmdObj) {
if (!client->getAuthorizationSession()->isAuthorizedForActionsOnResource(
ResourcePattern::forExactNamespace(NamespaceString(parseNs(dbname, cmdObj))),
ActionType::splitChunk)) {
return Status(ErrorCodes::Unauthorized, "Unauthorized");
}
return Status::OK();
}
virtual std::string parseNs(const std::string& dbname, const BSONObj& cmdObj) const {
return parseNsFullyQualified(dbname, cmdObj);
}
virtual bool adminOnly() const { return true; }
virtual bool slaveOk() const { return false; }
virtual bool isWriteCommandForConfigServer() const { return false; }
// Required
static BSONField nsField;
static BSONField > boundsField;
// Optional, if the merge is only valid for a particular epoch
static BSONField epochField;
// Optional, if our sharding state has not previously been initializeed
static BSONField shardNameField;
static BSONField configField;
bool run(TransactionExperiment* txn, const string& dbname,
BSONObj& cmdObj,
int,
string& errmsg,
BSONObjBuilder& result,
bool ) {
string ns = parseNs(dbname, cmdObj);
if ( ns.size() == 0 ) {
errmsg = "no namespace specified";
return false;
}
vector bounds;
if ( !FieldParser::extract( cmdObj, boundsField, &bounds, &errmsg ) ) {
return false;
}
if ( bounds.size() == 0 ) {
errmsg = "no bounds were specified";
return false;
}
if ( bounds.size() != 2 ) {
errmsg = "only a min and max bound may be specified";
return false;
}
BSONObj minKey = bounds[0];
BSONObj maxKey = bounds[1];
if ( minKey.isEmpty() ) {
errmsg = "no min key specified";
return false;
}
if ( maxKey.isEmpty() ) {
errmsg = "no max key specified";
return false;
}
//
// This might be the first call from mongos, so we may need to pass the config and shard
// information to initialize the shardingState.
//
string config;
FieldParser::FieldState extracted = FieldParser::extract( cmdObj,
configField,
&config,
&errmsg );
if ( !extracted ) return false;
if ( extracted != FieldParser::FIELD_NONE ) {
ShardingState::initialize( config );
}
else if ( !shardingState.enabled() ) {
errmsg =
"sharding state must be enabled or config server specified to merge chunks";
return false;
}
// ShardName is optional, but might not be set yet
string shardName;
extracted = FieldParser::extract( cmdObj, shardNameField, &shardName, &errmsg );
if ( !extracted ) return false;
if ( extracted != FieldParser::FIELD_NONE ) {
shardingState.gotShardName( shardName );
}
//
// Epoch is optional, and if not set indicates we should use the latest epoch
//
OID epoch;
if ( !FieldParser::extract( cmdObj, epochField, &epoch, &errmsg ) ) {
return false;
}
return mergeChunks( NamespaceString( ns ), minKey, maxKey, epoch, true, &errmsg );
}
};
BSONField MergeChunksCommand::nsField( "mergeChunks" );
BSONField > MergeChunksCommand::boundsField( "bounds" );
BSONField MergeChunksCommand::configField( "config" );
BSONField MergeChunksCommand::shardNameField( "shardName" );
BSONField MergeChunksCommand::epochField( "epoch" );
MONGO_INITIALIZER(InitMergeChunksCommand)(InitializerContext* context) {
// Leaked intentionally: a Command registers itself when constructed.
new MergeChunksCommand();
return Status::OK();
}
}