diff options
author | Benety Goh <benety@mongodb.com> | 2014-01-09 15:05:55 -0500 |
---|---|---|
committer | Benety Goh <benety@mongodb.com> | 2014-01-15 16:35:44 -0500 |
commit | 9b94912fe8847977c4d23a12604cb900e6f426b7 (patch) | |
tree | 688de4d301133aa1a409529926487349584c7204 /src/mongo/db/query/query_settings.cpp | |
parent | 26a8aaf0e660c247c2667e0f182bc4964096735b (diff) | |
download | mongo-9b94912fe8847977c4d23a12604cb900e6f426b7.tar.gz |
SERVER-8871 admin hints
Diffstat (limited to 'src/mongo/db/query/query_settings.cpp')
-rw-r--r-- | src/mongo/db/query/query_settings.cpp | 166 |
1 files changed, 166 insertions, 0 deletions
diff --git a/src/mongo/db/query/query_settings.cpp b/src/mongo/db/query/query_settings.cpp new file mode 100644 index 00000000000..35ab6de5694 --- /dev/null +++ b/src/mongo/db/query/query_settings.cpp @@ -0,0 +1,166 @@ +/** + * Copyright (C) 2014 MongoDB 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/>. + * + * 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/db/query/query_settings.h" +#include "mongo/db/query/canonical_query.h" +#include "mongo/db/query/plan_cache.h" + +namespace mongo { + + using std::vector; + + // + // HintOverride + // + + AllowedIndices::AllowedIndices(const std::vector<BSONObj>& indexKeyPatterns) { + for (std::vector<BSONObj>::const_iterator i = indexKeyPatterns.begin(); + i != indexKeyPatterns.end(); ++i) { + const BSONObj& indexKeyPattern = *i; + this->indexKeyPatterns.push_back(indexKeyPattern.copy()); + } + } + + AllowedIndices::~AllowedIndices() { } + + // + // AllowedIndexEntry + // + + AllowedIndexEntry::AllowedIndexEntry(const BSONObj& query, const BSONObj& sort, + const BSONObj& projection, + const std::vector<BSONObj>& indexKeyPatterns) + : query(query.copy()), + sort(sort.copy()), + projection(projection.copy()) { + for (std::vector<BSONObj>::const_iterator i = indexKeyPatterns.begin(); + i != indexKeyPatterns.end(); ++i) { + const BSONObj& indexKeyPattern = *i; + this->indexKeyPatterns.push_back(indexKeyPattern.copy()); + } + } + + AllowedIndexEntry::~AllowedIndexEntry() { } + + AllowedIndexEntry* AllowedIndexEntry::clone() const { + AllowedIndexEntry* entry = new AllowedIndexEntry(query, sort, projection, indexKeyPatterns); + return entry; + } + + // + // QuerySettings + // + + QuerySettings::QuerySettings() { } + + QuerySettings::~QuerySettings() { + _clear(); + } + + bool QuerySettings::getAllowedIndices(const CanonicalQuery& query, + AllowedIndices** allowedIndicesOut) const { + invariant(allowedIndicesOut); + + const PlanCacheKey& key = query.getPlanCacheKey(); + + boost::lock_guard<boost::mutex> cacheLock(_mutex); + AllowedIndexEntryMap::const_iterator cacheIter = _allowedIndexEntryMap.find(key); + + // Nothing to do if key does not exist in query settings. + if (cacheIter == _allowedIndexEntryMap.end()) { + *allowedIndicesOut = NULL; + return false; + } + + AllowedIndexEntry* entry = cacheIter->second; + + // Create a AllowedIndices from entry. + *allowedIndicesOut = new AllowedIndices(entry->indexKeyPatterns); + + return true; + } + + std::vector<AllowedIndexEntry*> QuerySettings::getAllAllowedIndices() const { + boost::lock_guard<boost::mutex> cacheLock(_mutex); + vector<AllowedIndexEntry*> entries; + for (AllowedIndexEntryMap::const_iterator i = _allowedIndexEntryMap.begin(); i != _allowedIndexEntryMap.end(); ++i) { + AllowedIndexEntry* entry = i->second; + entries.push_back(entry->clone()); + } + return entries; + } + + void QuerySettings::setAllowedIndices(const CanonicalQuery& canonicalQuery, + const std::vector<BSONObj>& indexes) { + const LiteParsedQuery& lpq = canonicalQuery.getParsed(); + const BSONObj& query = lpq.getFilter(); + const BSONObj& sort = lpq.getSort(); + const BSONObj& projection = lpq.getProj(); + AllowedIndexEntry* entry = new AllowedIndexEntry(query, sort, projection, indexes); + + const PlanCacheKey& key = canonicalQuery.getPlanCacheKey(); + boost::lock_guard<boost::mutex> cacheLock(_mutex); + AllowedIndexEntryMap::iterator i = _allowedIndexEntryMap.find(key); + // Replace existing entry. + if (i != _allowedIndexEntryMap.end()) { + AllowedIndexEntry* entry = i->second; + delete entry; + } + _allowedIndexEntryMap[key] = entry; + } + + void QuerySettings::removeAllowedIndices(const CanonicalQuery& canonicalQuery) { + const PlanCacheKey& key = canonicalQuery.getPlanCacheKey(); + boost::lock_guard<boost::mutex> cacheLock(_mutex); + AllowedIndexEntryMap::iterator i = _allowedIndexEntryMap.find(key); + + // Nothing to do if key does not exist in query settings. + if (i == _allowedIndexEntryMap.end()) { + return; + } + + // Free up resources and delete entry. + AllowedIndexEntry* entry = i->second; + _allowedIndexEntryMap.erase(i); + delete entry; + } + + void QuerySettings::clearAllowedIndices() { + boost::lock_guard<boost::mutex> cacheLock(_mutex); + _clear(); + } + + void QuerySettings::_clear() { + for (AllowedIndexEntryMap::const_iterator i = _allowedIndexEntryMap.begin(); i != _allowedIndexEntryMap.end(); ++i) { + AllowedIndexEntry* entry = i->second; + delete entry; + } + _allowedIndexEntryMap.clear(); + } + +} // namespace mongo |