diff options
Diffstat (limited to 'src/mongo/db/database.cpp')
-rw-r--r-- | src/mongo/db/database.cpp | 423 |
1 files changed, 423 insertions, 0 deletions
diff --git a/src/mongo/db/database.cpp b/src/mongo/db/database.cpp new file mode 100644 index 00000000000..2d55fd35626 --- /dev/null +++ b/src/mongo/db/database.cpp @@ -0,0 +1,423 @@ +// database.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 "pch.h" +#include "pdfile.h" +#include "database.h" +#include "instance.h" +#include "clientcursor.h" +#include "databaseholder.h" + +namespace mongo { + + bool Database::_openAllFiles = true; + + void assertDbAtLeastReadLocked(const Database *) { + // temp impl + d.dbMutex.assertAtLeastReadLocked(); + } + + void assertDbWriteLocked(const Database *) { + // temp impl + d.dbMutex.assertWriteLocked(); + } + + Database::~Database() { + d.dbMutex.assertWriteLocked(); + magic = 0; + size_t n = _files.size(); + for ( size_t i = 0; i < n; i++ ) + delete _files[i]; + if( ccByLoc.size() ) { + log() << "\n\n\nWARNING: ccByLoc not empty on database close! " << ccByLoc.size() << ' ' << name << endl; + } + } + + Database::Database(const char *nm, bool& newDb, const string& _path ) + : name(nm), path(_path), namespaceIndex( path, name ), + profileName(name + ".system.profile") + { + try { + { + // check db name is valid + size_t L = strlen(nm); + uassert( 10028 , "db name is empty", L > 0 ); + uassert( 10032 , "db name too long", L < 64 ); + uassert( 10029 , "bad db name [1]", *nm != '.' ); + uassert( 10030 , "bad db name [2]", nm[L-1] != '.' ); + uassert( 10031 , "bad char(s) in db name", strchr(nm, ' ') == 0 ); + } + newDb = namespaceIndex.exists(); + profile = cmdLine.defaultProfile; + checkDuplicateUncasedNames(true); + // If already exists, open. Otherwise behave as if empty until + // there's a write, then open. + if ( ! newDb || cmdLine.defaultProfile ) { + namespaceIndex.init(); + if( _openAllFiles ) + openAllFiles(); + } + magic = 781231; + } catch(std::exception& e) { + log() << "warning database " << path << ' ' << nm << " could not be opened" << endl; + log() << e.what() << endl; + // since destructor won't be called: + for ( size_t i = 0; i < _files.size(); i++ ) + delete _files[i]; + throw; + } + } + + void Database::checkDuplicateUncasedNames(bool inholderlock) const { + string duplicate = duplicateUncasedName(inholderlock, name, path ); + if ( !duplicate.empty() ) { + stringstream ss; + ss << "db already exists with different case other: [" << duplicate << "] me [" << name << "]"; + uasserted( DatabaseDifferCaseCode , ss.str() ); + } + } + + /*static*/ + string Database::duplicateUncasedName( bool inholderlock, const string &name, const string &path, set< string > *duplicates ) { + d.dbMutex.assertAtLeastReadLocked(); + + if ( duplicates ) { + duplicates->clear(); + } + + vector<string> others; + getDatabaseNames( others , path ); + + set<string> allShortNames; + dbHolder().getAllShortNames( inholderlock, allShortNames ); + + others.insert( others.end(), allShortNames.begin(), allShortNames.end() ); + + for ( unsigned i=0; i<others.size(); i++ ) { + + if ( strcasecmp( others[i].c_str() , name.c_str() ) ) + continue; + + if ( strcmp( others[i].c_str() , name.c_str() ) == 0 ) + continue; + + if ( duplicates ) { + duplicates->insert( others[i] ); + } else { + return others[i]; + } + } + if ( duplicates ) { + return duplicates->empty() ? "" : *duplicates->begin(); + } + return ""; + } + + boost::filesystem::path Database::fileName( int n ) const { + stringstream ss; + ss << name << '.' << n; + boost::filesystem::path fullName; + fullName = boost::filesystem::path(path); + if ( directoryperdb ) + fullName /= name; + fullName /= ss.str(); + return fullName; + } + + bool Database::openExistingFile( int n ) { + assert(this); + d.dbMutex.assertWriteLocked(); + { + // must not yet be visible to others as we aren't in the db's write lock and + // we will write to _files vector - thus this assert. + bool loaded = dbHolder().__isLoaded(name, path); + assert( !loaded ); + } + // additionally must be in the dbholder mutex (no assert for that yet) + + // todo: why here? that could be bad as we may be read locked only here + namespaceIndex.init(); + + if ( n < 0 || n >= DiskLoc::MaxFiles ) { + massert( 15924 , str::stream() << "getFile(): bad file number value " << n << " (corrupt db?): run repair", false); + } + + { + if( n < (int) _files.size() && _files[n] ) { + dlog(2) << "openExistingFile " << n << " is already open" << endl; + return true; + } + } + + { + boost::filesystem::path fullName = fileName( n ); + string fullNameString = fullName.string(); + MongoDataFile *df = new MongoDataFile(n); + try { + if( !df->openExisting( fullNameString.c_str() ) ) { + delete df; + return false; + } + } + catch ( AssertionException& ) { + delete df; + throw; + } + while ( n >= (int) _files.size() ) { + _files.push_back(0); + } + _files[n] = df; + } + + return true; + } + + // todo : we stop once a datafile dne. + // if one datafile were missing we should keep going for + // repair purposes yet we do not. + void Database::openAllFiles() { + //log() << "TEMP openallfiles " << path << ' ' << name << endl; + assert(this); + int n = 0; + while( openExistingFile(n) ) { + n++; + } + + /* + int n = 0; + while( exists(n) ) { + getFile(n); + n++; + } + // If last file is empty, consider it preallocated and make sure it's not mapped + // until a write is requested + if ( n > 1 && getFile( n - 1 )->getHeader()->isEmpty() ) { + delete _files[ n - 1 ]; + _files.pop_back(); + }*/ + } + + // todo: this is called a lot. streamline the common case + MongoDataFile* Database::getFile( int n, int sizeNeeded , bool preallocateOnly) { + assert(this); + DEV assertDbAtLeastReadLocked(this); + + namespaceIndex.init(); + if ( n < 0 || n >= DiskLoc::MaxFiles ) { + out() << "getFile(): n=" << n << endl; + massert( 10295 , "getFile(): bad file number value (corrupt db?): run repair", false); + } + DEV { + if ( n > 100 ) { + out() << "getFile(): n=" << n << endl; + } + } + MongoDataFile* p = 0; + if ( !preallocateOnly ) { + while ( n >= (int) _files.size() ) { + DEV if( !d.dbMutex.isWriteLocked() ) { + log() << "error: getFile() called in a read lock, yet file to return is not yet open" << endl; + log() << " getFile(" << n << ") _files.size:" <<_files.size() << ' ' << fileName(n).string() << endl; + log() << " context ns: " << cc().ns() << " openallfiles:" << _openAllFiles << endl; + } + assertDbWriteLocked(this); + _files.push_back(0); + } + p = _files[n]; + } + if ( p == 0 ) { + assertDbWriteLocked(this); + boost::filesystem::path fullName = fileName( n ); + string fullNameString = fullName.string(); + p = new MongoDataFile(n); + int minSize = 0; + if ( n != 0 && _files[ n - 1 ] ) + minSize = _files[ n - 1 ]->getHeader()->fileLength; + if ( sizeNeeded + DataFileHeader::HeaderSize > minSize ) + minSize = sizeNeeded + DataFileHeader::HeaderSize; + try { + p->open( fullNameString.c_str(), minSize, preallocateOnly ); + } + catch ( AssertionException& ) { + delete p; + throw; + } + if ( preallocateOnly ) + delete p; + else + _files[n] = p; + } + return preallocateOnly ? 0 : p; + } + + MongoDataFile* Database::addAFile( int sizeNeeded, bool preallocateNextFile ) { + assertDbWriteLocked(this); + int n = (int) _files.size(); + MongoDataFile *ret = getFile( n, sizeNeeded ); + if ( preallocateNextFile ) + preallocateAFile(); + return ret; + } + + bool fileIndexExceedsQuota( const char *ns, int fileIndex, bool enforceQuota ) { + return + cmdLine.quota && + enforceQuota && + fileIndex >= cmdLine.quotaFiles && + // we don't enforce the quota on "special" namespaces as that could lead to problems -- e.g. + // rejecting an index insert after inserting the main record. + !NamespaceString::special( ns ) && + NamespaceString( ns ).db != "local"; + } + + MongoDataFile* Database::suitableFile( const char *ns, int sizeNeeded, bool preallocate, bool enforceQuota ) { + + // check existing files + for ( int i=numFiles()-1; i>=0; i-- ) { + MongoDataFile* f = getFile( i ); + if ( f->getHeader()->unusedLength >= sizeNeeded ) { + if ( fileIndexExceedsQuota( ns, i-1, enforceQuota ) ) // NOTE i-1 is the value used historically for this check. + ; + else + return f; + } + } + + if ( fileIndexExceedsQuota( ns, numFiles(), enforceQuota ) ) + uasserted(12501, "quota exceeded"); + + // allocate files until we either get one big enough or hit maxSize + for ( int i = 0; i < 8; i++ ) { + MongoDataFile* f = addAFile( sizeNeeded, preallocate ); + + if ( f->getHeader()->unusedLength >= sizeNeeded ) + return f; + + if ( f->getHeader()->fileLength >= MongoDataFile::maxSize() ) // this is as big as they get so might as well stop + return f; + } + + uasserted(14810, "couldn't allocate space (suitableFile)"); // callers don't check for null return code + return 0; + } + + MongoDataFile* Database::newestFile() { + int n = numFiles(); + if ( n == 0 ) + return 0; + return getFile(n-1); + } + + + Extent* Database::allocExtent( const char *ns, int size, bool capped, bool enforceQuota ) { + // todo: when profiling, these may be worth logging into profile collection + bool fromFreeList = true; + Extent *e = DataFileMgr::allocFromFreeList( ns, size, capped ); + if( e == 0 ) { + fromFreeList = false; + e = suitableFile( ns, size, !capped, enforceQuota )->createExtent( ns, size, capped ); + } + LOG(1) << "allocExtent " << ns << " size " << size << ' ' << fromFreeList << endl; + return e; + } + + + bool Database::setProfilingLevel( int newLevel , string& errmsg ) { + if ( profile == newLevel ) + return true; + + if ( newLevel < 0 || newLevel > 2 ) { + errmsg = "profiling level has to be >=0 and <= 2"; + return false; + } + + if ( newLevel == 0 ) { + profile = 0; + return true; + } + + assert( cc().database() == this ); + + if ( ! namespaceIndex.details( profileName.c_str() ) ) { + log() << "creating profile collection: " << profileName << endl; + BSONObjBuilder spec; + spec.appendBool( "capped", true ); + spec.append( "size", 1024*1024 ); + if ( ! userCreateNS( profileName.c_str(), spec.done(), errmsg , false /* we don't replica profile messages */ ) ) { + return false; + } + } + profile = newLevel; + return true; + } + + bool Database::exists(int n) const { + return boost::filesystem::exists( fileName( n ) ); + } + + int Database::numFiles() const { + DEV assertDbAtLeastReadLocked(this); + return (int) _files.size(); + } + + void Database::flushFiles( bool sync ) { + assertDbAtLeastReadLocked(this); + for( vector<MongoDataFile*>::iterator i = _files.begin(); i != _files.end(); i++ ) { + MongoDataFile *f = *i; + f->flush(sync); + } + } + + long long Database::fileSize() const { + long long size=0; + for (int n=0; exists(n); n++) + size += boost::filesystem::file_size( fileName(n) ); + return size; + } + + Database* DatabaseHolder::getOrCreate( const string& ns , const string& path , bool& justCreated ) { + d.dbMutex.assertAtLeastReadLocked(); + + DBs& m = _paths[path]; + + string dbname = _todb( ns ); + + { + DBs::iterator i = m.find(dbname); + if( i != m.end() ) { + justCreated = false; + return i->second; + } + } + + // todo: protect against getting sprayed with requests for different db names that DNE - + // that would make the DBs map very large. not clear what to do to handle though, + // perhaps just log it, which is what we do here with the "> 40" : + bool cant = !d.dbMutex.isWriteLocked(); + if( logLevel >= 1 || m.size() > 40 || cant || DEBUG_BUILD ) { + log() << "opening db: " << (path==dbpath?"":path) << ' ' << dbname << endl; + } + massert(15927, "can't open database in a read lock. if db was just closed, consider retrying the query. might otherwise indicate an internal error", !cant); + + Database *db = new Database( dbname.c_str() , justCreated , path ); + m[dbname] = db; + _size++; + return db; + } + +} // namespace mongo |