diff options
author | David Storch <david.storch@10gen.com> | 2014-09-09 16:20:31 -0400 |
---|---|---|
committer | David Storch <david.storch@10gen.com> | 2014-09-09 17:09:25 -0400 |
commit | 4d33f84dda05fa9b72e1b1b50d1cfcae13e36470 (patch) | |
tree | 85a812ef23322e950b58e038cd6f79d712f8bf8c | |
parent | c0bc7efa03c71b3929846bad059af5beda5ab231 (diff) | |
download | mongo-4d33f84dda05fa9b72e1b1b50d1cfcae13e36470.tar.gz |
SERVER-15012 SubplanRunner must check for missing cache data
-rw-r--r-- | src/mongo/db/query/subplan_runner.cpp | 7 | ||||
-rw-r--r-- | src/mongo/dbtests/query_subplan_runner.cpp | 120 |
2 files changed, 127 insertions, 0 deletions
diff --git a/src/mongo/db/query/subplan_runner.cpp b/src/mongo/db/query/subplan_runner.cpp index 48e00165085..2fb7ca30073 100644 --- a/src/mongo/db/query/subplan_runner.cpp +++ b/src/mongo/db/query/subplan_runner.cpp @@ -340,6 +340,13 @@ namespace mongo { QuerySolution* bestSoln = solutions[bestPlan]; + // Check that we have good cache data. For example, we don't cache things for 2d + // indices. + if (NULL == bestSoln->cacheData.get()) { + QLOG() << "Subplanner: No cache data for subchild: " << orChild->toString(); + return false; + } + if (SolutionCacheData::USE_INDEX_TAGS_SOLN != bestSoln->cacheData->solnType) { QLOG() << "Subplanner: No indexed cache data for subchild " << orChild->toString(); diff --git a/src/mongo/dbtests/query_subplan_runner.cpp b/src/mongo/dbtests/query_subplan_runner.cpp new file mode 100644 index 00000000000..0c8c5179386 --- /dev/null +++ b/src/mongo/dbtests/query_subplan_runner.cpp @@ -0,0 +1,120 @@ +/** + * 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/client/dbclientcursor.h" +#include "mongo/db/catalog/collection.h" +#include "mongo/db/catalog/database.h" +#include "mongo/db/jsobj.h" +#include "mongo/db/json.h" +#include "mongo/db/query/canonical_query.h" +#include "mongo/db/query/get_runner.h" +#include "mongo/db/query/subplan_runner.h" +#include "mongo/dbtests/dbtests.h" + +namespace QueryStageSubplan { + + class QuerySubplanRunnerBase { + public: + QuerySubplanRunnerBase() { } + + virtual ~QuerySubplanRunnerBase() { + Client::WriteContext ctx(ns()); + _client.dropCollection(ns()); + } + + void addIndex(const BSONObj& obj) { + _client.ensureIndex(ns(), obj); + } + + void insert(const BSONObj& doc) { + _client.insert(ns(), doc); + } + + static const char* ns() { return "unittests.QuerySubplanRunner"; } + + private: + static DBDirectClient _client; + }; + + DBDirectClient QuerySubplanRunnerBase::_client; + + /** + * SERVER-15012: test that the subplan stage does not crash when the winning solution + * for an $or clause uses a '2d' index. We don't produce cache data for '2d'. The subplanner + * should gracefully fail after finding that no cache data is available, allowing us to fall + * back to regular planning. + */ + class QuerySubplanRunnerGeo2dOr : public QuerySubplanRunnerBase { + public: + void run() { + Client::WriteContext ctx(ns()); + addIndex(BSON("a" << "2d")); + addIndex(BSON("b" << 1)); + + BSONObj query = + fromjson("{$or: [{a: {$geoWithin: {$box: [[0,0],[1,1]]}}, b: 1}," + "{a: {$geoWithin: {$box: [[1,1],[1,1]]}}, b: 1}]}"); + + CanonicalQuery* cq; + ASSERT_OK(CanonicalQuery::canonicalize(ns(), query, &cq)); + + Collection* collection = ctx.ctx().db()->getCollection(ns()); + + // Get planner params. + QueryPlannerParams plannerParams; + fillOutPlannerParams(collection, cq, &plannerParams); + + // Create the subplan runner. + SubplanRunner* subplanRunner; + ASSERT_OK(SubplanRunner::make(collection, plannerParams, cq, &subplanRunner)); + + // On success, the subplan runner should be non-NULL. + ASSERT(subplanRunner); + + // Make sure that we can run the plan to EOF without crashing. + BSONObj objOut; + Runner::RunnerState state = Runner::RUNNER_ADVANCED; + while (Runner::RUNNER_ADVANCED == state) { + state = subplanRunner->getNext(&objOut, NULL); + } + + // Make sure we hit EOF successfully. + ASSERT(Runner::RUNNER_EOF == state); + } + }; + + class All : public Suite { + public: + All() : Suite("query_subplan_runner") {} + + void setupTests() { + add<QuerySubplanRunnerGeo2dOr>(); + } + } all; + +} // namespace QueryStageSubplan |