From d05d73f13a768ef239d11ab01bdf208d03b6683c Mon Sep 17 00:00:00 2001 From: Teon Banek Date: Wed, 7 Feb 2018 11:02:03 +0100 Subject: [PATCH] Fix distributed plan caching and dispatching race condition Summary: Distributed plan is now dispatched before inserting into a cache. This prevents a race condition when another thread would take the cached plan and assume it was dispatched, thus crashing during execution. The fix introduces a possibility that the same plan may be distributed and dispatched twice (with different plan IDs). This shouldn't raise an issue during execution, because only one plan will be cached and thus executed. Obviously, we still need to invalidate the unused plan from worker's caches. Reviewers: florijan, msantl Reviewed By: florijan Subscribers: pullbot Differential Revision: https://phabricator.memgraph.io/D1178 --- src/query/interpreter.cpp | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/query/interpreter.cpp b/src/query/interpreter.cpp index 229e0c7ca..e0d6bd75d 100644 --- a/src/query/interpreter.cpp +++ b/src/query/interpreter.cpp @@ -89,10 +89,6 @@ Interpreter::Results Interpreter::operator()( ctx.symbol_table_, std::move(ast_storage)); } - if (FLAGS_query_plan_cache) { - plan = plan_cache_.access().insert(stripped.hash(), plan).first->second; - } - // Dispatch plans to workers (if we have any for them). if (plan->distributed_plan().worker_plan) { auto &dispatcher = db_accessor.db().plan_dispatcher(); @@ -100,6 +96,12 @@ Interpreter::Results Interpreter::operator()( plan->distributed_plan().worker_plan, plan->symbol_table()); } + + if (FLAGS_query_plan_cache) { + // TODO: If the same plan was already cached, invalidate the dispatched + // plan (above) from workers. + plan = plan_cache_.access().insert(stripped.hash(), plan).first->second; + } } auto *logical_plan = &plan->plan();