Skip to content

Commit

Permalink
Group AND expressions properly to account for nesting
Browse files Browse the repository at this point in the history
  • Loading branch information
d-m-u committed Apr 30, 2019
1 parent 0e34dea commit 8875a22
Show file tree
Hide file tree
Showing 2 changed files with 27 additions and 9 deletions.
2 changes: 1 addition & 1 deletion lib/miq_expression.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1379,7 +1379,7 @@ def to_arel(exp, tz)
next if arel.blank?
result << arel
end
Arel::Nodes::And.new(operands)
Arel::Nodes::Grouping.new(Arel::Nodes::And.new(operands))
when "or"
operands = exp[operator].each_with_object([]) do |operand, result|
next if operand.blank?
Expand Down
34 changes: 26 additions & 8 deletions spec/lib/miq_expression_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -379,14 +379,14 @@
exp1 = {"STARTS WITH" => {"field" => "Vm-name", "value" => "foo"}}
exp2 = {"ENDS WITH" => {"field" => "Vm-name", "value" => "bar"}}
sql, * = MiqExpression.new("AND" => [exp1, exp2]).to_sql
expect(sql).to eq("\"vms\".\"name\" LIKE 'foo%' AND \"vms\".\"name\" LIKE '%bar'")
expect(sql).to eq("(\"vms\".\"name\" LIKE 'foo%' AND \"vms\".\"name\" LIKE '%bar')")
end

it "generates the SQL for an AND expression where only one is supported by SQL" do
exp1 = {"STARTS WITH" => {"field" => "Vm-name", "value" => "foo"}}
exp2 = {"ENDS WITH" => {"field" => "Vm-platform", "value" => "bar"}}
sql, * = MiqExpression.new("AND" => [exp1, exp2]).to_sql
expect(sql).to eq("\"vms\".\"name\" LIKE 'foo%'")
expect(sql).to eq("(\"vms\".\"name\" LIKE 'foo%')")
end

it "returns nil for an AND expression where none is supported by SQL" do
Expand Down Expand Up @@ -417,12 +417,30 @@
expect(sql).to be_nil
end

it "properly groups the items in an AND/OR expression" do
exp = {"AND" => [{"EQUAL" => {"field" => "Vm-power_state", "value" => "on"}},
{"OR" => [{"EQUAL" => {"field" => "Vm-name", "value" => "foo"}},
{"EQUAL" => {"field" => "Vm-name", "value" => "bar"}}]}]}
sql, * = described_class.new(exp).to_sql
expect(sql).to eq(%q("vms"."power_state" = 'on' AND ("vms"."name" = 'foo' OR "vms"."name" = 'bar')))
context "nested expressions" do
it "properly groups the items in an AND/OR expression" do
exp = {"AND" => [{"EQUAL" => {"field" => "Vm-power_state", "value" => "on"}},
{"OR" => [{"EQUAL" => {"field" => "Vm-name", "value" => "foo"}},
{"EQUAL" => {"field" => "Vm-name", "value" => "bar"}}]}]}
sql, * = described_class.new(exp).to_sql
expect(sql).to eq(%q(("vms"."power_state" = 'on' AND ("vms"."name" = 'foo' OR "vms"."name" = 'bar'))))
end

it "properly groups the items in an OR/AND expression" do
exp = {"OR" => [{"EQUAL" => {"field" => "Vm-power_state", "value" => "on"}},
{"AND" => [{"EQUAL" => {"field" => "Vm-name", "value" => "foo"}},
{"EQUAL" => {"field" => "Vm-name", "value" => "bar"}}]}]}
sql, * = described_class.new(exp).to_sql
expect(sql).to eq(%q(("vms"."power_state" = 'on' OR ("vms"."name" = 'foo' AND "vms"."name" = 'bar'))))
end

it "properly groups the items in an OR/OR expression" do
exp = {"OR" => [{"EQUAL" => {"field" => "Vm-power_state", "value" => "on"}},
{"OR" => [{"EQUAL" => {"field" => "Vm-name", "value" => "foo"}},
{"EQUAL" => {"field" => "Vm-name", "value" => "bar"}}]}]}
sql, * = described_class.new(exp).to_sql
expect(sql).to eq(%q(("vms"."power_state" = 'on' OR ("vms"."name" = 'foo' OR "vms"."name" = 'bar'))))
end
end

it "generates the SQL for a NOT expression" do
Expand Down

0 comments on commit 8875a22

Please sign in to comment.