require "minitest" require "minitest/autorun" require "webmock/minitest" require "mocha/minitest" require "search_flip" require "active_record" require "factory_bot" require "timecop" require "yaml" WebMock.allow_net_connect! ActiveRecord::Base.establish_connection YAML.load_file(File.expand_path("../database.yml", __FILE__)) SearchFlip::Config[:auto_refresh] = true ActiveRecord::Base.connection.execute "DROP TABLE IF EXISTS products" ActiveRecord::Base.connection.create_table :products do |t| t.string :title t.text :description t.float :price t.string :category t.integer :version, default: 1 t.integer :rank, :default => 0 t.integer :user_id t.timestamps null: false end ActiveRecord::Base.connection.add_index :products, :user_id class Product < ActiveRecord::Base belongs_to :user has_many :comments end FactoryBot.define do factory :product end ActiveRecord::Base.connection.execute "DROP TABLE IF EXISTS users" ActiveRecord::Base.connection.create_table :users do |t| t.string :name t.timestamps null: false end class User < ActiveRecord::Base has_many :products end FactoryBot.define do factory :user end ActiveRecord::Base.connection.execute "DROP TABLE IF EXISTS comments" ActiveRecord::Base.connection.create_table :comments do |t| t.text :message t.integer :product_id t.timestamps null: false end ActiveRecord::Base.connection.add_index :comments, :product_id class Comment < ActiveRecord::Base belongs_to :product end FactoryBot.define do factory :comment end class CommentIndex include SearchFlip::Index def self.type_name "comments" end def self.model Comment end def self.serialize(comment) { id: comment.id, message: comment.message } end end CommentIndex.delete_index if CommentIndex.index_exists? CommentIndex.create_index CommentIndex.update_mapping class ProductIndex include SearchFlip::Index def self.mapping if SearchFlip.version.to_i >= 5 { products: { properties: { category: { type: "text", fielddata: true }, title: { type: "text", fielddata: true }, description: { type: "text", fielddata: true } } } } else { products: {} } end end def self.type_name "products" end def self.model Product end def self.serialize(product) { id: product.id, title: product.title, description: product.description, category: product.category, price: product.price, rank: product.rank, created_at: product.created_at, updated_at: product.updated_at } end end ProductIndex.delete_index if ProductIndex.index_exists? ProductIndex.create_index ProductIndex.update_mapping class TestIndex include SearchFlip::Index def self.mapping { test: { properties: { test_field: { type: "date" } } } } end def self.type_name "test" end end TestIndex.delete_index if TestIndex.index_exists? class SearchFlip::TestCase < MiniTest::Test include FactoryBot::Syntax::Methods def self.should_delegate_method(method, to:, subject:, as: method) define_method :"test_delegate_#{method}_to_#{to}" do assert subject.respond_to?(method), "subject doesn't respond to #{method}" target = subject.send(to) assert target.respond_to?(as), "#{to} doesn't respond to #{as}" params = subject.method(method).arity.abs.times.map { |i| "param-#{i}" } mock_target = mock mock_target.expects(as).with(*params) subject.stubs(to).returns(mock_target) subject.send(method, *params) end end def self.should_delegate_methods(*methods, to:, subject:) methods.each do |method| should_delegate_method method, to: to, subject: subject end end def assert_difference(expressions, difference = 1, &block) callables = Array(expressions).map { |e| lambda { eval(e, block.binding) } } before = callables.map(&:call) res = yield Array(expressions).zip(callables).each_with_index do |(code, callable), i| assert_equal before[i] + difference, callable.call, "#{code.inspect} didn't change by #{difference}" end res end def assert_no_difference(expressions, &block) assert_difference(expressions, 0, &block) end def assert_not_nil(object) assert !object.nil?, "shouldn't be nil" end def assert_present(object) assert object.present?, "should be present" end def assert_blank(object) assert object.blank?, "should be blank" end def refute_present(object) refute object.present?, "shouldn't be present" end def refute_blank(object) refute object.blank?, "shouldn't be blank" end def setup ProductIndex.match_all.delete Product.delete_all end end