0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-11-30 00:56:44 +01:00
mongodb/db/dbhelpers.cpp

245 lines
7.6 KiB
C++

// dbhelpers.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 "db.h"
#include "dbhelpers.h"
#include "query.h"
#include "json.h"
#include "queryoptimizer.h"
#include "btree.h"
#include "pdfile.h"
namespace mongo {
CursorIterator::CursorIterator( auto_ptr<Cursor> c , BSONObj filter )
: _cursor( c ){
if ( ! filter.isEmpty() )
_matcher.reset( new CoveredIndexMatcher( filter , BSONObj() ) );
_advance();
}
BSONObj CursorIterator::next(){
BSONObj o = _o;
_advance();
return o;
}
bool CursorIterator::hasNext(){
return ! _o.isEmpty();
}
void CursorIterator::_advance(){
if ( ! _cursor->ok() ){
_o = BSONObj();
return;
}
while ( _cursor->ok() ){
_o = _cursor->current();
_cursor->advance();
if ( _matcher.get() == 0 || _matcher->matches( _o ) )
return;
}
_o = BSONObj();
}
void Helpers::ensureIndex(const char *ns, BSONObj keyPattern, bool unique, const char *name) {
NamespaceDetails *d = nsdetails(ns);
if( d == 0 )
return;
{
NamespaceDetails::IndexIterator i = d->ii();
while( i.more() ) {
if( i.next().keyPattern().woCompare(keyPattern) == 0 )
return;
}
}
if( d->nIndexes >= NamespaceDetails::NIndexesMax ) {
problem() << "Helper::ensureIndex fails, MaxIndexes exceeded " << ns << '\n';
return;
}
string system_indexes = cc().database()->name + ".system.indexes";
BSONObjBuilder b;
b.append("name", name);
b.append("ns", ns);
b.append("key", keyPattern);
b.appendBool("unique", unique);
BSONObj o = b.done();
theDataFileMgr.insert(system_indexes.c_str(), o.objdata(), o.objsize());
}
class FindOne : public QueryOp {
public:
FindOne( bool requireIndex ) : requireIndex_( requireIndex ) {}
virtual void init() {
if ( requireIndex_ && strcmp( qp().indexKey().firstElement().fieldName(), "$natural" ) == 0 )
throw MsgAssertionException( "Not an index cursor" );
c_ = qp().newCursor();
if ( !c_->ok() )
setComplete();
else
matcher_.reset( new CoveredIndexMatcher( qp().query(), qp().indexKey() ) );
}
virtual void next() {
if ( !c_->ok() ) {
setComplete();
return;
}
if ( matcher_->matches( c_->currKey(), c_->currLoc() ) ) {
one_ = c_->current();
setComplete();
} else {
c_->advance();
}
}
virtual bool mayRecordPlan() const { return false; }
virtual QueryOp *clone() const { return new FindOne( requireIndex_ ); }
BSONObj one() const { return one_; }
private:
bool requireIndex_;
auto_ptr< Cursor > c_;
auto_ptr< CoveredIndexMatcher > matcher_;
BSONObj one_;
};
/* fetch a single object from collection ns that matches query
set your db SavedContext first
*/
bool Helpers::findOne(const char *ns, BSONObj query, BSONObj& result, bool requireIndex) {
QueryPlanSet s( ns, query, BSONObj(), 0, !requireIndex );
FindOne original( requireIndex );
shared_ptr< FindOne > res = s.runOp( original );
massert( res->exceptionMessage(), res->complete() );
if ( res->one().isEmpty() )
return false;
result = res->one();
return true;
}
auto_ptr<CursorIterator> Helpers::find( const char *ns , BSONObj query , bool requireIndex ){
uassert( "requireIndex not supported in Helpers::find yet" , ! requireIndex );
auto_ptr<CursorIterator> i;
i.reset( new CursorIterator( DataFileMgr::findAll( ns ) , query ) );
return i;
}
bool Helpers::findById(const char *ns, BSONObj query, BSONObj& result ){
NamespaceDetails *d = nsdetails(ns);
if ( ! d )
return false;
int idxNo = d->findIdIndex();
if ( idxNo < 0 )
return false;
IndexDetails& i = d->idx( idxNo );
BSONObj key = i.getKeyFromQuery( query );
DiskLoc loc = i.head.btree()->findSingle( i , i.head , key );
if ( loc.isNull() )
return false;
result = loc.obj();
return true;
}
/* Get the first object from a collection. Generally only useful if the collection
only ever has a single object -- which is a "singleton collection.
Returns: true if object exists.
*/
bool Helpers::getSingleton(const char *ns, BSONObj& result) {
DBContext context(ns);
auto_ptr<Cursor> c = DataFileMgr::findAll(ns);
if ( !c->ok() )
return false;
result = c->current();
return true;
}
void Helpers::putSingleton(const char *ns, BSONObj obj) {
DBContext context(ns);
stringstream ss;
updateObjects(ns, obj, /*pattern=*/BSONObj(), /*upsert=*/true, /*multi=*/false, ss, /*logop=*/true );
}
void Helpers::emptyCollection(const char *ns) {
DBContext context(ns);
deleteObjects(ns, BSONObj(), false);
}
DbSet::~DbSet() {
if ( name_.empty() )
return;
try {
DBContext c( name_.c_str() );
if ( nsdetails( name_.c_str() ) ) {
string errmsg;
BSONObjBuilder result;
dropCollection( name_, errmsg, result );
}
} catch ( ... ) {
problem() << "exception cleaning up DbSet" << endl;
}
}
void DbSet::reset( const string &name, const BSONObj &key ) {
if ( !name.empty() )
name_ = name;
if ( !key.isEmpty() )
key_ = key.getOwned();
DBContext c( name_.c_str() );
if ( nsdetails( name_.c_str() ) ) {
Helpers::emptyCollection( name_.c_str() );
} else {
string err;
massert( err, userCreateNS( name_.c_str(), fromjson( "{autoIndexId:false}" ), err, false ) );
}
Helpers::ensureIndex( name_.c_str(), key_, true, "setIdx" );
}
bool DbSet::get( const BSONObj &obj ) const {
DBContext c( name_.c_str() );
BSONObj temp;
return Helpers::findOne( name_.c_str(), obj, temp, true );
}
void DbSet::set( const BSONObj &obj, bool val ) {
DBContext c( name_.c_str() );
if ( val ) {
try {
BSONObj k = obj;
theDataFileMgr.insert( name_.c_str(), k, false );
} catch ( DBException& ) {
// dup key - already in set
}
} else {
deleteObjects( name_.c_str(), obj, true, false, false );
}
}
} // namespace mongo