Skip to content

Commit

Permalink
feat(fuzzer): Update make[merge|NLJ]Plan functions to generate multi-…
Browse files Browse the repository at this point in the history
…join plans (#11941)

Summary:

Change makeMergeJoinPlan and makeNestedLoopJoinPlan functions to produce cascading multi-joins.

Differential Revision: D67607605
  • Loading branch information
Daniel Hunte authored and facebook-github-bot committed Dec 27, 2024
1 parent 20e0c80 commit a284fb8
Showing 1 changed file with 73 additions and 58 deletions.
131 changes: 73 additions & 58 deletions velox/exec/fuzzer/JoinFuzzer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -145,21 +145,19 @@ class JoinFuzzer {
const std::vector<std::string>& filterList);

JoinFuzzer::PlanWithSplits makeMergeJoinPlan(
core::JoinType joinType,
const std::vector<std::string>& probeKeys,
const std::vector<std::string>& buildKeys,
const std::vector<RowVectorPtr>& probeInput,
const std::vector<RowVectorPtr>& buildInput,
const std::vector<std::string>& outputColumns,
const std::string& filter);
const std::vector<core::JoinType>& joinTypes,
const std::vector<std::vector<std::string>>& probeKeysList,
const std::vector<std::vector<std::string>>& buildKeysList,
const std::vector<std::vector<RowVectorPtr>>& inputs,
const std::vector<std::vector<std::string>>& outputColumnsList,
const std::vector<std::string>& filterList);

// Returns a PlanWithSplits for NestedLoopJoin with inputs from Values nodes.
JoinFuzzer::PlanWithSplits makeNestedLoopJoinPlan(
core::JoinType joinType,
const std::vector<RowVectorPtr>& probeInput,
const std::vector<RowVectorPtr>& buildInput,
const std::vector<std::string>& outputColumns,
const std::string& joinCondition);
const std::vector<core::JoinType>& joinTypes,
const std::vector<std::vector<RowVectorPtr>>& inputs,
const std::vector<std::vector<std::string>>& outputColumnsList,
const std::vector<std::string>& joinConditionList);

// Makes the default query plan with table scan as inputs for all of the
// inputs.
Expand Down Expand Up @@ -842,46 +840,65 @@ void addFlippedJoinPlan(
}

JoinFuzzer::PlanWithSplits JoinFuzzer::makeMergeJoinPlan(
core::JoinType joinType,
const std::vector<std::string>& probeKeys,
const std::vector<std::string>& buildKeys,
const std::vector<RowVectorPtr>& probeInput,
const std::vector<RowVectorPtr>& buildInput,
const std::vector<std::string>& outputColumns,
const std::string& filter) {
const std::vector<core::JoinType>& joinTypes,
const std::vector<std::vector<std::string>>& probeKeysList,
const std::vector<std::vector<std::string>>& buildKeysList,
const std::vector<std::vector<RowVectorPtr>>& inputs,
const std::vector<std::vector<std::string>>& outputColumnsList,
const std::vector<std::string>& filterList) {
VELOX_CHECK(inputs.size() > 1);
auto planNodeIdGenerator = std::make_shared<core::PlanNodeIdGenerator>();
return JoinFuzzer::PlanWithSplits{PlanBuilder(planNodeIdGenerator)
.values(probeInput)
.orderBy(probeKeys, false)
.mergeJoin(
probeKeys,
buildKeys,
PlanBuilder(planNodeIdGenerator)
.values(buildInput)
.orderBy(buildKeys, false)
.planNode(),
filter,
outputColumns,
joinType)
.planNode()};
PlanBuilder plan = PlanBuilder(planNodeIdGenerator)
.values(inputs[0])
.orderBy(probeKeysList[0], false)
.mergeJoin(
probeKeysList[0],
buildKeysList[0],
PlanBuilder(planNodeIdGenerator)
.values(inputs[1])
.orderBy(buildKeysList[0], false)
.planNode(),
filterList[0],
outputColumnsList[0],
joinTypes[0]);
for (auto i = 1; i < inputs.size() - 1; i++) {
plan = plan.mergeJoin(
probeKeysList[i],
buildKeysList[i],
PlanBuilder(planNodeIdGenerator)
.values(inputs[i + 1])
.orderBy(buildKeysList[i], false)
.planNode(),
filterList[i],
outputColumnsList[i],
joinTypes[i]);
}
return PlanWithSplits{plan.planNode()};
}

