require File.expand_path("../spec_helper", __dir__) RSpec.describe SearchFlip::Criteria do describe "delegation" do subject { SearchFlip::Criteria.new(target: ProductIndex) } methods = [ :total_entries, :current_page, :previous_page, :prev_page, :next_page, :first_page?, :last_page?, :out_of_range?, :total_pages, :hits, :ids, :count, :size, :length, :took, :aggregations, :suggestions, :scope, :results, :records, :scroll_id, :raw_response ] methods.each do |method| it { should delegate(method).to(:response) } end it { should delegate(:connection).to(:target) } end describe "#to_query" do it "returns the added must, filter and must_not clauses" do query = ProductIndex .must(term: { category: "category1" }) .filter(term: { category: "category2" }) .must_not(term: { category: "category3" }) expect(query.to_query).to eq( bool: { must: [{ term: { category: "category1" } }], filter: [{ term: { category: "category2" } }], must_not: [{ term: { category: "category3" } }] } ) end it "returns only the must clause, if there is only a single must clause" do query = ProductIndex.must(term: { category: "category" }) expect(query.to_query).to eq(term: { category: "category" }) end it "generates an executable query for must, filter and must_not clauses" do query = ProductIndex .must(term: { category: "category1" }) .filter(term: { category: "category2" }) .must_not(term: { category: "category3" }) expect { ProductIndex.must(query.to_query).execute }.not_to raise_error end it "returns the added post_must, post_filter and post_must_not clauses" do query = ProductIndex .post_must(term: { category: "category1" }) .post_filter(term: { category: "category2" }) .post_must_not(term: { category: "category3" }) expect(query.to_query).to eq( bool: { must_not: [{ term: { category: "category3" } }], filter: [ { term: { category: "category1" } }, { term: { category: "category2" } } ] } ) end it "generates an executable query for post_must, post_filter and post_must_not clauses" do query = ProductIndex .post_must(term: { category: "category1" }) .post_filter(term: { category: "category2" }) .post_must_not(term: { category: "category3" }) expect { ProductIndex.must(query.to_query).execute }.not_to raise_error end end describe "#merge" do it "merges criterias" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query = ProductIndex.where(price: 50..250).aggregate(:category).merge(ProductIndex.where(category: "category1")) expect(query.records).to eq([product1]) end describe "assignments" do methods = [ :profile_value, :failsafe_value, :terminate_after_value, :timeout_value, :offset_value, :limit_value, :scroll_args, :source_value, :preference_value, :search_type_value, :routing_value, :track_total_hits_value, :explain_value, :http_timeout_value ] methods.each do |method| it "replaces the values" do criteria1 = SearchFlip::Criteria.new(target: TestIndex) criteria1.send("#{method}=", "value1") criteria2 = SearchFlip::Criteria.new(target: TestIndex) criteria2.send("#{method}=", "value2") expect(criteria1.merge(criteria2).send(method)).to eq("value2") end it "handles false values correctly" do criteria1 = SearchFlip::Criteria.new(target: TestIndex) criteria1.send("#{method}=", true) criteria2 = SearchFlip::Criteria.new(target: TestIndex) criteria2.send("#{method}=", false) expect(criteria1.merge(criteria2).send(method)).to eq(false) end end end describe "array concatenations" do methods = [ :sort_values, :includes_values, :preload_values, :eager_load_values, :must_values, :must_not_values, :filter_values, :post_must_values, :post_must_not_values, :post_filter_values ] methods.each do |method| it "concatenates the values for #{method}" do criteria1 = SearchFlip::Criteria.new(target: TestIndex) criteria1.send("#{method}=", ["value1"]) criteria2 = SearchFlip::Criteria.new(target: TestIndex) criteria2.send("#{method}=", ["value2"]) result = criteria1.merge(criteria2) expect(result.send(method)).to eq(["value1", "value2"]) end end end describe "hash merges" do methods = [ :highlight_values, :suggest_values, :custom_value, :aggregation_values ] methods.each do |method| it "merges the values for #{method}" do criteria1 = SearchFlip::Criteria.new(target: TestIndex) criteria1.send("#{method}=", key1: "value1") criteria2 = SearchFlip::Criteria.new(target: TestIndex) criteria2.send("#{method}=", key2: "value2") result = criteria1.merge(criteria2) expect(result.send(method)).to eq(key1: "value1", key2: "value2") end end end end describe "#criteria" do it "returns self" do criteria = ProductIndex.criteria expect(criteria.criteria.object_id).to eq(criteria.object_id) end end describe "#all" do it "returns self" do criteria = ProductIndex.criteria expect(criteria.all.object_id).to eq(criteria.object_id) end end describe "#timeout" do it "sets the query timeout" do query = ProductIndex.timeout("1s") expect(query.request[:timeout]).to eq("1s") expect { query.execute }.not_to raise_error end end describe "#http_timeout" do it "sets the query timeout" do http_client = double("client").as_null_object allow(http_client).to receive(:timeout).and_return(http_client) allow(http_client).to receive(:post).and_raise(SearchFlip::TimeoutError) allow(ProductIndex.connection).to receive(:http_client).and_return(http_client) expect { ProductIndex.http_timeout(1).execute }.to raise_error(SearchFlip::TimeoutError) expect(http_client).to have_received(:timeout).with(1) end it "executes without errors" do expect { ProductIndex.http_timeout(1).execute }.not_to raise_error end end describe "#terminate_after" do it "sets the terminate after value" do query = ProductIndex.terminate_after(1) expect(query.request[:terminate_after]).to eq(1) expect { query.execute }.not_to raise_error end end describe "#where" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.where(price: 100..200) query2 = query1.where(category: "category1") expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end it "works with arrays" do expected1 = create(:product, title: "expected1") expected2 = create(:product, title: "expected2") rejected = create(:product, title: "rejected") ProductIndex.import [expected1, expected2, rejected] query = ProductIndex.where(title: ["expected1", "expected2"]) expect(query.records.to_set).to eq([expected1, expected2].to_set) end it "works with ranges" do expected1 = create(:product, price: 100) expected2 = create(:product, price: 200) rejected = create(:product, price: 300) ProductIndex.import [expected1, expected2, rejected] query = ProductIndex.where(price: 100..200) expect(query.records.to_set).to eq([expected1, expected2].to_set) end it "works with nils" do expected = create(:product, price: nil) rejected = create(:product, price: 100) ProductIndex.import [expected, rejected] query = ProductIndex.where(price: nil) expect(query.records).to eq([expected]) end end describe "#where_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.where_not(price: 250..350) query2 = query1.where_not(category: "category2") expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end it "works with arrays" do expected = create(:product, title: "expected") rejected1 = create(:product, title: "rejected1") rejected2 = create(:product, title: "rejected2") ProductIndex.import [expected, rejected1, rejected2] query = ProductIndex.where_not(title: ["rejected1", "rejected2"]) expect(query.records).to eq([expected]) end it "works with ranges" do expected = create(:product, price: 100) rejected1 = create(:product, price: 200) rejected2 = create(:product, price: 300) ProductIndex.import [expected, rejected1, rejected2] query = ProductIndex.where_not(price: 200..300) expect(query.records).to eq([expected]) end it "works with nils" do expected = create(:product, price: 100) rejected = create(:product, price: nil) ProductIndex.import [expected, rejected] query = ProductIndex.where_not(price: nil) expect(query.records).to eq([expected]) end end describe "#with_settings" do it "sets the target to the new anonymous class" do query = ProductIndex.where(id: 1).with_settings(index_name: "new_user_index") expect(query.target.name).to be_nil expect(query.target.index_name).to eq("new_user_index") end it "keeps the constraints" do expected = create(:product) rejected = create(:product) ProductIndex.import [expected, rejected] query = ProductIndex.where(id: expected.id).with_settings(index_name: ProductIndex.index_name) expect(query.records).to eq([expected]) end end describe "#filter" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.filter(range: { price: { gte: 100, lte: 200 } }) query2 = query1.filter(term: { category: "category1" }) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end end describe "#must" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.must(range: { price: { gte: 100, lte: 200 } }) query2 = query1.must(term: { category: "category1" }) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end end describe "#must_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.must_not(range: { price: { gt: 200, lte: 300 } }) query2 = query1.must_not(term: { category: "category2" }) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end end describe "#should" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query = ProductIndex.should([ { range: { price: { gte: 100, lt: 200 } } }, { term: { category: "category2" } } ]) expect(query.records.to_set).to eq([product1, product2].to_set) end end describe "#range" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100) product2 = create(:product, price: 200) product3 = create(:product, price: 300) ProductIndex.import [product1, product2, product3] query1 = ProductIndex.range(:price, gte: 100, lte: 200) query2 = query1.range(:price, gte: 200, lte: 300) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product2]) end end describe "#match_all" do it "matches all documents" do expected1 = create(:product) expected2 = create(:product) ProductIndex.import [expected1, expected2] query = ProductIndex.match_all expect(query.records.to_set).to eq([expected1, expected2].to_set) end end describe "#match_none" do it "does not match any documents" do if ProductIndex.connection.version.to_i >= 5 ProductIndex.import create(:product) query = ProductIndex.match_none expect(query.records).to eq([]) end end end describe "#exists" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: "title1", description: "description1") product2 = create(:product, title: "title2", description: nil) product3 = create(:product, title: nil, description: "description2") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.exists(:title) query2 = query1.exists(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end end describe "#exists_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: nil, description: nil) product2 = create(:product, title: nil, description: "description2") product3 = create(:product, title: "title3", description: "description3") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.exists_not(:title) query2 = query1.exists_not(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) end end describe "#post_search" do it "sets up the constraints correctly and is chainable" do if ProductIndex.connection.version.to_i >= 2 product1 = create(:product, title: "title1", category: "category1") product2 = create(:product, title: "title2", category: "category2") product3 = create(:product, title: "title3", category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_search("title1 OR title2") query2 = query1.post_search("category1") expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations1 = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations1).to eq("category1" => 2, "category2" => 1) aggregations2 = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations2).to eq("category1" => 2, "category2" => 1) end end end describe "#post_where" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_where(price: 100..200) query2 = query1.post_where(category: "category1") expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations1 = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations1).to eq("category1" => 2, "category2" => 1) aggregations2 = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations2).to eq("category1" => 2, "category2" => 1) end it "works with arrays" do expected1 = create(:product, title: "expected1", category: "category1") expected2 = create(:product, title: "expected2", category: "category2") rejected = create(:product, title: "rejected", category: "category1") ProductIndex.import [expected1, expected2, rejected] query = ProductIndex.aggregate(:category).post_where(title: ["expected1", "expected2"]) expect(query.records.to_set).to eq([expected1, expected2].to_set) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end it "works with ranges" do expected1 = create(:product, price: 100, category: "category1") expected2 = create(:product, price: 200, category: "category2") rejected = create(:product, price: 300, category: "category1") ProductIndex.import [expected1, expected2, rejected] query = ProductIndex.aggregate(:category).post_where(price: 100..200) expect(query.records.to_set).to eq([expected1, expected2].to_set) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end it "works with nil" do expected1 = create(:product, price: nil, category: "category1") expected2 = create(:product, price: nil, category: "category2") rejected = create(:product, price: 300, category: "category1") ProductIndex.import [expected1, expected2, rejected] query = ProductIndex.aggregate(:category).post_where(price: nil) expect(query.records.to_set).to eq([expected1, expected2].to_set) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_where_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_where_not(price: 250..350) query2 = query1.post_where_not(category: "category2") expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end it "works with arrays" do expected = create(:product, title: "expected", category: "category1") rejected1 = create(:product, title: "rejected1", category: "category2") rejected2 = create(:product, title: "rejected2", category: "category1") ProductIndex.import [expected, rejected1, rejected2] query = ProductIndex.aggregate(:category).post_where_not(title: ["rejected1", "rejected2"]) expect(query.records).to eq([expected]) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end it "works with ranges" do expected = create(:product, price: 100, category: "category1") rejected1 = create(:product, price: 200, category: "category2") rejected2 = create(:product, price: 300, category: "category1") ProductIndex.import [expected, rejected1, rejected2] query = ProductIndex.aggregate(:category).post_where_not(price: 200..300) expect(query.records).to eq([expected]) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end it "works with nils" do expected = create(:product, price: 100, category: "category1") rejected1 = create(:product, price: nil, category: "category2") rejected2 = create(:product, price: nil, category: "category1") ProductIndex.import [expected, rejected1, rejected2] query = ProductIndex.aggregate(:category).post_where_not(price: nil) expect(query.records).to eq([expected]) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_filter" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_filter(range: { price: { gte: 100, lte: 200 } }) query2 = query1.post_filter(term: { category: "category1" }) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_must" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_must(range: { price: { gte: 100, lte: 200 } }) query2 = query1.post_must(term: { category: "category1" }) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_must_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_must_not(range: { price: { gte: 50, lte: 150 } }) query2 = query1.post_must_not(term: { category: "category1" }) expect(query1.records.to_set).to eq([product2, product3].to_set) expect(query2.records).to eq([product2]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_should" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category3") product3 = create(:product, price: 300, category: "category2") product4 = create(:product, price: 400, category: "category1") ProductIndex.import [product1, product2, product3, product4] query1 = ProductIndex.aggregate(:category).post_should([ { term: { category: "category1" } }, { term: { category: "category2" } } ]) query2 = query1.post_should([ { range: { price: { gte: 50, lte: 150 } } }, { range: { price: { gte: 250, lte: 350 } } } ]) expect(query1.records.to_set).to eq([product1, product3, product4].to_set) expect(query2.records.to_set).to eq([product1, product3].to_set) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1, "category3" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1, "category3" => 1) end end describe "#post_range" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, price: 100, category: "category1") product2 = create(:product, price: 200, category: "category2") product3 = create(:product, price: 300, category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_range(:price, gte: 100, lte: 200) query2 = query1.post_range(:price, gte: 200, lte: 300) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product2]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_exists" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: "title1", description: "description1", category: "category1") product2 = create(:product, title: "title2", description: nil, category: "category2") product3 = create(:product, title: nil, description: "description2", category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_exists(:title) query2 = query1.post_exists(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_exists" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: "title1", description: "description1", category: "category1") product2 = create(:product, title: "title2", description: nil, category: "category2") product3 = create(:product, title: nil, description: "description2", category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_exists(:title) query2 = query1.post_exists(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_exists" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: "title1", description: "description1", category: "category1") product2 = create(:product, title: "title2", description: nil, category: "category2") product3 = create(:product, title: nil, description: "description2", category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_exists(:title) query2 = query1.post_exists(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#post_exists_not" do it "sets up the constraints correctly and is chainable" do product1 = create(:product, title: nil, description: nil, category: "category1") product2 = create(:product, title: nil, description: "description2", category: "category2") product3 = create(:product, title: "title3", description: "description3", category: "category1") ProductIndex.import [product1, product2, product3] query1 = ProductIndex.aggregate(:category).post_exists_not(:title) query2 = query1.post_exists_not(:description) expect(query1.records.to_set).to eq([product1, product2].to_set) expect(query2.records).to eq([product1]) aggregations = query1.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) aggregations = query2.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 2, "category2" => 1) end end describe "#aggregate" do it "sets up the constraints correctly and is chainable" do ProductIndex.import create_list(:product, 3, category: "category1", price: 10) ProductIndex.import create_list(:product, 2, category: "category2", price: 20) ProductIndex.import create_list(:product, 1, category: "category3", price: 30) query = ProductIndex.aggregate(:category, size: 2).aggregate(price_sum: { sum: { field: "price" } }) category_aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } price_aggregation = query.aggregations(:price_sum).value expect(category_aggregations).to eq("category1" => 3, "category2" => 2) expect(price_aggregation).to eq(100) end it "works with hashes" do ProductIndex.import create_list(:product, 3, category: "category1") ProductIndex.import create_list(:product, 2, category: "category2") ProductIndex.import create_list(:product, 1, category: "category3") query = ProductIndex.aggregate(category: { terms: { field: :category } }) aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } expect(aggregations).to eq("category1" => 3, "category2" => 2, "category3" => 1) end it "allows sub-aggregations" do ProductIndex.import create_list(:product, 3, category: "category1", price: 15) ProductIndex.import create_list(:product, 2, category: "category2", price: 20) ProductIndex.import create_list(:product, 1, category: "category3", price: 25) query = ProductIndex.aggregate(:category) do |aggregation| aggregation.aggregate(price_sum: { sum: { field: "price" } }) end category_aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.doc_count } price_sum_aggregations = query.aggregations(:category).each_with_object({}) { |(key, agg), hash| hash[key] = agg.price_sum.value } expect(category_aggregations).to eq("category1" => 3, "category2" => 2, "category3" => 1) expect(price_sum_aggregations).to eq("category1" => 45, "category2" => 40, "category3" => 25) end end describe "#profile" do it "sets up the constraints correctly" do if ProductIndex.connection.version.to_i >= 2 expect(ProductIndex.profile(true).raw_response["profile"]).not_to be_nil end end end describe "#scroll" do it "scrolls over the full result set" do products = create_list(:product, 15) ProductIndex.import products criteria = ProductIndex.limit(10).scroll(timeout: "1m") result = [] iterations = 0 until criteria.records.empty? result += criteria.records iterations += 1 criteria = criteria.scroll(id: criteria.scroll_id, timeout: "1m") end expect(products.to_set).to eq(result.to_set) expect(iterations).to eq(2) end end describe "#delete" do it "delets the matching documents" do product1, product2, product3 = create_list(:product, 3) ProductIndex.import [product1, product2, product3] expect { ProductIndex.where(id: [product1.id, product2.id]).delete }.to change { ProductIndex.total_count }.by(-2) end it "accepts additional params" do product = create(:product) ProductIndex.import(product) expect { ProductIndex.where(id: product.id).delete(conflicts: "proceed") }.to change { ProductIndex.total_count }.by(-1) end end describe "#source" do it "constraints the returned source fields" do product = create(:product, title: "Title", price: 10) ProductIndex.import product results = ProductIndex.where(id: product.id).results expect(results.first.id).not_to be_nil expect(results.first.title).not_to be_nil expect(results.first.price).not_to be_nil results = ProductIndex.where(id: product.id).source([:id, :price]).results expect(results.first.id).not_to be_nil expect(results.first.title).to be_nil expect(results.first.price).not_to be_nil end end describe "#includes" do it "does not raise any errors" do user = create(:user) comments = create_list(:comment, 2) product = create(:product, user: user, comments: comments) ProductIndex.import product record = ProductIndex.includes(:user).includes(:comments).records.first expect(record).not_to be_nil expect(record.user).to eq(user) expect(record.comments.to_set).to eq(comments.to_set) end end describe "#eager_load" do it "does not raise any errors" do user = create(:user) comments = create_list(:comment, 2) product = create(:product, user: user, comments: comments) ProductIndex.import product record = ProductIndex.eager_load(:user).eager_load(:comments).records.first expect(record).not_to be_nil expect(record.user).to eq(user) expect(record.comments.to_set).to eq(comments.to_set) end end describe "#preload" do it "does not raise any errors" do user = create(:user) comments = create_list(:comment, 2) product = create(:product, user: user, comments: comments) ProductIndex.import product record = ProductIndex.preload(:user).preload(:comments).records.first expect(record).not_to be_nil expect(record.user).to eq(user) expect(record.comments.to_set).to eq(comments.to_set) end end describe "#sort" do it "sorts correctly and is chainable" do product1 = create(:product, rank: 2, price: 100) product2 = create(:product, rank: 2, price: 90) product3 = create(:product, rank: 1, price: 120) product4 = create(:product, rank: 0, price: 110) ProductIndex.import [product1, product2, product3, product4] expect(ProductIndex.sort({ rank: :desc }, price: :asc).records).to eq([product2, product1, product3, product4]) expect(ProductIndex.sort(rank: :desc).sort(:price).records).to eq([product2, product1, product3, product4]) expect(ProductIndex.sort(:price).sort(rank: :desc).records).to eq([product2, product1, product4, product3]) end end describe "#resort" do it "overwrites existing sort criterias" do product1 = create(:product, rank: 2, price: 100) product2 = create(:product, rank: 2, price: 90) product3 = create(:product, rank: 1, price: 120) product4 = create(:product, rank: 0, price: 110) ProductIndex.import [product1, product2, product3, product4] expect(ProductIndex.sort(:price).resort({ rank: :desc }, price: :asc).records).to eq([product2, product1, product3, product4]) expect(ProductIndex.sort(rank: :desc).resort(:price).records).to eq([product2, product1, product4, product3]) end end describe "#offset" do it "sets the query document offset" do product1 = create(:product, rank: 1) product2 = create(:product, rank: 2) product3 = create(:product, rank: 3) ProductIndex.import [product1, product2, product3] query = ProductIndex.sort(:rank).offset(1) expect(query.records).to eq([product2, product3]) expect(query.offset(2).records).to eq([product3]) end end describe "#limit" do it "sets the query document limit" do product1 = create(:product, rank: 1) product2 = create(:product, rank: 2) product3 = create(:product, rank: 3) ProductIndex.import [product1, product2, product3] query = ProductIndex.sort(:rank).limit(1) expect(query.records).to eq([product1]) expect(query.limit(2).records).to eq([product1, product2]) end end describe "#paginate" do it "sets the query document offset and limit" do product1 = create(:product, rank: 1) product2 = create(:product, rank: 2) product3 = create(:product, rank: 3) ProductIndex.import [product1, product2, product3] query = ProductIndex.sort(:rank).paginate(page: 1, per_page: 2) expect(query.records).to eq([product1, product2]) expect(query.paginate(page: 2, per_page: 2).records).to eq([product3]) end end describe "#page" do it "sets the query document offset" do expect(ProductIndex.page(1).offset_value).to eq(0) expect(ProductIndex.page(2).offset_value).to eq(30) expect(ProductIndex.page(3).per(50).offset_value).to eq(100) end end describe "#limit" do it "sets the query document limit" do expect(ProductIndex.per(50).limit_value).to eq(50) end end describe "#search" do it "sets up the constraints correctly" do product1 = create(:product, title: "Title1", description: "Description1", price: 10) product2 = create(:product, title: "Title2", description: "Description2", price: 20) product3 = create(:product, title: "Title3", description: "Description2", price: 30) ProductIndex.import [product1, product2, product3] expect(ProductIndex.search("Title1 OR Title3").records.to_set).to eq([product1, product3].to_set) expect(ProductIndex.search("Title1 Title3", default_operator: :OR).records.to_set).to eq([product1, product3].to_set) expect(ProductIndex.search("Title1 OR Title2").search("Title1 OR Title3").records).to eq([product1]) expect(ProductIndex.search("Title1 OR Title3").where(price: 5..15).records).to eq([product1]) end end describe "#highlight" do it "adds highlighting to the query and is chainable" do product1 = create(:product, title: "Title1 highlight", description: "Description1 highlight") product2 = create(:product, title: "Title2 highlight", description: "Description2 highlight") ProductIndex.import [product1, product2] results = ProductIndex.sort(:id).highlight([:title, :description]).search("title:highlight description:highlight").results expect(results[0]._hit.highlight.title).to eq(["Title1 highlight"]) expect(results[0]._hit.highlight.description).to eq(["Description1 highlight"]) expect(results[1]._hit.highlight.title).to eq(["Title2 highlight"]) expect(results[1]._hit.highlight.description).to eq(["Description2 highlight"]) results = ProductIndex.sort(:id).highlight([:title, :description], require_field_match: false).search("highlight").results expect(results[0]._hit.highlight.title).to eq(["Title1 highlight"]) expect(results[0]._hit.highlight.description).to eq(["Description1 highlight"]) expect(results[1]._hit.highlight.title).to eq(["Title2 highlight"]) expect(results[1]._hit.highlight.description).to eq(["Description2 highlight"]) query = ProductIndex.sort(:id).search("title:highlight") query = query.highlight(:title, require_field_match: true).highlight(:description, require_field_match: true) results = query.results expect(results[0]._hit.highlight.title).to eq(["Title1 highlight"]) expect(results[0]._hit.highlight.description).to be_nil expect(results[1]._hit.highlight.title).to eq(["Title2 highlight"]) expect(results[1]._hit.highlight.description).to be_nil end end describe "#suggest" do it "adds suggest to the query" do product = create(:product, title: "Title", description: "Description") ProductIndex.import product suggestions = ProductIndex.suggest(:suggestion, text: "Desciption", term: { field: "description" }).suggestions(:suggestion) expect(suggestions.first["text"]).to eq("description") end end describe "#find_in_batches" do it "iterates the records in batches of the specified size" do expected1 = create(:product, title: "expected", rank: 1) expected2 = create(:product, title: "expected", rank: 2) expected3 = create(:product, title: "expected", rank: 3) rejected = create(:product, title: "rejected") create :product, title: "rejected" ProductIndex.import [expected1, expected2, expected3, rejected] actual = ProductIndex.where(title: "expected").sort(:rank).find_in_batches(batch_size: 2).to_a expect(actual).to eq([[expected1, expected2], [expected3]]) end end describe "#find_results_in_batches" do it "iterates the results in batches of the specified size" do expected1 = create(:product, title: "expected", rank: 1) expected2 = create(:product, title: "expected", rank: 2) expected3 = create(:product, title: "expected", rank: 3) rejected = create(:product, title: "rejected") create :product, title: "rejected" ProductIndex.import [expected1, expected2, expected3, rejected] actual = ProductIndex.where(title: "expected").sort(:rank).find_results_in_batches(batch_size: 2).map { |batch| batch.map(&:id) } expect(actual).to eq([[expected1.id, expected2.id], [expected3.id]]) end end describe "#find_each" do it "iterates the records" do expected1 = create(:product, title: "expected", rank: 1) expected2 = create(:product, title: "expected", rank: 2) expected3 = create(:product, title: "expected", rank: 3) rejected = create(:product, title: "rejected") create :product, title: "rejected" ProductIndex.import [expected1, expected2, expected3, rejected] actual = ProductIndex.where(title: "expected").sort(:rank).find_each(batch_size: 2).to_a expect(actual).to eq([expected1, expected2, expected3]) end end describe "#find_each_result" do it "iterates the results" do expected1 = create(:product, title: "expected", rank: 1) expected2 = create(:product, title: "expected", rank: 2) expected3 = create(:product, title: "expected", rank: 3) rejected = create(:product, title: "rejected") create :product, title: "rejected" ProductIndex.import [expected1, expected2, expected3, rejected] actual = ProductIndex.where(title: "expected").sort(:rank).find_each_result(batch_size: 2).map(&:id) expect(actual).to eq([expected1.id, expected2.id, expected3.id]) end end describe "#failsafe" do [SearchFlip::ConnectionError, SearchFlip::TimeoutError, SearchFlip::ResponseError.new(code: "code", body: "body")].each do |error| it "prevents #{error}" do http_client = double("client").as_null_object allow(http_client).to receive(:post).and_raise(error) allow(ProductIndex.connection).to receive(:http_client).and_return(http_client) expect { ProductIndex.all.execute }.to raise_error(error) query = ProductIndex.failsafe(true) expect(query.records).to eq([]) expect(query.total_entries).to eq(0) end end end describe "#fresh" do it "returns a new criteria without a cached response" do create :product query = ProductIndex.criteria.tap(&:records) expect(query.instance_variable_get(:@request)).not_to be_nil expect(query.instance_variable_get(:@response)).not_to be_nil expect(query.object_id).not_to eq(query.fresh.object_id) expect(query.fresh.instance_variable_get(:@request)).to be_nil expect(query.fresh.instance_variable_get(:@response)).to be_nil end end describe "#respond_to?" do it "checks whether or not the index class responds to the method" do temp_index = Class.new(ProductIndex) expect(temp_index.criteria.respond_to?(:test_scope)).to eq(false) temp_index.scope(:test_scope) { match_all } expect(temp_index.criteria.respond_to?(:test_scope)).to eq(true) end end describe "#method_missing" do it "delegates to the index class" do temp_index = Class.new(ProductIndex) expected = create(:product, title: "expected") rejected = create(:product, title: "rejected") temp_index.import [expected, rejected] temp_index.scope(:with_title) { |title| where(title: title) } records = temp_index.criteria.with_title("expected").records expect(records).to eq([expected]) end end describe "#execute" do around do |example| default_instrumenter = SearchFlip::Config[:instrumenter] SearchFlip::Config[:instrumenter] = ActiveSupport::Notifications.instrumenter begin example.run ensure SearchFlip::Config[:instrumenter] = default_instrumenter end end let(:notifications) { [] } let!(:subscriber) do ActiveSupport::Notifications.subscribe("request.search_flip") do |*args| notifications << args end end after { ActiveSupport::Notifications.unsubscribe(subscriber) } it "instruments the request" do ProductIndex.match_all.execute expect(notifications).to be_present end it "passes the index, request and response" do ProductIndex.match_all.execute expect(notifications.first[4][:index]).to eq(ProductIndex) expect(notifications.first[4][:request]).to be_present expect(notifications.first[4][:response]).to be_present end end describe "#track_total_hits" do it "is added to the request" do if ProductIndex.connection.version.to_i >= 7 query = ProductIndex.track_total_hits(false) expect(query.request[:track_total_hits]).to eq(false) expect { query.execute }.not_to raise_error end end end describe "#explain" do it "returns the explaination" do ProductIndex.import create(:product) query = ProductIndex.match_all.explain(true) expect(query.results.first._hit.key?(:_explanation)).to eq(true) end end describe "#custom" do it "adds a custom entry to the request" do request = ProductIndex.custom(custom_key: "custom_value").request expect(request[:custom_key]).to eq("custom_value") end end describe "#preference" do it "sets the preference" do url = ProductIndex.connection.version.to_i < 8 ? "products/products" : "products" stub_request(:post, "http://127.0.0.1:9200/#{url}/_search?preference=value") .to_return(status: 200, headers: { content_type: "application/json" }, body: "{}") ProductIndex.preference("value").execute expect(WebMock).to have_requested(:post, "http://127.0.0.1:9200/#{url}/_search?preference=value") end end describe "#search_type" do it "sets the search_type" do url = ProductIndex.connection.version.to_i < 8 ? "products/products" : "products" stub_request(:post, "http://127.0.0.1:9200/#{url}/_search?search_type=value") .to_return(status: 200, headers: { content_type: "application/json" }, body: "{}") ProductIndex.search_type("value").execute expect(WebMock).to have_requested(:post, "http://127.0.0.1:9200/#{url}/_search?search_type=value") end end describe "#routing" do it "sets the search_type" do url = ProductIndex.connection.version.to_i < 8 ? "products/products" : "products" stub_request(:post, "http://127.0.0.1:9200/#{url}/_search?routing=value") .to_return(status: 200, headers: { content_type: "application/json" }, body: "{}") ProductIndex.routing("value").execute expect(WebMock).to have_requested(:post, "http://127.0.0.1:9200/#{url}/_search?routing=value") end end end