JoinFuzzer::PlanWithSplits JoinFuzzer::makeNestedLoopJoinPlan(
core::JoinType joinType,
const std::vector<RowVectorPtr>& probeInput,
const std::vector<RowVectorPtr>& buildInput,
const std::vector<std::string>& outputColumns,
const std::string& joinCondition) {
const std::vector<core::JoinType>& joinTypes,
const std::vector<std::vector<RowVectorPtr>>& inputs,
const std::vector<std::vector<std::string>>& outputColumnsList,
const std::vector<std::string>& joinConditionList) {
VELOX_CHECK(inputs.size() > 1);
auto planNodeIdGenerator = std::make_shared<core::PlanNodeIdGenerator>();
return JoinFuzzer::PlanWithSplits{
PlanBuilder plan =
PlanBuilder(planNodeIdGenerator)
.values(probeInput)
.values(inputs[0])
.nestedLoopJoin(
PlanBuilder(planNodeIdGenerator).values(buildInput).planNode(),
joinCondition,
outputColumns,
joinType)
.planNode()};
PlanBuilder(planNodeIdGenerator).values(inputs[1]).planNode(),
joinConditionList[0],
outputColumnsList[0],
joinTypes[0]);
for (auto i = 1; i < inputs.size() - 1; i++) {
plan = plan.nestedLoopJoin(
PlanBuilder(planNodeIdGenerator).values(inputs[i + 1]).planNode(),
joinConditionList[i],
outputColumnsList[i],
joinTypes[i]);
}
return PlanWithSplits{plan.planNode()};
}

void JoinFuzzer::makeAlternativePlans(
Expand Down Expand Up @@ -924,13 +941,12 @@ void JoinFuzzer::makeAlternativePlans(
// Use OrderBy + MergeJoin
if (core::MergeJoinNode::isSupported(joinNode->joinType())) {
auto planWithSplits = makeMergeJoinPlan(
joinType,
probeKeys,
buildKeys,
probeInput,
buildInput,
outputColumns,
filter);
{joinType},
{probeKeys},
{buildKeys},
{probeInput, buildInput},
{outputColumns},
{filter});
plans.push_back(planWithSplits);

addFlippedJoinPlan<core::MergeJoinNode>(planWithSplits, plans);
Expand All @@ -943,7 +959,7 @@ void JoinFuzzer::makeAlternativePlans(
: fmt::format(
"{} AND {}", makeJoinFilter(probeKeys, buildKeys), filter);
auto planWithSplits = makeNestedLoopJoinPlan(
joinType, probeInput, buildInput, outputColumns, joinCondition);
{joinType}, {probeInput, buildInput}, {outputColumns}, {joinCondition});
plans.push_back(planWithSplits);

addFlippedJoinPlan<core::NestedLoopJoinNode>(planWithSplits, plans);
Expand Down Expand Up @@ -986,11 +1002,10 @@ RowVectorPtr JoinFuzzer::testCrossProduct(
->names();

auto plan = makeNestedLoopJoinPlan(
joinType,
probeInput,
buildInput,
outputColumns,
/*filter=*/"");
{joinType},
{probeInput, buildInput},
{outputColumns},
/*filterList=*/{""});
const auto expected = execute(plan, /*injectSpill=*/false);

// If OOM injection is not enabled verify the results against Reference query
Expand Down

0 comments on commit a284fb8

Please sign in to comment.