diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
index a89774b1..226586b9 100644
--- a/.github/workflows/ci.yml
+++ b/.github/workflows/ci.yml
@@ -36,33 +36,28 @@ jobs:
strategy:
fail-fast: false
matrix:
- ruby-version: ['2.7', '3.0', '3.1']
+ ruby-version: ['2.7', '3.0', '3.1', '3.2']
active-record-version-env:
- - ACTIVE_RECORD_VERSION="~> 5.2.0"
- ACTIVE_RECORD_VERSION="~> 6.0.0"
- ACTIVE_RECORD_VERSION="~> 6.1.0"
- ACTIVE_RECORD_VERSION="~> 7.0.0"
allow-failure: [false]
include:
- - ruby-version: '3.1'
+ - ruby-version: '3.2'
active-record-version-env: ACTIVE_RECORD_BRANCH="main"
allow-failure: true
- - ruby-version: '3.1'
+ - ruby-version: '3.2'
active-record-version-env: ACTIVE_RECORD_BRANCH="7-0-stable"
allow-failure: true
- - ruby-version: '3.1'
+ - ruby-version: '3.2'
active-record-version-env: ACTIVE_RECORD_BRANCH="6-1-stable"
allow-failure: true
- exclude:
- - ruby-version: '3.0'
- active-record-version-env: ACTIVE_RECORD_VERSION="~> 5.2.0"
- allow-failure: false
- - ruby-version: '3.1'
- active-record-version-env: ACTIVE_RECORD_VERSION="~> 5.2.0"
- allow-failure: false
+ - ruby-version: '3.3.0-preview1'
+ active-record-version-env: ACTIVE_RECORD_VERSION="~> 7.0.0"
+ allow-failure: true
continue-on-error: ${{ matrix.allow-failure }}
steps:
- - uses: actions/checkout@v2
+ - uses: actions/checkout@v3
- name: Set up Ruby
uses: ruby/setup-ruby@v1
with:
diff --git a/.rubocop.yml b/.rubocop.yml
index 5bb6075e..7d946471 100644
--- a/.rubocop.yml
+++ b/.rubocop.yml
@@ -1,137 +1,34 @@
require:
+ - standard
- rubocop-performance
- rubocop-rails
- rubocop-rake
- rubocop-rspec
+inherit_gem:
+ standard: config/base.yml
+
AllCops:
TargetRubyVersion: 2.7
- NewCops: enable
+ NewCops: disable
Exclude:
- bin/**/*
- vendor/**/*
-Style/StringLiterals:
- Enabled: false
-
-Layout/LineLength:
- Max: 120
-
-Metrics/MethodLength:
- Max: 15
-
-Metrics/BlockLength:
- Exclude:
- - spec/**/*
-
-Layout/ParameterAlignment:
- EnforcedStyle: with_fixed_indentation
-
-Style/NumericPredicate:
- Enabled: false
-
-Style/PercentLiteralDelimiters:
- PreferredDelimiters:
- '%w': '[]'
- '%W': '[]'
-
-Style/GuardClause:
- Enabled: false
-
-Naming/VariableNumber:
- EnforcedStyle: snake_case
-
-Bundler/OrderedGems:
- Enabled: false
-
-Bundler/DuplicatedGem:
- Enabled: false
-
-Style/EmptyMethod:
- EnforcedStyle: expanded
-
-Layout/FirstArrayElementIndentation:
- EnforcedStyle: consistent
-
-Style/Documentation:
- Enabled: false
-
-Style/WordArray:
- EnforcedStyle: percent
- MinSize: 3
-
-Style/HashEachMethods:
+Lint/RedundantCopDisableDirective:
Enabled: true
-Style/HashTransformKeys:
+Lint/RedundantCopEnableDirective:
Enabled: true
-Style/HashTransformValues:
- Enabled: true
-
-Rails/ApplicationRecord:
+Bundler/DuplicatedGem:
Enabled: false
-Rails/TimeZone:
+Rails/ApplicationRecord:
Enabled: false
-RSpec/ContextWording:
- Prefixes:
- - using
- - via
- - when
- - with
- - without
-
-Lint/RaiseException:
- Enabled: true
-
-Lint/StructNewOverride:
- Enabled: true
-
-Layout/SpaceAroundMethodCallOperator:
- Enabled: true
-
-Style/ExponentialNotation:
- Enabled: true
-
-RSpec/DescribedClass:
- Enabled: true
-
-RSpec/ExpectInHook:
+Rails/RakeEnvironment:
Enabled: false
-RSpec/FilePath:
- CustomTransform:
- TSearch: "tsearch"
- DMetaphone: "dmetaphone"
-
-Layout/EmptyLinesAroundAttributeAccessor:
- Enabled: true
-
-Lint/DeprecatedOpenSSLConstant:
- Enabled: true
-
-Style/SlicingWithRange:
- Enabled: true
-
-Lint/MixedRegexpCaptureTypes:
- Enabled: true
-
-Style/RedundantFetchBlock:
- Enabled: true
-
-Style/RedundantRegexpCharacterClass:
- Enabled: true
-
-Style/RedundantRegexpEscape:
- Enabled: true
-
-RSpec/MultipleExpectations:
- Max: 5
-
-RSpec/ExampleLength:
- Max: 15
-
-Rails/RakeEnvironment:
+Rails/TimeZone:
Enabled: false
diff --git a/.tool-versions b/.tool-versions
new file mode 100644
index 00000000..974865fc
--- /dev/null
+++ b/.tool-versions
@@ -0,0 +1 @@
+ruby 2.7.6
diff --git a/Gemfile b/Gemfile
index ea35689d..6d374066 100644
--- a/Gemfile
+++ b/Gemfile
@@ -1,15 +1,30 @@
# frozen_string_literal: true
-source 'https://rubygems.org'
+source "https://rubygems.org"
gemspec
-gem 'pg', '>= 0.21.0', platform: :ruby
+gem "pg", ">= 0.21.0", platform: :ruby
gem "activerecord-jdbcpostgresql-adapter", ">= 1.3.1", platform: :jruby
-if ENV['ACTIVE_RECORD_BRANCH']
- gem 'activerecord', git: 'https://github.com/rails/rails.git', branch: ENV.fetch('ACTIVE_RECORD_BRANCH', nil)
- gem 'arel', git: 'https://github.com/rails/arel.git' if ENV.fetch('ACTIVE_RECORD_BRANCH', nil) == 'master'
+if ENV["ACTIVE_RECORD_BRANCH"]
+ gem "activerecord", git: "https://github.com/rails/rails.git", branch: ENV.fetch("ACTIVE_RECORD_BRANCH", nil)
+ gem "arel", git: "https://github.com/rails/arel.git" if ENV.fetch("ACTIVE_RECORD_BRANCH", nil) == "master"
end
-gem 'activerecord', ENV.fetch('ACTIVE_RECORD_VERSION', nil) if ENV['ACTIVE_RECORD_VERSION']
+gem "activerecord", ENV.fetch("ACTIVE_RECORD_VERSION", nil) if ENV["ACTIVE_RECORD_VERSION"]
+
+gem "pry"
+gem "rake"
+gem "rspec"
+gem "rubocop"
+gem "rubocop-performance"
+gem "rubocop-rails"
+gem "rubocop-rake"
+gem "rubocop-rspec"
+gem "simplecov"
+gem "simplecov-lcov"
+gem "standard", ">= 1.23.0"
+gem "undercover"
+gem "warning"
+gem "with_model"
diff --git a/README.md b/README.md
index 3b70c3b5..b452b37e 100644
--- a/README.md
+++ b/README.md
@@ -14,7 +14,7 @@ Read the blog post introducing PgSearch at https://tanzu.vmware.com/content/blog
## REQUIREMENTS
* Ruby 2.7+
-* ActiveRecord 5.2+
+* Active Record 6.0+
* PostgreSQL 9.2+
* [PostgreSQL extensions](https://github.com/Casecommons/pg_search/wiki/Installing-PostgreSQL-Extensions) for certain features
@@ -979,7 +979,7 @@ Sentence.word_similarity_like("word") # => [sentence]
### Limiting Fields When Combining Features
Sometimes when doing queries combining different features you
-might want to searching against only some of the fields with certain features.
+might want to search against only some of the fields with certain features.
For example perhaps you want to only do a trigram search against the shorter fields
so that you don't need to reduce the threshold excessively. You can specify
which fields using the 'only' option:
diff --git a/Rakefile b/Rakefile
index fd5be07e..6167eb4c 100644
--- a/Rakefile
+++ b/Rakefile
@@ -1,9 +1,9 @@
# frozen_string_literal: true
-require 'bundler'
+require "bundler"
Bundler::GemHelper.install_tasks
-require 'rspec/core/rake_task'
+require "rspec/core/rake_task"
RSpec::Core::RakeTask.new(:spec)
require "rubocop/rake_task"
diff --git a/lib/pg_search/configuration.rb b/lib/pg_search/configuration.rb
index ce37edf9..5dd88f4d 100644
--- a/lib/pg_search/configuration.rb
+++ b/lib/pg_search/configuration.rb
@@ -80,7 +80,7 @@ def order_within_rank
attr_reader :options
def default_options
- { using: :tsearch }
+ {using: :tsearch}
end
VALID_KEYS = %w[
diff --git a/lib/pg_search/configuration/association.rb b/lib/pg_search/configuration/association.rb
index d9c17108..90faf879 100644
--- a/lib/pg_search/configuration/association.rb
+++ b/lib/pg_search/configuration/association.rb
@@ -40,21 +40,20 @@ def selects
def selects_for_singular_association
columns.map do |column|
if column.tsvector_column
- "tsvector_agg(#{column.full_name}) AS #{column.alias}"
+ "#{column.full_name}::tsvector AS #{column.alias}"
else
- case postgresql_version
- when 0..90000
- "array_to_string(array_agg(#{column.full_name}::text), ' ') AS #{column.alias}"
- else
- "string_agg(#{column.full_name}::text, ' ') AS #{column.alias}"
- end
+ "#{column.full_name}::text AS #{column.alias}"
end
end.join(", ")
end
def selects_for_multiple_association
columns.map do |column|
- "string_agg(#{column.full_name}::text, ' ') AS #{column.alias}"
+ if column.tsvector_column
+ "tsvector_agg(#{column.full_name}) AS #{column.alias}"
+ else
+ "string_agg(#{column.full_name}::text, ' ') AS #{column.alias}"
+ end
end.join(", ")
end
diff --git a/lib/pg_search/configuration/column.rb b/lib/pg_search/configuration/column.rb
index 1cf9dedb..b61d4060 100644
--- a/lib/pg_search/configuration/column.rb
+++ b/lib/pg_search/configuration/column.rb
@@ -1,6 +1,6 @@
# frozen_string_literal: true
-require 'digest'
+require "digest"
module PgSearch
class Configuration
@@ -8,8 +8,8 @@ class Column
attr_reader :weight, :tsvector_column, :name
def initialize(column_name, weight, model)
- @name = column_name.to_s
- @column_name = column_name.to_s
+ @name = column_name.to_s
+ @column_name = column_name
if weight.is_a?(Hash)
@weight = weight[:weight]
@tsvector_column = weight[:tsvector_column]
@@ -21,14 +21,16 @@ def initialize(column_name, weight, model)
end
def full_name
+ return @column_name if @column_name.is_a?(Arel::Nodes::SqlLiteral)
+
"#{table_name}.#{column_name}"
end
def to_sql
if tsvector_column
- "coalesce(#{expression}, '')"
+ "coalesce((#{expression})::tsvector, '')"
else
- "coalesce(#{expression}::text, '')"
+ "coalesce((#{expression})::text, '')"
end
end
@@ -39,7 +41,7 @@ def table_name
end
def column_name
- @connection.quote_column_name(@column_name)
+ @connection.quote_column_name(@name)
end
def expression
diff --git a/lib/pg_search/configuration/foreign_column.rb b/lib/pg_search/configuration/foreign_column.rb
index 3886c160..b4956b01 100644
--- a/lib/pg_search/configuration/foreign_column.rb
+++ b/lib/pg_search/configuration/foreign_column.rb
@@ -1,6 +1,6 @@
# frozen_string_literal: true
-require 'digest'
+require "digest"
module PgSearch
class Configuration
diff --git a/lib/pg_search/document.rb b/lib/pg_search/document.rb
index e0f3d3ec..e41eaaa1 100644
--- a/lib/pg_search/document.rb
+++ b/lib/pg_search/document.rb
@@ -1,12 +1,12 @@
# frozen_string_literal: true
-require 'logger'
+require "logger"
module PgSearch
class Document < ActiveRecord::Base
include PgSearch::Model
- self.table_name = 'pg_search_documents'
+ self.table_name = "pg_search_documents"
belongs_to :searchable, polymorphic: true
# The logger might not have loaded yet.
@@ -17,12 +17,12 @@ def self.logger
pg_search_scope :search, lambda { |*args|
options = if PgSearch.multisearch_options.respond_to?(:call)
- PgSearch.multisearch_options.call(*args)
- else
- { query: args.first }.merge(PgSearch.multisearch_options)
- end
+ PgSearch.multisearch_options.call(*args)
+ else
+ {query: args.first}.merge(PgSearch.multisearch_options)
+ end
- { against: :content }.merge(options)
+ {against: :content}.merge(options)
}
end
end
diff --git a/lib/pg_search/features/dmetaphone.rb b/lib/pg_search/features/dmetaphone.rb
index 295f79be..eb6852c6 100644
--- a/lib/pg_search/features/dmetaphone.rb
+++ b/lib/pg_search/features/dmetaphone.rb
@@ -7,7 +7,7 @@ module Features
class DMetaphone
def initialize(query, options, columns, model, normalizer)
dmetaphone_normalizer = Normalizer.new(normalizer)
- options = (options || {}).merge(dictionary: 'simple')
+ options = (options || {}).merge(dictionary: "simple")
@tsearch = TSearch.new(query, options, columns, model, dmetaphone_normalizer)
end
diff --git a/lib/pg_search/features/trigram.rb b/lib/pg_search/features/trigram.rb
index 47f3cc68..80bdfed9 100644
--- a/lib/pg_search/features/trigram.rb
+++ b/lib/pg_search/features/trigram.rb
@@ -35,17 +35,17 @@ def word_similarity?
def similarity_function
if word_similarity?
- 'word_similarity'
+ "word_similarity"
else
- 'similarity'
+ "similarity"
end
end
def infix_operator
if word_similarity?
- '<%'
+ "<%"
else
- '%'
+ "%"
end
end
diff --git a/lib/pg_search/features/tsearch.rb b/lib/pg_search/features/tsearch.rb
index a0e5e89f..82c11885 100644
--- a/lib/pg_search/features/tsearch.rb
+++ b/lib/pg_search/features/tsearch.rb
@@ -1,11 +1,11 @@
# frozen_string_literal: true
require "active_support/core_ext/module/delegation"
-require 'active_support/deprecation'
+require "active_support/deprecation"
module PgSearch
module Features
- class TSearch < Feature # rubocop:disable Metrics/ClassLength
+ class TSearch < Feature
def self.valid_options
super + %i[dictionary prefix negation any_word normalization tsvector_column highlight]
end
@@ -36,7 +36,7 @@ def ts_headline
end
def ts_headline_options
- return '' unless options[:highlight].is_a?(Hash)
+ return "" unless options[:highlight].is_a?(Hash)
headline_options
.merge(deprecated_headline_options)
@@ -58,7 +58,7 @@ def headline_options
end
end
- def deprecated_headline_options # rubocop:disable Metrics/MethodLength
+ def deprecated_headline_options
indifferent_options = options.with_indifferent_access
%w[
@@ -94,11 +94,11 @@ def ts_headline_option_value(value)
end
end
- DISALLOWED_TSQUERY_CHARACTERS = /['?\\:‘’ʻʼ]/.freeze
+ DISALLOWED_TSQUERY_CHARACTERS = /['?\\:‘’ʻʼ]/
def tsquery_for_term(unsanitized_term)
if options[:negation] && unsanitized_term.start_with?("!")
- unsanitized_term[0] = ''
+ unsanitized_term[0] = ""
negated = true
end
@@ -116,7 +116,7 @@ def tsquery_for_term(unsanitized_term)
# If :negated is true, then the term will have ! prepended to the front.
def tsquery_expression(term_sql, negated:, prefix:)
terms = [
- (Arel::Nodes.build_quoted('!') if negated),
+ (Arel::Nodes.build_quoted("!") if negated),
Arel::Nodes.build_quoted("' "),
term_sql,
Arel::Nodes.build_quoted(" '"),
@@ -133,7 +133,7 @@ def tsquery
query_terms = query.split.compact
tsquery_terms = query_terms.map { |term| tsquery_for_term(term) }
- tsquery_terms.join(options[:any_word] ? ' || ' : ' && ')
+ tsquery_terms.join(options[:any_word] ? " || " : " && ")
end
def tsdocument
@@ -151,7 +151,7 @@ def tsdocument
end
end
- tsdocument_terms.join(' || ')
+ tsdocument_terms.join(" || ")
end
# From http://www.postgresql.org/docs/8.3/static/textsearch-controls.html
diff --git a/lib/pg_search/migration/dmetaphone_generator.rb b/lib/pg_search/migration/dmetaphone_generator.rb
index 46e7888c..556a22e6 100644
--- a/lib/pg_search/migration/dmetaphone_generator.rb
+++ b/lib/pg_search/migration/dmetaphone_generator.rb
@@ -1,12 +1,12 @@
# frozen_string_literal: true
-require 'pg_search/migration/generator'
+require "pg_search/migration/generator"
module PgSearch
module Migration
class DmetaphoneGenerator < Generator
def migration_name
- 'add_pg_search_dmetaphone_support_functions'
+ "add_pg_search_dmetaphone_support_functions"
end
end
end
diff --git a/lib/pg_search/migration/generator.rb b/lib/pg_search/migration/generator.rb
index b0a422ae..2cf06a46 100644
--- a/lib/pg_search/migration/generator.rb
+++ b/lib/pg_search/migration/generator.rb
@@ -1,7 +1,7 @@
# frozen_string_literal: true
-require 'active_record'
-require 'rails/generators/base'
+require "active_record"
+require "rails/generators/base"
module PgSearch
module Migration
@@ -10,19 +10,19 @@ class Generator < Rails::Generators::Base
def self.inherited(subclass)
super
- subclass.source_root File.expand_path('templates', __dir__)
+ subclass.source_root File.expand_path("templates", __dir__)
end
def create_migration
now = Time.now.utc
- filename = "#{now.strftime('%Y%m%d%H%M%S')}_#{migration_name}.rb"
+ filename = "#{now.strftime("%Y%m%d%H%M%S")}_#{migration_name}.rb"
template "#{migration_name}.rb.erb", "db/migrate/#{filename}", migration_version
end
private
def read_sql_file(filename)
- sql_directory = File.expand_path('../../../sql', __dir__)
+ sql_directory = File.expand_path("../../../sql", __dir__)
source_path = File.join(sql_directory, "#{filename}.sql")
File.read(source_path).strip
end
diff --git a/lib/pg_search/migration/multisearch_generator.rb b/lib/pg_search/migration/multisearch_generator.rb
index c97fed4b..8f8cd860 100644
--- a/lib/pg_search/migration/multisearch_generator.rb
+++ b/lib/pg_search/migration/multisearch_generator.rb
@@ -1,12 +1,12 @@
# frozen_string_literal: true
-require 'pg_search/migration/generator'
+require "pg_search/migration/generator"
module PgSearch
module Migration
class MultisearchGenerator < Generator
def migration_name
- 'create_pg_search_documents'
+ "create_pg_search_documents"
end
end
end
diff --git a/lib/pg_search/model.rb b/lib/pg_search/model.rb
index eaa5ddf6..ceb7a0f1 100644
--- a/lib/pg_search/model.rb
+++ b/lib/pg_search/model.rb
@@ -7,12 +7,12 @@ module Model
module ClassMethods
def pg_search_scope(name, options)
options_proc = if options.respond_to?(:call)
- options
- elsif options.respond_to?(:merge)
- ->(query) { { query: query }.merge(options) }
- else
- raise ArgumentError, 'pg_search_scope expects a Hash or Proc'
- end
+ options
+ elsif options.respond_to?(:merge)
+ ->(query) { {query: query}.merge(options) }
+ else
+ raise ArgumentError, "pg_search_scope expects a Hash or Proc"
+ end
define_singleton_method(name) do |*args|
config = Configuration.new(options_proc.call(*args), self)
diff --git a/lib/pg_search/multisearchable.rb b/lib/pg_search/multisearchable.rb
index 10a142f6..5309c1c0 100644
--- a/lib/pg_search/multisearchable.rb
+++ b/lib/pg_search/multisearchable.rb
@@ -7,12 +7,12 @@ module Multisearchable
def self.included(mod)
mod.class_eval do
has_one :pg_search_document,
- as: :searchable,
- class_name: "PgSearch::Document",
- dependent: :delete
+ as: :searchable,
+ class_name: "PgSearch::Document",
+ dependent: :delete
after_save :update_pg_search_document,
- if: -> { PgSearch.multisearch_enabled? }
+ if: -> { PgSearch.multisearch_enabled? }
end
end
@@ -39,7 +39,7 @@ def should_update_pg_search_document?
conditions.all? { |condition| condition.to_proc.call(self) }
end
- def update_pg_search_document # rubocop:disable Metrics/AbcSize
+ def update_pg_search_document
if_conditions = Array(pg_search_multisearchable_options[:if])
unless_conditions = Array(pg_search_multisearchable_options[:unless])
diff --git a/lib/pg_search/normalizer.rb b/lib/pg_search/normalizer.rb
index aed3f721..370b6a83 100644
--- a/lib/pg_search/normalizer.rb
+++ b/lib/pg_search/normalizer.rb
@@ -10,11 +10,11 @@ def add_normalization(sql_expression)
return sql_expression unless config.ignore.include?(:accents)
sql_node = case sql_expression
- when Arel::Nodes::Node
- sql_expression
- else
- Arel.sql(sql_expression)
- end
+ when Arel::Nodes::Node
+ sql_expression
+ else
+ Arel.sql(sql_expression)
+ end
Arel::Nodes::NamedFunction.new(
PgSearch.unaccent_function,
diff --git a/lib/pg_search/scope_options.rb b/lib/pg_search/scope_options.rb
index 369a603a..b03d13d0 100644
--- a/lib/pg_search/scope_options.rb
+++ b/lib/pg_search/scope_options.rb
@@ -91,9 +91,9 @@ def subquery
def conditions
config.features
- .reject { |_feature_name, feature_options| feature_options && feature_options[:sort_only] }
- .map { |feature_name, _feature_options| feature_for(feature_name).conditions }
- .inject { |accumulator, expression| Arel::Nodes::Or.new(accumulator, expression) }
+ .reject { |_feature_name, feature_options| feature_options && feature_options[:sort_only] }
+ .map { |feature_name, _feature_options| feature_for(feature_name).conditions }
+ .inject { |accumulator, expression| Arel::Nodes::Or.new(accumulator, expression) }
end
def order_within_rank
@@ -108,7 +108,7 @@ def subquery_join
if config.associations.any?
config.associations.map do |association|
association.join(primary_key)
- end.join(' ')
+ end.join(" ")
end
end
diff --git a/lib/pg_search/tasks.rb b/lib/pg_search/tasks.rb
index 04fec234..3b5740f4 100644
--- a/lib/pg_search/tasks.rb
+++ b/lib/pg_search/tasks.rb
@@ -1,7 +1,7 @@
# frozen_string_literal: true
-require 'rake'
-require 'pg_search'
+require "rake"
+require "pg_search"
namespace :pg_search do
namespace :multisearch do
diff --git a/lib/pg_search/version.rb b/lib/pg_search/version.rb
index 2a007da3..afff11ae 100644
--- a/lib/pg_search/version.rb
+++ b/lib/pg_search/version.rb
@@ -1,5 +1,5 @@
# frozen_string_literal: true
module PgSearch
- VERSION = '2.3.6'
+ VERSION = "2.3.6"
end
diff --git a/pg_search.gemspec b/pg_search.gemspec
index ada1908b..103a9a38 100644
--- a/pg_search.gemspec
+++ b/pg_search.gemspec
@@ -1,39 +1,25 @@
# frozen_string_literal: true
-$LOAD_PATH.push File.expand_path('lib', __dir__)
-require 'pg_search/version'
+$LOAD_PATH.push File.expand_path("lib", __dir__)
+require "pg_search/version"
Gem::Specification.new do |s|
- s.name = 'pg_search'
- s.version = PgSearch::VERSION
- s.platform = Gem::Platform::RUBY
- s.authors = ['Grant Hutchins', 'Case Commons, LLC']
- s.email = %w[gems@nertzy.com casecommons-dev@googlegroups.com]
- s.homepage = 'https://github.com/Casecommons/pg_search'
- s.summary = "PgSearch builds Active Record named scopes that take advantage of PostgreSQL's full text search"
+ s.name = "pg_search"
+ s.version = PgSearch::VERSION
+ s.platform = Gem::Platform::RUBY
+ s.authors = ["Grant Hutchins", "Case Commons, LLC"]
+ s.email = %w[gems@nertzy.com casecommons-dev@googlegroups.com]
+ s.homepage = "https://github.com/Casecommons/pg_search"
+ s.summary = "PgSearch builds Active Record named scopes that take advantage of PostgreSQL's full text search"
s.description = "PgSearch builds Active Record named scopes that take advantage of PostgreSQL's full text search"
- s.licenses = ['MIT']
+ s.licenses = ["MIT"]
s.metadata["rubygems_mfa_required"] = "true"
- s.files = `git ls-files -z`.split("\x0")
- s.require_paths = ['lib']
+ s.files = `git ls-files -z`.split("\x0")
+ s.require_paths = ["lib"]
- s.add_dependency 'activerecord', '>= 5.2'
- s.add_dependency 'activesupport', '>= 5.2'
+ s.add_dependency "activerecord", ">= 6.0"
+ s.add_dependency "activesupport", ">= 6.0"
- s.add_development_dependency 'pry'
- s.add_development_dependency 'rake'
- s.add_development_dependency 'rspec'
- s.add_development_dependency 'rubocop'
- s.add_development_dependency 'rubocop-performance'
- s.add_development_dependency 'rubocop-rails'
- s.add_development_dependency 'rubocop-rake'
- s.add_development_dependency 'rubocop-rspec'
- s.add_development_dependency 'simplecov'
- s.add_development_dependency 'simplecov-lcov'
- s.add_development_dependency 'undercover'
- s.add_development_dependency 'warning'
- s.add_development_dependency 'with_model'
-
- s.required_ruby_version = '>= 2.7'
+ s.required_ruby_version = ">= 2.7"
end
diff --git a/spec/.rubocop.yml b/spec/.rubocop.yml
index 6e3e3040..ee624fbe 100644
--- a/spec/.rubocop.yml
+++ b/spec/.rubocop.yml
@@ -1,14 +1,27 @@
inherit_from:
- ../.rubocop.yml
-Layout/LineLength:
- Enabled: false
+RSpec/ContextWording:
+ Prefixes:
+ - using
+ - via
+ - when
+ - with
+ - without
-Lint/SuppressedException:
- Enabled: false
+RSpec/DescribedClass:
+ Enabled: true
-Lint/UselessAssignment:
- Enabled: false
+RSpec/ExampleLength:
+ Max: 15
-Style/BlockDelimiters:
+RSpec/ExpectInHook:
Enabled: false
+
+RSpec/FilePath:
+ CustomTransform:
+ TSearch: "tsearch"
+ DMetaphone: "dmetaphone"
+
+RSpec/MultipleExpectations:
+ Max: 5
diff --git a/spec/integration/.rubocop.yml b/spec/integration/.rubocop.yml
index 09ebbc02..b76a90e7 100644
--- a/spec/integration/.rubocop.yml
+++ b/spec/integration/.rubocop.yml
@@ -4,8 +4,8 @@ inherit_from:
RSpec/DescribeClass:
Enabled: false
-RSpec/MultipleExpectations:
+RSpec/ExampleLength:
Enabled: false
-RSpec/ExampleLength:
+RSpec/MultipleExpectations:
Enabled: false
diff --git a/spec/integration/associations_spec.rb b/spec/integration/associations_spec.rb
index a0801ed6..e929c2af 100644
--- a/spec/integration/associations_spec.rb
+++ b/spec/integration/associations_spec.rb
@@ -20,23 +20,23 @@
model do
include PgSearch::Model
- belongs_to :another_model, class_name: 'AssociatedModel'
+ belongs_to :another_model, class_name: "AssociatedModel"
- pg_search_scope :with_another, associated_against: { another_model: :title }
+ pg_search_scope :with_another, associated_against: {another_model: :title}
end
end
it "returns rows that match the query in the columns of the associated model only" do
- associated = AssociatedModel.create!(title: 'abcdef')
+ associated = AssociatedModel.create!(title: "abcdef")
included = [
- ModelWithoutAgainst.create!(title: 'abcdef', another_model: associated),
- ModelWithoutAgainst.create!(title: 'ghijkl', another_model: associated)
+ ModelWithoutAgainst.create!(title: "abcdef", another_model: associated),
+ ModelWithoutAgainst.create!(title: "ghijkl", another_model: associated)
]
excluded = [
- ModelWithoutAgainst.create!(title: 'abcdef')
+ ModelWithoutAgainst.create!(title: "abcdef")
]
- results = ModelWithoutAgainst.with_another('abcdef')
+ results = ModelWithoutAgainst.with_another("abcdef")
expect(results.map(&:title)).to match_array(included.map(&:title))
expect(results).not_to include(excluded)
end
@@ -45,34 +45,34 @@
context "via a belongs_to association" do
with_model :AssociatedModel do
table do |t|
- t.string 'title'
+ t.string "title"
end
end
with_model :ModelWithBelongsTo do
table do |t|
- t.string 'title'
- t.belongs_to 'another_model', index: false
+ t.string "title"
+ t.belongs_to "another_model", index: false
end
model do
include PgSearch::Model
- belongs_to :another_model, class_name: 'AssociatedModel'
+ belongs_to :another_model, class_name: "AssociatedModel"
- pg_search_scope :with_associated, against: :title, associated_against: { another_model: :title }
+ pg_search_scope :with_associated, against: :title, associated_against: {another_model: :title}
end
end
it "returns rows that match the query in either its own columns or the columns of the associated model" do
- associated = AssociatedModel.create!(title: 'abcdef')
+ associated = AssociatedModel.create!(title: "abcdef")
included = [
- ModelWithBelongsTo.create!(title: 'ghijkl', another_model: associated),
- ModelWithBelongsTo.create!(title: 'abcdef')
+ ModelWithBelongsTo.create!(title: "ghijkl", another_model: associated),
+ ModelWithBelongsTo.create!(title: "abcdef")
]
- excluded = ModelWithBelongsTo.create!(title: 'mnopqr',
- another_model: AssociatedModel.create!(title: 'stuvwx'))
+ excluded = ModelWithBelongsTo.create!(title: "mnopqr",
+ another_model: AssociatedModel.create!(title: "stuvwx"))
- results = ModelWithBelongsTo.with_associated('abcdef')
+ results = ModelWithBelongsTo.with_associated("abcdef")
expect(results.map(&:title)).to match_array(included.map(&:title))
expect(results).not_to include(excluded)
end
@@ -81,61 +81,61 @@
context "via a has_many association" do
with_model :AssociatedModelWithHasMany do
table do |t|
- t.string 'title'
- t.belongs_to 'ModelWithHasMany', index: false
+ t.string "title"
+ t.belongs_to "ModelWithHasMany", index: false
end
end
with_model :ModelWithHasMany do
table do |t|
- t.string 'title'
+ t.string "title"
end
model do
include PgSearch::Model
- has_many :other_models, class_name: 'AssociatedModelWithHasMany', foreign_key: 'ModelWithHasMany_id'
+ has_many :other_models, class_name: "AssociatedModelWithHasMany", foreign_key: "ModelWithHasMany_id"
- pg_search_scope :with_associated, against: [:title], associated_against: { other_models: :title }
+ pg_search_scope :with_associated, against: [:title], associated_against: {other_models: :title}
end
end
it "returns rows that match the query in either its own columns or the columns of the associated model" do
included = [
- ModelWithHasMany.create!(title: 'abcdef', other_models: [
- AssociatedModelWithHasMany.create!(title: 'foo'),
- AssociatedModelWithHasMany.create!(title: 'bar')
+ ModelWithHasMany.create!(title: "abcdef", other_models: [
+ AssociatedModelWithHasMany.create!(title: "foo"),
+ AssociatedModelWithHasMany.create!(title: "bar")
]),
- ModelWithHasMany.create!(title: 'ghijkl', other_models: [
- AssociatedModelWithHasMany.create!(title: 'foo bar'),
- AssociatedModelWithHasMany.create!(title: 'mnopqr')
+ ModelWithHasMany.create!(title: "ghijkl", other_models: [
+ AssociatedModelWithHasMany.create!(title: "foo bar"),
+ AssociatedModelWithHasMany.create!(title: "mnopqr")
]),
- ModelWithHasMany.create!(title: 'foo bar')
+ ModelWithHasMany.create!(title: "foo bar")
]
- excluded = ModelWithHasMany.create!(title: 'stuvwx', other_models: [
- AssociatedModelWithHasMany.create!(title: 'abcdef')
+ excluded = ModelWithHasMany.create!(title: "stuvwx", other_models: [
+ AssociatedModelWithHasMany.create!(title: "abcdef")
])
- results = ModelWithHasMany.with_associated('foo bar')
+ results = ModelWithHasMany.with_associated("foo bar")
expect(results.map(&:title)).to match_array(included.map(&:title))
expect(results).not_to include(excluded)
end
it "uses an unscoped relation of the associated model" do
- excluded = ModelWithHasMany.create!(title: 'abcdef', other_models: [
- AssociatedModelWithHasMany.create!(title: 'abcdef')
+ excluded = ModelWithHasMany.create!(title: "abcdef", other_models: [
+ AssociatedModelWithHasMany.create!(title: "abcdef")
])
included = [
- ModelWithHasMany.create!(title: 'abcdef', other_models: [
- AssociatedModelWithHasMany.create!(title: 'foo'),
- AssociatedModelWithHasMany.create!(title: 'bar')
+ ModelWithHasMany.create!(title: "abcdef", other_models: [
+ AssociatedModelWithHasMany.create!(title: "foo"),
+ AssociatedModelWithHasMany.create!(title: "bar")
])
]
results = ModelWithHasMany
- .limit(1)
- .order(Arel.sql("#{ModelWithHasMany.quoted_table_name}.id ASC"))
- .with_associated('foo bar')
+ .limit(1)
+ .order(Arel.sql("#{ModelWithHasMany.quoted_table_name}.id ASC"))
+ .with_associated("foo bar")
expect(results.map(&:title)).to match_array(included.map(&:title))
expect(results).not_to include(excluded)
@@ -146,36 +146,36 @@
context "when on different tables" do
with_model :FirstAssociatedModel do
table do |t|
- t.string 'title'
- t.belongs_to 'ModelWithManyAssociations', index: false
+ t.string "title"
+ t.belongs_to "ModelWithManyAssociations", index: false
end
end
with_model :SecondAssociatedModel do
table do |t|
- t.string 'title'
+ t.string "title"
end
end
with_model :ModelWithManyAssociations do
table do |t|
- t.string 'title'
- t.belongs_to 'model_of_second_type', index: false
+ t.string "title"
+ t.belongs_to "model_of_second_type", index: false
end
model do
include PgSearch::Model
has_many :models_of_first_type,
- class_name: 'FirstAssociatedModel',
- foreign_key: 'ModelWithManyAssociations_id'
+ class_name: "FirstAssociatedModel",
+ foreign_key: "ModelWithManyAssociations_id"
belongs_to :model_of_second_type,
- class_name: 'SecondAssociatedModel'
+ class_name: "SecondAssociatedModel"
pg_search_scope :with_associated,
- against: :title,
- associated_against: { models_of_first_type: :title, model_of_second_type: :title }
+ against: :title,
+ associated_against: {models_of_first_type: :title, model_of_second_type: :title}
end
end
@@ -184,25 +184,25 @@
unmatching_second = SecondAssociatedModel.create!(title: "uiop")
included = [
- ModelWithManyAssociations.create!(title: 'abcdef', models_of_first_type: [
- FirstAssociatedModel.create!(title: 'foo'),
- FirstAssociatedModel.create!(title: 'bar')
+ ModelWithManyAssociations.create!(title: "abcdef", models_of_first_type: [
+ FirstAssociatedModel.create!(title: "foo"),
+ FirstAssociatedModel.create!(title: "bar")
]),
- ModelWithManyAssociations.create!(title: 'ghijkl', models_of_first_type: [
- FirstAssociatedModel.create!(title: 'foo bar'),
- FirstAssociatedModel.create!(title: 'mnopqr')
+ ModelWithManyAssociations.create!(title: "ghijkl", models_of_first_type: [
+ FirstAssociatedModel.create!(title: "foo bar"),
+ FirstAssociatedModel.create!(title: "mnopqr")
]),
- ModelWithManyAssociations.create!(title: 'foo bar'),
- ModelWithManyAssociations.create!(title: 'qwerty', model_of_second_type: matching_second)
+ ModelWithManyAssociations.create!(title: "foo bar"),
+ ModelWithManyAssociations.create!(title: "qwerty", model_of_second_type: matching_second)
]
excluded = [
- ModelWithManyAssociations.create!(title: 'stuvwx', models_of_first_type: [
- FirstAssociatedModel.create!(title: 'abcdef')
+ ModelWithManyAssociations.create!(title: "stuvwx", models_of_first_type: [
+ FirstAssociatedModel.create!(title: "abcdef")
]),
- ModelWithManyAssociations.create!(title: 'qwerty', model_of_second_type: unmatching_second)
+ ModelWithManyAssociations.create!(title: "qwerty", model_of_second_type: unmatching_second)
]
- results = ModelWithManyAssociations.with_associated('foo bar')
+ results = ModelWithManyAssociations.with_associated("foo bar")
expect(results.map(&:title)).to match_array(included.map(&:title))
excluded.each { |object| expect(results).not_to include(object) }
end
@@ -211,58 +211,58 @@
context "when on the same table" do
with_model :DoublyAssociatedModel do
table do |t|
- t.string 'title'
- t.belongs_to 'ModelWithDoubleAssociation', index: false
- t.belongs_to 'ModelWithDoubleAssociation_again', index: false
+ t.string "title"
+ t.belongs_to "ModelWithDoubleAssociation", index: false
+ t.belongs_to "ModelWithDoubleAssociation_again", index: false
end
end
with_model :ModelWithDoubleAssociation do
table do |t|
- t.string 'title'
+ t.string "title"
end
model do
include PgSearch::Model
has_many :things,
- class_name: 'DoublyAssociatedModel',
- foreign_key: 'ModelWithDoubleAssociation_id'
+ class_name: "DoublyAssociatedModel",
+ foreign_key: "ModelWithDoubleAssociation_id"
has_many :thingamabobs,
- class_name: 'DoublyAssociatedModel',
- foreign_key: 'ModelWithDoubleAssociation_again_id'
+ class_name: "DoublyAssociatedModel",
+ foreign_key: "ModelWithDoubleAssociation_again_id"
pg_search_scope :with_associated, against: :title,
- associated_against: { things: :title, thingamabobs: :title }
+ associated_against: {things: :title, thingamabobs: :title}
end
end
it "returns rows that match the query in either its own columns or the columns of the associated model" do
included = [
- ModelWithDoubleAssociation.create!(title: 'abcdef', things: [
- DoublyAssociatedModel.create!(title: 'foo'),
- DoublyAssociatedModel.create!(title: 'bar')
+ ModelWithDoubleAssociation.create!(title: "abcdef", things: [
+ DoublyAssociatedModel.create!(title: "foo"),
+ DoublyAssociatedModel.create!(title: "bar")
]),
- ModelWithDoubleAssociation.create!(title: 'ghijkl', things: [
- DoublyAssociatedModel.create!(title: 'foo bar'),
- DoublyAssociatedModel.create!(title: 'mnopqr')
+ ModelWithDoubleAssociation.create!(title: "ghijkl", things: [
+ DoublyAssociatedModel.create!(title: "foo bar"),
+ DoublyAssociatedModel.create!(title: "mnopqr")
]),
- ModelWithDoubleAssociation.create!(title: 'foo bar'),
- ModelWithDoubleAssociation.create!(title: 'qwerty', thingamabobs: [
+ ModelWithDoubleAssociation.create!(title: "foo bar"),
+ ModelWithDoubleAssociation.create!(title: "qwerty", thingamabobs: [
DoublyAssociatedModel.create!(title: "foo bar")
])
]
excluded = [
- ModelWithDoubleAssociation.create!(title: 'stuvwx', things: [
- DoublyAssociatedModel.create!(title: 'abcdef')
+ ModelWithDoubleAssociation.create!(title: "stuvwx", things: [
+ DoublyAssociatedModel.create!(title: "abcdef")
]),
- ModelWithDoubleAssociation.create!(title: 'qwerty', thingamabobs: [
+ ModelWithDoubleAssociation.create!(title: "qwerty", thingamabobs: [
DoublyAssociatedModel.create!(title: "uiop")
])
]
- results = ModelWithDoubleAssociation.with_associated('foo bar')
+ results = ModelWithDoubleAssociation.with_associated("foo bar")
expect(results.map(&:title)).to match_array(included.map(&:title))
excluded.each { |object| expect(results).not_to include(object) }
end
@@ -272,21 +272,21 @@
context "when against multiple attributes on one association" do
with_model :AssociatedModel do
table do |t|
- t.string 'title'
- t.text 'author'
+ t.string "title"
+ t.text "author"
end
end
with_model :ModelWithAssociation do
table do |t|
- t.belongs_to 'another_model', index: false
+ t.belongs_to "another_model", index: false
end
model do
include PgSearch::Model
- belongs_to :another_model, class_name: 'AssociatedModel'
+ belongs_to :another_model, class_name: "AssociatedModel"
- pg_search_scope :with_associated, associated_against: { another_model: %i[title author] }
+ pg_search_scope :with_associated, associated_against: {another_model: %i[title author]}
end
end
@@ -314,7 +314,7 @@
)
]
- results = ModelWithAssociation.with_associated('foo bar')
+ results = ModelWithAssociation.with_associated("foo bar")
expect(results.to_sql.scan("INNER JOIN #{AssociatedModel.quoted_table_name}").length).to eq(1)
included.each { |object| expect(results).to include(object) }
@@ -325,21 +325,21 @@
context "when against non-text columns" do
with_model :AssociatedModel do
table do |t|
- t.integer 'number'
+ t.integer "number"
end
end
with_model :Model do
table do |t|
- t.integer 'number'
- t.belongs_to 'another_model', index: false
+ t.integer "number"
+ t.belongs_to "another_model", index: false
end
model do
include PgSearch::Model
- belongs_to :another_model, class_name: 'AssociatedModel'
+ belongs_to :another_model, class_name: "AssociatedModel"
- pg_search_scope :with_associated, associated_against: { another_model: :number }
+ pg_search_scope :with_associated, associated_against: {another_model: :number}
end
end
@@ -353,7 +353,7 @@
Model.create!(number: 123)
]
- results = Model.with_associated('123')
+ results = Model.with_associated("123")
expect(results.map(&:number)).to match_array(included.map(&:number))
expect(results).not_to include(excluded)
end
@@ -384,10 +384,10 @@
# https://github.com/Casecommons/pg_search/issues/14
it "supports queries with periods" do
- included = Parent.create!(name: 'bar.foo')
- excluded = Parent.create!(name: 'foo.bar')
+ included = Parent.create!(name: "bar.foo")
+ excluded = Parent.create!(name: "foo.bar")
- results = Parent.search_name('bar.foo').includes(:children)
+ results = Parent.search_name("bar.foo").includes(:children)
results.to_a
expect(results).to include(included)
@@ -477,7 +477,7 @@
Position.create!(company_id: company.id, title: "penn 1")
]
- results = company.positions.search('teller 1')
+ results = company.positions.search("teller 1")
expect(results).to include(*included)
expect(results).not_to include(*excluded)
diff --git a/spec/integration/pg_search_spec.rb b/spec/integration/pg_search_spec.rb
index 64a86d7e..00f2ed96 100644
--- a/spec/integration/pg_search_spec.rb
+++ b/spec/integration/pg_search_spec.rb
@@ -6,10 +6,10 @@
describe "an Active Record model which includes PgSearch" do
with_model :ModelWithPgSearch do
table do |t|
- t.string 'title'
- t.text 'content'
- t.integer 'parent_model_id'
- t.integer 'importance'
+ t.string "title"
+ t.text "content"
+ t.integer "parent_model_id"
+ t.integer "importance"
end
model do
@@ -39,26 +39,38 @@
context "when passed a lambda" do
it "builds a dynamic scope" do
ModelWithPgSearch.pg_search_scope :search_title_or_content,
- lambda { |query, pick_content|
- {
- query: query.gsub("-remove-", ""),
- against: pick_content ? :content : :title
- }
- }
+ lambda { |query, pick_content|
+ {
+ query: query.gsub("-remove-", ""),
+ against: pick_content ? :content : :title
+ }
+ }
- included = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
- excluded = ModelWithPgSearch.create!(title: 'bar', content: 'foo')
+ included = ModelWithPgSearch.create!(title: "foo", content: "bar")
+ ModelWithPgSearch.create!(title: "bar", content: "foo")
- expect(ModelWithPgSearch.search_title_or_content('fo-remove-o', false)).to eq([included])
- expect(ModelWithPgSearch.search_title_or_content('b-remove-ar', true)).to eq([included])
+ expect(ModelWithPgSearch.search_title_or_content("fo-remove-o", false)).to eq([included])
+ expect(ModelWithPgSearch.search_title_or_content("b-remove-ar", true)).to eq([included])
+ end
+ end
+
+ context "when passed an invalid argument" do
+ it "builds a dynamic scope" do
+ expect {
+ ModelWithPgSearch.pg_search_scope :search_title_or_content, :some_symbol
+ }.to(
+ raise_exception(ArgumentError).with_message(
+ "pg_search_scope expects a Hash or Proc"
+ )
+ )
end
end
context "when an unknown option is passed in" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_option,
- against: :content,
- foo: :bar
+ against: :content,
+ foo: :bar
expect {
ModelWithPgSearch.with_unknown_option("foo")
@@ -68,7 +80,7 @@
context "with a lambda" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_option,
- ->(*) { { against: :content, foo: :bar } }
+ ->(*) { {against: :content, foo: :bar} }
expect {
ModelWithPgSearch.with_unknown_option("foo")
@@ -80,8 +92,8 @@
context "when an unknown :using is passed" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_using,
- against: :content,
- using: :foo
+ against: :content,
+ using: :foo
expect {
ModelWithPgSearch.with_unknown_using("foo")
@@ -91,7 +103,7 @@
context "with a lambda" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_using,
- ->(*) { { against: :content, using: :foo } }
+ ->(*) { {against: :content, using: :foo} }
expect {
ModelWithPgSearch.with_unknown_using("foo")
@@ -103,8 +115,8 @@
context "when an unknown :ignoring is passed" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_ignoring,
- against: :content,
- ignoring: :foo
+ against: :content,
+ ignoring: :foo
expect {
ModelWithPgSearch.with_unknown_ignoring("foo")
@@ -114,7 +126,7 @@
context "with a lambda" do
it "raises an exception when invoked" do
ModelWithPgSearch.pg_search_scope :with_unknown_ignoring,
- ->(*) { { against: :content, ignoring: :foo } }
+ ->(*) { {against: :content, ignoring: :foo} }
expect {
ModelWithPgSearch.with_unknown_ignoring("foo")
@@ -168,15 +180,15 @@
context "when chained after a select() scope" do
it "honors the select" do
- included = ModelWithPgSearch.create!(content: 'foo', title: 'bar')
- excluded = ModelWithPgSearch.create!(content: 'bar', title: 'foo')
+ included = ModelWithPgSearch.create!(content: "foo", title: "bar")
+ excluded = ModelWithPgSearch.create!(content: "bar", title: "foo")
- results = ModelWithPgSearch.select('id, title').search_content('foo')
+ results = ModelWithPgSearch.select("id, title").search_content("foo")
expect(results).to include(included)
expect(results).not_to include(excluded)
- expect(results.first.attributes.key?('content')).to be false
+ expect(results.first.attributes.key?("content")).to be false
expect(results.select { |record| record.title == "bar" }).to eq [included]
expect(results.reject { |record| record.title == "bar" }).to be_empty
@@ -185,15 +197,15 @@
context "when chained before a select() scope" do
it "honors the select" do
- included = ModelWithPgSearch.create!(content: 'foo', title: 'bar')
- excluded = ModelWithPgSearch.create!(content: 'bar', title: 'foo')
+ included = ModelWithPgSearch.create!(content: "foo", title: "bar")
+ excluded = ModelWithPgSearch.create!(content: "bar", title: "foo")
- results = ModelWithPgSearch.search_content('foo').select('id, title')
+ results = ModelWithPgSearch.search_content("foo").select("id, title")
expect(results).to include(included)
expect(results).not_to include(excluded)
- expect(results.first.attributes.key?('content')).to be false
+ expect(results.first.attributes.key?("content")).to be false
expect(results.select { |record| record.title == "bar" }).to eq [included]
expect(results.reject { |record| record.title == "bar" }).to be_empty
@@ -202,15 +214,15 @@
context "when surrouned by select() scopes" do
it "honors the select" do
- included = ModelWithPgSearch.create!(content: 'foo', title: 'bar')
- excluded = ModelWithPgSearch.create!(content: 'bar', title: 'foo')
+ included = ModelWithPgSearch.create!(content: "foo", title: "bar")
+ excluded = ModelWithPgSearch.create!(content: "bar", title: "foo")
- results = ModelWithPgSearch.select('id').search_content('foo').select('title')
+ results = ModelWithPgSearch.select("id").search_content("foo").select("title")
expect(results).to include(included)
expect(results).not_to include(excluded)
- expect(results.first.attributes.key?('content')).to be false
+ expect(results.first.attributes.key?("content")).to be false
expect(results.select { |record| record.title == "bar" }).to eq [included]
expect(results.reject { |record| record.title == "bar" }).to be_empty
@@ -241,7 +253,7 @@
has_many :houses
pg_search_scope :named, against: [:name]
scope :with_house_in_city, lambda { |city|
- joins(:houses).where(House.table_name.to_sym => { city: city })
+ joins(:houses).where(House.table_name.to_sym => {city: city})
}
scope :house_search_city, lambda { |query|
joins(:houses).merge(House.search_city(query))
@@ -285,7 +297,7 @@
context "when chaining merged scopes" do
it "does not raise an exception" do
- relation = Person.named('foo').house_search_city('bar')
+ relation = Person.named("foo").house_search_city("bar")
expect { relation.to_a }.not_to raise_error
end
@@ -298,49 +310,49 @@
end
it "does not raise an exception" do
- relation = ModelWithPgSearch.search_content('foo').search_title('bar')
+ relation = ModelWithPgSearch.search_content("foo").search_title("bar")
expect { relation.to_a }.not_to raise_error
end
end
it "returns an empty array when a blank query is passed in" do
- ModelWithPgSearch.create!(content: 'foo')
+ ModelWithPgSearch.create!(content: "foo")
- results = ModelWithPgSearch.search_content('')
+ results = ModelWithPgSearch.search_content("")
expect(results).to eq([])
end
it "returns rows where the column contains the term in the query" do
- included = ModelWithPgSearch.create!(content: 'foo')
- excluded = ModelWithPgSearch.create!(content: 'bar')
+ included = ModelWithPgSearch.create!(content: "foo")
+ excluded = ModelWithPgSearch.create!(content: "bar")
- results = ModelWithPgSearch.search_content('foo')
+ results = ModelWithPgSearch.search_content("foo")
expect(results).to include(included)
expect(results).not_to include(excluded)
end
it "returns the correct count" do
- ModelWithPgSearch.create!(content: 'foo')
- ModelWithPgSearch.create!(content: 'bar')
+ ModelWithPgSearch.create!(content: "foo")
+ ModelWithPgSearch.create!(content: "bar")
- results = ModelWithPgSearch.search_content('foo')
+ results = ModelWithPgSearch.search_content("foo")
expect(results.count).to eq 1
end
it "returns the correct count(:all)" do
- ModelWithPgSearch.create!(content: 'foo')
- ModelWithPgSearch.create!(content: 'bar')
+ ModelWithPgSearch.create!(content: "foo")
+ ModelWithPgSearch.create!(content: "bar")
- results = ModelWithPgSearch.search_content('foo')
+ results = ModelWithPgSearch.search_content("foo")
expect(results.count(:all)).to eq 1
end
it "supports #select" do
- record = ModelWithPgSearch.create!(content: 'foo')
- other_record = ModelWithPgSearch.create!(content: 'bar')
+ record = ModelWithPgSearch.create!(content: "foo")
+ ModelWithPgSearch.create!(content: "bar")
- records_with_only_id = ModelWithPgSearch.search_content('foo').select('id')
+ records_with_only_id = ModelWithPgSearch.search_content("foo").select("id")
expect(records_with_only_id.length).to eq 1
returned_record = records_with_only_id.first
@@ -349,36 +361,36 @@
end
it "supports #pluck" do
- record = ModelWithPgSearch.create!(content: 'foo')
- other_record = ModelWithPgSearch.create!(content: 'bar')
+ record = ModelWithPgSearch.create!(content: "foo")
+ ModelWithPgSearch.create!(content: "bar")
- ids = ModelWithPgSearch.search_content('foo').pluck('id')
+ ids = ModelWithPgSearch.search_content("foo").pluck("id")
expect(ids).to eq [record.id]
end
it "supports adding where clauses using the pg_search.rank" do
- once = ModelWithPgSearch.create!(content: 'foo bar')
- twice = ModelWithPgSearch.create!(content: 'foo foo')
+ ModelWithPgSearch.create!(content: "foo bar")
+ twice = ModelWithPgSearch.create!(content: "foo foo")
- records = ModelWithPgSearch.search_content('foo')
- .where("#{PgSearch::Configuration.alias(ModelWithPgSearch.table_name)}.rank > 0.07")
+ records = ModelWithPgSearch.search_content("foo")
+ .where("#{PgSearch::Configuration.alias(ModelWithPgSearch.table_name)}.rank > 0.07")
expect(records).to eq [twice]
end
it "returns rows where the column contains all the terms in the query in any order" do
- included = [ModelWithPgSearch.create!(content: 'foo bar'),
- ModelWithPgSearch.create!(content: 'bar foo')]
- excluded = ModelWithPgSearch.create!(content: 'foo')
+ included = [ModelWithPgSearch.create!(content: "foo bar"),
+ ModelWithPgSearch.create!(content: "bar foo")]
+ excluded = ModelWithPgSearch.create!(content: "foo")
- results = ModelWithPgSearch.search_content('foo bar')
+ results = ModelWithPgSearch.search_content("foo bar")
expect(results).to match_array(included)
expect(results).not_to include(excluded)
end
it "returns rows that match the query but not its case" do
included = [ModelWithPgSearch.create!(content: "foo"),
- ModelWithPgSearch.create!(content: "FOO")]
+ ModelWithPgSearch.create!(content: "FOO")]
results = ModelWithPgSearch.search_content("Foo")
expect(results).to match_array(included)
@@ -397,8 +409,8 @@
end
it "returns rows that match the query but not rows that are prefixed by the query" do
- included = ModelWithPgSearch.create!(content: 'pre')
- excluded = ModelWithPgSearch.create!(content: 'prefix')
+ included = ModelWithPgSearch.create!(content: "pre")
+ excluded = ModelWithPgSearch.create!(content: "prefix")
results = ModelWithPgSearch.search_content("pre")
expect(results).to eq([included])
@@ -407,36 +419,36 @@
it "returns rows that match the query exactly and not those that match the query when stemmed by the default english dictionary" do
included = ModelWithPgSearch.create!(content: "jumped")
- excluded = [ModelWithPgSearch.create!(content: "jump"),
- ModelWithPgSearch.create!(content: "jumping")]
+ ModelWithPgSearch.create!(content: "jump")
+ ModelWithPgSearch.create!(content: "jumping")
results = ModelWithPgSearch.search_content("jumped")
expect(results).to eq([included])
end
it "returns rows that match sorted by rank" do
- loser = ModelWithPgSearch.create!(content: 'foo')
- winner = ModelWithPgSearch.create!(content: 'foo foo')
+ loser = ModelWithPgSearch.create!(content: "foo")
+ winner = ModelWithPgSearch.create!(content: "foo foo")
results = ModelWithPgSearch.search_content("foo").with_pg_search_rank
expect(results[0].pg_search_rank).to be > results[1].pg_search_rank
expect(results).to eq([winner, loser])
end
- it 'preserves column selection when with_pg_search_rank is chained after a select()' do
- loser = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
+ it "preserves column selection when with_pg_search_rank is chained after a select()" do
+ ModelWithPgSearch.create!(title: "foo", content: "bar")
- results = ModelWithPgSearch.search_content('bar').select(:content).with_pg_search_rank
+ results = ModelWithPgSearch.search_content("bar").select(:content).with_pg_search_rank
expect(results.length).to be 1
- expect(results.first.as_json.keys).to contain_exactly('id', 'content', 'pg_search_rank')
+ expect(results.first.as_json.keys).to contain_exactly("id", "content", "pg_search_rank")
end
- it 'allows pg_search_rank along with a join' do
+ it "allows pg_search_rank along with a join" do
parent_1 = ParentModel.create!(id: 98)
parent_2 = ParentModel.create!(id: 99)
- loser = ModelWithPgSearch.create!(content: 'foo', parent_model: parent_2)
- winner = ModelWithPgSearch.create!(content: 'foo foo', parent_model: parent_1)
+ loser = ModelWithPgSearch.create!(content: "foo", parent_model: parent_2)
+ winner = ModelWithPgSearch.create!(content: "foo foo", parent_model: parent_1)
results = ModelWithPgSearch.joins(:parent_model).merge(ParentModel.active).search_content("foo").with_pg_search_rank
expect(results.map(&:id)).to eq [winner.id, loser.id]
@@ -445,8 +457,8 @@
end
it "returns results that match sorted by primary key for records that rank the same" do
- sorted_results = [ModelWithPgSearch.create!(content: 'foo'),
- ModelWithPgSearch.create!(content: 'foo')].sort_by(&:id)
+ sorted_results = [ModelWithPgSearch.create!(content: "foo"),
+ ModelWithPgSearch.create!(content: "foo")].sort_by(&:id)
results = ModelWithPgSearch.search_content("foo")
expect(results).to eq(sorted_results)
@@ -454,16 +466,16 @@
it "returns results that match a query with multiple space-separated search terms" do
included = [
- ModelWithPgSearch.create!(content: 'foo bar'),
- ModelWithPgSearch.create!(content: 'bar foo'),
- ModelWithPgSearch.create!(content: 'bar foo baz')
+ ModelWithPgSearch.create!(content: "foo bar"),
+ ModelWithPgSearch.create!(content: "bar foo"),
+ ModelWithPgSearch.create!(content: "bar foo baz")
]
excluded = [
- ModelWithPgSearch.create!(content: 'foo'),
- ModelWithPgSearch.create!(content: 'foo baz')
+ ModelWithPgSearch.create!(content: "foo"),
+ ModelWithPgSearch.create!(content: "foo baz")
]
- results = ModelWithPgSearch.search_content('foo bar')
+ results = ModelWithPgSearch.search_content("foo bar")
expect(results).to match_array(included)
expect(results).not_to include(excluded)
end
@@ -493,7 +505,7 @@
# WARNING: searching timestamps is not something PostgreSQL
# full-text search is good at. Use at your own risk.
pg_search_scope :search_timestamps,
- against: %i[created_at updated_at]
+ against: %i[created_at updated_at]
end
end
@@ -514,15 +526,15 @@
it "returns rows whose columns contain all of the terms in the query across columns" do
included = [
- ModelWithPgSearch.create!(title: 'foo', content: 'bar'),
- ModelWithPgSearch.create!(title: 'bar', content: 'foo')
+ ModelWithPgSearch.create!(title: "foo", content: "bar"),
+ ModelWithPgSearch.create!(title: "bar", content: "foo")
]
excluded = [
- ModelWithPgSearch.create!(title: 'foo', content: 'foo'),
- ModelWithPgSearch.create!(title: 'bar', content: 'bar')
+ ModelWithPgSearch.create!(title: "foo", content: "foo"),
+ ModelWithPgSearch.create!(title: "bar", content: "bar")
]
- results = ModelWithPgSearch.search_title_and_content('foo bar')
+ results = ModelWithPgSearch.search_title_and_content("foo bar")
expect(results).to match_array(included)
excluded.each do |result|
@@ -531,17 +543,17 @@
end
it "returns rows where at one column contains all of the terms in the query and another does not" do
- in_title = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
- in_content = ModelWithPgSearch.create!(title: 'bar', content: 'foo')
+ in_title = ModelWithPgSearch.create!(title: "foo", content: "bar")
+ in_content = ModelWithPgSearch.create!(title: "bar", content: "foo")
- results = ModelWithPgSearch.search_title_and_content('foo')
- expect(results).to match_array([in_title, in_content])
+ results = ModelWithPgSearch.search_title_and_content("foo")
+ expect(results).to contain_exactly(in_title, in_content)
end
# Searching with a NULL column will prevent any matches unless we coalesce it.
it "returns rows where at one column contains all of the terms in the query and another is NULL" do
- included = ModelWithPgSearch.create!(title: 'foo', content: nil)
- results = ModelWithPgSearch.search_title_and_content('foo')
+ included = ModelWithPgSearch.create!(title: "foo", content: nil)
+ results = ModelWithPgSearch.search_title_and_content("foo")
expect(results).to eq([included])
end
end
@@ -552,21 +564,21 @@
end
it "returns rows where one searchable column and the query share enough trigrams" do
- included = ModelWithPgSearch.create!(title: 'abcdefghijkl', content: nil)
- results = ModelWithPgSearch.with_trigrams('cdefhijkl')
+ included = ModelWithPgSearch.create!(title: "abcdefghijkl", content: nil)
+ results = ModelWithPgSearch.with_trigrams("cdefhijkl")
expect(results).to eq([included])
end
it "returns rows where multiple searchable columns and the query share enough trigrams" do
- included = ModelWithPgSearch.create!(title: 'abcdef', content: 'ghijkl')
- results = ModelWithPgSearch.with_trigrams('cdefhijkl')
+ included = ModelWithPgSearch.create!(title: "abcdef", content: "ghijkl")
+ results = ModelWithPgSearch.with_trigrams("cdefhijkl")
expect(results).to eq([included])
end
context "when a threshold is specified" do
before do
- ModelWithPgSearch.pg_search_scope :with_strict_trigrams, against: %i[title content], using: { trigram: { threshold: 0.5 } }
- ModelWithPgSearch.pg_search_scope :with_permissive_trigrams, against: %i[title content], using: { trigram: { threshold: 0.1 } }
+ ModelWithPgSearch.pg_search_scope :with_strict_trigrams, against: %i[title content], using: {trigram: {threshold: 0.5}}
+ ModelWithPgSearch.pg_search_scope :with_permissive_trigrams, against: %i[title content], using: {trigram: {threshold: 0.1}}
end
it "uses the threshold in the trigram expression" do
@@ -591,16 +603,16 @@
context "when using tsearch" do
before do
ModelWithPgSearch.pg_search_scope :search_title_with_prefixes,
- against: :title,
- using: {
- tsearch: { prefix: true }
- }
+ against: :title,
+ using: {
+ tsearch: {prefix: true}
+ }
end
context "with prefix: true" do
it "returns rows that match the query and that are prefixed by the query" do
- included = ModelWithPgSearch.create!(title: 'prefix')
- excluded = ModelWithPgSearch.create!(title: 'postfix')
+ included = ModelWithPgSearch.create!(title: "prefix")
+ excluded = ModelWithPgSearch.create!(title: "postfix")
results = ModelWithPgSearch.search_title_with_prefixes("pre")
expect(results).to eq([included])
@@ -608,8 +620,8 @@
end
it "returns rows that match the query when the query has a hyphen" do
- included = ModelWithPgSearch.create!(title: 'foo-bar')
- excluded = ModelWithPgSearch.create!(title: 'foo bar')
+ included = ModelWithPgSearch.create!(title: "foo-bar")
+ excluded = ModelWithPgSearch.create!(title: "foo bar")
results = ModelWithPgSearch.search_title_with_prefixes("foo-bar")
expect(results).to include(included)
@@ -620,16 +632,16 @@
context "with the english dictionary" do
before do
ModelWithPgSearch.pg_search_scope :search_content_with_english,
- against: :content,
- using: {
- tsearch: { dictionary: :english }
- }
+ against: :content,
+ using: {
+ tsearch: {dictionary: :english}
+ }
end
it "returns rows that match the query when stemmed by the english dictionary" do
included = [ModelWithPgSearch.create!(content: "jump"),
- ModelWithPgSearch.create!(content: "jumped"),
- ModelWithPgSearch.create!(content: "jumping")]
+ ModelWithPgSearch.create!(content: "jumped"),
+ ModelWithPgSearch.create!(content: "jumping")]
results = ModelWithPgSearch.search_content_with_english("jump")
expect(results).to match_array(included)
@@ -639,14 +651,14 @@
describe "highlighting" do
before do
["Strip Down", "Down", "Down and Out", "Won't Let You Down"].each do |name|
- ModelWithPgSearch.create! title: 'Just a title', content: name
+ ModelWithPgSearch.create! title: "Just a title", content: name
end
end
context "with highlight turned on" do
before do
ModelWithPgSearch.pg_search_scope :search_content,
- against: :content
+ against: :content
end
it "adds a #pg_search_highlight method to each returned model record" do
@@ -661,31 +673,31 @@
expect(result.pg_search_highlight).to eq("Won't Let You Down")
end
- it 'preserves column selection when with_pg_search_highlight is chained after a select()' do
+ it "preserves column selection when with_pg_search_highlight is chained after a select()" do
result = ModelWithPgSearch.search_content("Let").select(:content).with_pg_search_highlight.first
- expect(result.as_json.keys).to contain_exactly('id', 'content', 'pg_search_highlight')
+ expect(result.as_json.keys).to contain_exactly("id", "content", "pg_search_highlight")
end
end
context "with custom highlighting options" do
before do
- ModelWithPgSearch.create! content: "#{'text ' * 2}Let #{'text ' * 2}Let #{'text ' * 2}"
+ ModelWithPgSearch.create! content: "#{"text " * 2}Let #{"text " * 2}Let #{"text " * 2}"
ModelWithPgSearch.pg_search_scope :search_content,
- against: :content,
- using: {
- tsearch: {
- highlight: {
- StartSel: '',
- StopSel: '',
- FragmentDelimiter: '',
- MaxFragments: 2,
- MaxWords: 2,
- MinWords: 1
- }
- }
- }
+ against: :content,
+ using: {
+ tsearch: {
+ highlight: {
+ StartSel: '',
+ StopSel: "",
+ FragmentDelimiter: '',
+ MaxFragments: 2,
+ MaxWords: 2,
+ MinWords: 1
+ }
+ }
+ }
end
it "applies the options to the excerpts" do
@@ -714,10 +726,10 @@
context "with a normalization specified" do
before do
ModelWithPgSearch.pg_search_scope :search_content_with_normalization,
- against: :content,
- using: {
- tsearch: { normalization: 2 }
- }
+ against: :content,
+ using: {
+ tsearch: {normalization: 2}
+ }
end
it "ranks the results for documents with less text higher" do
@@ -731,8 +743,8 @@
context "with no normalization" do
before do
ModelWithPgSearch.pg_search_scope :search_content_without_normalization,
- against: :content,
- using: :tsearch
+ against: :content,
+ using: :tsearch
end
it "ranks the results equally" do
@@ -747,14 +759,14 @@
context "when against columns ranked with arrays" do
before do
ModelWithPgSearch.pg_search_scope :search_weighted_by_array_of_arrays,
- against: [[:content, 'B'], [:title, 'A']]
+ against: [[:content, "B"], [:title, "A"]]
end
it "returns results sorted by weighted rank" do
- loser = ModelWithPgSearch.create!(title: 'bar', content: 'foo')
- winner = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
+ loser = ModelWithPgSearch.create!(title: "bar", content: "foo")
+ winner = ModelWithPgSearch.create!(title: "foo", content: "bar")
- results = ModelWithPgSearch.search_weighted_by_array_of_arrays('foo').with_pg_search_rank
+ results = ModelWithPgSearch.search_weighted_by_array_of_arrays("foo").with_pg_search_rank
expect(results[0].pg_search_rank).to be > results[1].pg_search_rank
expect(results).to eq([winner, loser])
end
@@ -763,14 +775,14 @@
context "when against columns ranked with a hash" do
before do
ModelWithPgSearch.pg_search_scope :search_weighted_by_hash,
- against: { content: 'B', title: 'A' }
+ against: {content: "B", title: "A"}
end
it "returns results sorted by weighted rank" do
- loser = ModelWithPgSearch.create!(title: 'bar', content: 'foo')
- winner = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
+ loser = ModelWithPgSearch.create!(title: "bar", content: "foo")
+ winner = ModelWithPgSearch.create!(title: "foo", content: "bar")
- results = ModelWithPgSearch.search_weighted_by_hash('foo').with_pg_search_rank
+ results = ModelWithPgSearch.search_weighted_by_hash("foo").with_pg_search_rank
expect(results[0].pg_search_rank).to be > results[1].pg_search_rank
expect(results).to eq([winner, loser])
end
@@ -779,14 +791,14 @@
context "when against columns of which only some are ranked" do
before do
ModelWithPgSearch.pg_search_scope :search_weighted,
- against: [:content, [:title, 'A']]
+ against: [:content, [:title, "A"]]
end
it "returns results sorted by weighted rank using an implied low rank for unranked columns" do
- loser = ModelWithPgSearch.create!(title: 'bar', content: 'foo')
- winner = ModelWithPgSearch.create!(title: 'foo', content: 'bar')
+ loser = ModelWithPgSearch.create!(title: "bar", content: "foo")
+ winner = ModelWithPgSearch.create!(title: "foo", content: "bar")
- results = ModelWithPgSearch.search_weighted('foo').with_pg_search_rank
+ results = ModelWithPgSearch.search_weighted("foo").with_pg_search_rank
expect(results[0].pg_search_rank).to be > results[1].pg_search_rank
expect(results).to eq([winner, loser])
end
@@ -795,17 +807,17 @@
context "when searching any_word option" do
before do
ModelWithPgSearch.pg_search_scope :search_title_with_any_word,
- against: :title,
- using: {
- tsearch: { any_word: true }
- }
+ against: :title,
+ using: {
+ tsearch: {any_word: true}
+ }
ModelWithPgSearch.pg_search_scope :search_title_with_all_words,
- against: :title
+ against: :title
end
it "returns all results containing any word in their title" do
- numbers = %w[one two three four].map { |number| ModelWithPgSearch.create!(title: number) }
+ %w[one two three four].map { |number| ModelWithPgSearch.create!(title: number) }
results = ModelWithPgSearch.search_title_with_any_word("one two three four")
@@ -820,10 +832,10 @@
context "with :negation" do
before do
ModelWithPgSearch.pg_search_scope :search_with_negation,
- against: :title,
- using: {
- tsearch: { negation: true }
- }
+ against: :title,
+ using: {
+ tsearch: {negation: true}
+ }
end
it "doesn't return results that contain terms prepended with '!'" do
@@ -847,10 +859,10 @@
context "without :negation" do
before do
ModelWithPgSearch.pg_search_scope :search_without_negation,
- against: :title,
- using: {
- tsearch: {}
- }
+ against: :title,
+ using: {
+ tsearch: {}
+ }
end
it "return results that contain terms prepended with '!'" do
@@ -873,38 +885,37 @@
context "when using dmetaphone" do
before do
ModelWithPgSearch.pg_search_scope :with_dmetaphones,
- against: %i[title content],
- using: :dmetaphone
+ against: %i[title content],
+ using: :dmetaphone
end
it "returns rows where one searchable column and the query share enough dmetaphones" do
- included = ModelWithPgSearch.create!(title: 'Geoff', content: nil)
- excluded = ModelWithPgSearch.create!(title: 'Bob', content: nil)
- results = ModelWithPgSearch.with_dmetaphones('Jeff')
+ included = ModelWithPgSearch.create!(title: "Geoff", content: nil)
+ ModelWithPgSearch.create!(title: "Bob", content: nil)
+ results = ModelWithPgSearch.with_dmetaphones("Jeff")
expect(results).to eq([included])
end
it "returns rows where multiple searchable columns and the query share enough dmetaphones" do
- included = ModelWithPgSearch.create!(title: 'Geoff', content: 'George')
- excluded = ModelWithPgSearch.create!(title: 'Bob', content: 'Jones')
- results = ModelWithPgSearch.with_dmetaphones('Jeff Jorge')
+ included = ModelWithPgSearch.create!(title: "Geoff", content: "George")
+ ModelWithPgSearch.create!(title: "Bob", content: "Jones")
+ results = ModelWithPgSearch.with_dmetaphones("Jeff Jorge")
expect(results).to eq([included])
end
it "returns rows that match dmetaphones that are English stopwords" do
- included = ModelWithPgSearch.create!(title: 'White', content: nil)
- excluded = ModelWithPgSearch.create!(title: 'Black', content: nil)
- results = ModelWithPgSearch.with_dmetaphones('Wight')
+ included = ModelWithPgSearch.create!(title: "White", content: nil)
+ ModelWithPgSearch.create!(title: "Black", content: nil)
+ results = ModelWithPgSearch.with_dmetaphones("Wight")
expect(results).to eq([included])
end
it "can handle terms that do not have a dmetaphone equivalent" do
- term_with_blank_metaphone = "w"
-
- included = ModelWithPgSearch.create!(title: 'White', content: nil)
- excluded = ModelWithPgSearch.create!(title: 'Black', content: nil)
+ included = ModelWithPgSearch.create!(title: "White", content: nil)
+ ModelWithPgSearch.create!(title: "Black", content: nil)
- results = ModelWithPgSearch.with_dmetaphones('Wight W')
+ # "W" does not have a dmetaphone equivalent
+ results = ModelWithPgSearch.with_dmetaphones("Wight W")
expect(results).to eq([included])
end
end
@@ -912,35 +923,35 @@
context "when using multiple features" do
before do
ModelWithPgSearch.pg_search_scope :with_tsearch,
- against: :title,
- using: [
- [:tsearch, { dictionary: 'english' }]
- ]
+ against: :title,
+ using: [
+ [:tsearch, {dictionary: "english"}]
+ ]
ModelWithPgSearch.pg_search_scope :with_trigram,
- against: :title,
- using: :trigram
+ against: :title,
+ using: :trigram
ModelWithPgSearch.pg_search_scope :with_trigram_and_ignoring_accents,
- against: :title,
- ignoring: :accents,
- using: :trigram
+ against: :title,
+ ignoring: :accents,
+ using: :trigram
ModelWithPgSearch.pg_search_scope :with_tsearch_and_trigram,
- against: :title,
- using: [
- [:tsearch, { dictionary: 'english' }],
- :trigram
- ]
+ against: :title,
+ using: [
+ [:tsearch, {dictionary: "english"}],
+ :trigram
+ ]
ModelWithPgSearch.pg_search_scope :complex_search,
- against: %i[content title],
- ignoring: :accents,
- using: {
- tsearch: { dictionary: 'english' },
- dmetaphone: {},
- trigram: {}
- }
+ against: %i[content title],
+ ignoring: :accents,
+ using: {
+ tsearch: {dictionary: "english"},
+ dmetaphone: {},
+ trigram: {}
+ }
end
it "returns rows that match using any of the features" do
@@ -982,13 +993,13 @@
end
context "with feature-specific configuration" do
- let(:tsearch_config) { { dictionary: 'english' } }
- let(:trigram_config) { { foo: 'bar' } }
+ let(:tsearch_config) { {dictionary: "english"} }
+ let(:trigram_config) { {foo: "bar"} }
before do
ModelWithPgSearch.pg_search_scope :with_tsearch_and_trigram_using_hash,
- against: :title,
- using: { tsearch: tsearch_config, trigram: trigram_config }
+ against: :title,
+ using: {tsearch: tsearch_config, trigram: trigram_config}
end
it "passes the custom configuration down to the specified feature" do
@@ -1029,8 +1040,8 @@
with_model :Post do
table do |t|
- t.text 'content'
- t.tsvector 'content_tsvector'
+ t.text "content"
+ t.tsvector "content_tsvector"
end
model do
@@ -1039,8 +1050,8 @@
end
end
- let!(:expected) { Post.create!(content: 'phooey') }
- let!(:unexpected) { Post.create!(content: 'longcat is looooooooong') }
+ let!(:expected) { Post.create!(content: "phooey") }
+ let!(:unexpected) { Post.create!(content: "longcat is looooooooong") }
before do
ActiveRecord::Base.connection.execute <<~SQL.squish
@@ -1048,17 +1059,17 @@
SET content_tsvector = to_tsvector('english'::regconfig, #{Post.quoted_table_name}."content")
SQL
- expected.comments.create(body: 'commentone')
- unexpected.comments.create(body: 'commentwo')
+ expected.comments.create(body: "commentone")
+ unexpected.comments.create(body: "commentwo")
Post.pg_search_scope :search_by_content_with_tsvector,
- associated_against: { comments: [:body] },
- using: {
- tsearch: {
- tsvector_column: 'content_tsvector',
- dictionary: 'english'
- }
- }
+ associated_against: {comments: [:body]},
+ using: {
+ tsearch: {
+ tsvector_column: "content_tsvector",
+ dictionary: "english"
+ }
+ }
end
it "finds by the tsvector column" do
@@ -1069,7 +1080,7 @@
expect(Post.search_by_content_with_tsvector("commentone").map(&:id)).to eq([expected.id])
end
- it 'finds by a combination of the two' do
+ it "finds by a combination of the two" do
expect(Post.search_by_content_with_tsvector("phooey commentone").map(&:id)).to eq([expected.id])
end
end
@@ -1153,8 +1164,8 @@
end
it "concats tsvector columns" do
- expected = "coalesce(#{ModelWithTsvector.quoted_table_name}.\"content_tsvector\", '') || "\
- "coalesce(#{ModelWithTsvector.quoted_table_name}.\"message_tsvector\", '')"
+ expected = "coalesce((#{ModelWithTsvector.quoted_table_name}.\"content_tsvector\")::tsvector, '') || "\
+ "coalesce((#{ModelWithTsvector.quoted_table_name}.\"message_tsvector\")::tsvector, '')"
expect(ModelWithTsvector.search_by_multiple_tsvector_columns("something").to_sql).to include(expected)
end
@@ -1180,10 +1191,10 @@
end
it 'concats tsvector columns' do
- expected = "setweight(coalesce(#{ModelWithTsvector.quoted_table_name}.\"title_tsvector\", ''), 'A') || "\
- "coalesce(#{ModelWithTsvector.quoted_table_name}.\"content_tsvector\", '') || "\
- "setweight(coalesce(#{ModelWithTsvector.quoted_table_name}.\"message_tsvector\", ''), 'B')"
-
+ expected = "setweight(coalesce((#{ModelWithTsvector.quoted_table_name}.\"title_tsvector\")::tsvector, ''), 'A') || "\
+ "coalesce((#{ModelWithTsvector.quoted_table_name}.\"content_tsvector\")::tsvector, '') || "\
+ "setweight(coalesce((#{ModelWithTsvector.quoted_table_name}.\"message_tsvector\")::tsvector, ''), 'B')"
+
expect(ModelWithTsvector.search_by_multiple_tsvector_columns("something").to_sql).to include(expected)
end
end
@@ -1228,7 +1239,7 @@
with_model :AnotherModel do
table do |t|
t.string :content_tsvector # the type of the column doesn't matter
- t.belongs_to :model_with_tsvector
+ t.belongs_to :model_with_tsvector, index: { name: :boopilooopi_id_ix }
end
end
@@ -1248,7 +1259,7 @@
table do |t|
t.text 'content'
t.tsvector 'content_tsvector'
- t.belongs_to :model_with_tsvector
+ t.belongs_to :model_with_tsvector, index: { name: :hoopaboopa_id_ix }
end
end
@@ -1318,7 +1329,7 @@
table do |t|
t.text 'content'
t.tsvector 'content_tsvector'
- t.belongs_to :model_with_tsvector
+ t.belongs_to :model_with_tsvector, index: { name: :anoterh_model_id_ix }
end
end
@@ -1385,17 +1396,17 @@
include PgSearch::Model
pg_search_scope :search_by_multiple_tsvector_columns,
- against: ['content', 'message'],
- using: {
- tsearch: {
- tsvector_column: ['content_tsvector', 'message_tsvector'],
- dictionary: 'english'
- }
- }
+ against: ["content", "message"],
+ using: {
+ tsearch: {
+ tsvector_column: ["content_tsvector", "message_tsvector"],
+ dictionary: "english"
+ }
+ }
end
end
- it 'concats tsvector columns' do
+ it "concats tsvector columns" do
expected = "#{ModelWithTsvector.quoted_table_name}.\"content_tsvector\" || " \
"#{ModelWithTsvector.quoted_table_name}.\"message_tsvector\""
@@ -1406,17 +1417,17 @@
context "when using a tsvector column with" do
with_model :ModelWithTsvector do
table do |t|
- t.text 'content'
- t.tsvector 'content_tsvector'
+ t.text "content"
+ t.tsvector "content_tsvector"
end
model { include PgSearch::Model }
end
- let!(:expected) { ModelWithTsvector.create!(content: 'tiling is grouty') }
+ let!(:expected) { ModelWithTsvector.create!(content: "tiling is grouty") }
before do
- ModelWithTsvector.create!(content: 'longcat is looooooooong')
+ ModelWithTsvector.create!(content: "longcat is looooooooong")
ActiveRecord::Base.connection.execute <<~SQL.squish
UPDATE #{ModelWithTsvector.quoted_table_name}
@@ -1424,13 +1435,13 @@
SQL
ModelWithTsvector.pg_search_scope :search_by_content_with_tsvector,
- against: :content,
- using: {
- tsearch: {
- tsvector_column: 'content_tsvector',
- dictionary: 'english'
- }
- }
+ against: :content,
+ using: {
+ tsearch: {
+ tsvector_column: "content_tsvector",
+ dictionary: "english"
+ }
+ }
end
it "does not use to_tsvector in the query" do
@@ -1464,8 +1475,8 @@
context "when ignoring accents" do
before do
ModelWithPgSearch.pg_search_scope :search_title_without_accents,
- against: :title,
- ignoring: :accents
+ against: :title,
+ ignoring: :accents
end
it "returns rows that match the query but not its accents" do
@@ -1484,7 +1495,7 @@
let(:results) { ModelWithPgSearch.search_title_without_accents(term) }
before do
- ModelWithPgSearch.create!(title: 'FooBar')
+ ModelWithPgSearch.create!(title: "FooBar")
end
it "does not create an erroneous tsquery expression" do
@@ -1496,25 +1507,25 @@
context "when passed a :ranked_by expression" do
before do
ModelWithPgSearch.pg_search_scope :search_content_with_default_rank,
- against: :content
+ against: :content
ModelWithPgSearch.pg_search_scope :search_content_with_importance_as_rank,
- against: :content,
- ranked_by: "importance"
+ against: :content,
+ ranked_by: "importance"
ModelWithPgSearch.pg_search_scope :search_content_with_importance_as_rank_multiplier,
- against: :content,
- ranked_by: ":tsearch * importance"
+ against: :content,
+ ranked_by: ":tsearch * importance"
end
it "returns records with a rank attribute equal to the :ranked_by expression" do
- ModelWithPgSearch.create!(content: 'foo', importance: 10)
+ ModelWithPgSearch.create!(content: "foo", importance: 10)
results = ModelWithPgSearch.search_content_with_importance_as_rank("foo").with_pg_search_rank
expect(results.first.pg_search_rank).to eq(10)
end
it "substitutes :tsearch with the tsearch rank expression in the :ranked_by expression" do
- ModelWithPgSearch.create!(content: 'foo', importance: 10)
+ ModelWithPgSearch.create!(content: "foo", importance: 10)
tsearch_result =
ModelWithPgSearch.search_content_with_default_rank("foo").with_pg_search_rank.first
@@ -1523,8 +1534,8 @@
multiplied_result =
ModelWithPgSearch.search_content_with_importance_as_rank_multiplier("foo")
- .with_pg_search_rank
- .first
+ .with_pg_search_rank
+ .first
multiplied_rank = multiplied_result.pg_search_rank
@@ -1533,9 +1544,9 @@
it "returns results in descending order of the value of the rank expression" do
records = [
- ModelWithPgSearch.create!(content: 'foo', importance: 1),
- ModelWithPgSearch.create!(content: 'foo', importance: 3),
- ModelWithPgSearch.create!(content: 'foo', importance: 2)
+ ModelWithPgSearch.create!(content: "foo", importance: 1),
+ ModelWithPgSearch.create!(content: "foo", importance: 3),
+ ModelWithPgSearch.create!(content: "foo", importance: 2)
]
results = ModelWithPgSearch.search_content_with_importance_as_rank("foo")
@@ -1548,32 +1559,32 @@
before do
ModelWithPgSearch.pg_search_scope scope_name,
- against: :content,
- ranked_by: ":#{feature}"
+ against: :content,
+ ranked_by: ":#{feature}"
- ModelWithPgSearch.create!(content: 'foo')
+ ModelWithPgSearch.create!(content: "foo")
end
context "when .with_pg_search_rank is chained after" do
specify "its results respond to #pg_search_rank" do
- result = ModelWithPgSearch.send(scope_name, 'foo').with_pg_search_rank.first
+ result = ModelWithPgSearch.send(scope_name, "foo").with_pg_search_rank.first
expect(result).to respond_to(:pg_search_rank)
end
it "returns the rank when #pg_search_rank is called on a result" do
- results = ModelWithPgSearch.send(scope_name, 'foo').with_pg_search_rank
+ results = ModelWithPgSearch.send(scope_name, "foo").with_pg_search_rank
expect(results.first.pg_search_rank).to be_a Float
end
end
context "when .with_pg_search_rank is not chained after" do
specify "its results do not respond to #pg_search_rank" do
- result = ModelWithPgSearch.send(scope_name, 'foo').first
+ result = ModelWithPgSearch.send(scope_name, "foo").first
expect(result).not_to respond_to(:pg_search_rank)
end
it "raises PgSearch::PgSearchRankNotSelected when #pg_search_rank is called on a result" do
- result = ModelWithPgSearch.send(scope_name, 'foo').first
+ result = ModelWithPgSearch.send(scope_name, "foo").first
expect {
result.pg_search_rank
}.to raise_exception(PgSearch::PgSearchRankNotSelected)
@@ -1585,14 +1596,14 @@
context "when using the tsearch ranking algorithm" do
it "sorts results by the tsearch rank" do
ModelWithPgSearch.pg_search_scope :search_content_ranked_by_tsearch,
- using: :tsearch,
- against: :content,
- ranked_by: ":tsearch"
+ using: :tsearch,
+ against: :content,
+ ranked_by: ":tsearch"
- once = ModelWithPgSearch.create!(content: 'foo bar')
- twice = ModelWithPgSearch.create!(content: 'foo foo')
+ once = ModelWithPgSearch.create!(content: "foo bar")
+ twice = ModelWithPgSearch.create!(content: "foo foo")
- results = ModelWithPgSearch.search_content_ranked_by_tsearch('foo')
+ results = ModelWithPgSearch.search_content_ranked_by_tsearch("foo")
expect(results.find_index(twice)).to be < results.find_index(once)
end
end
@@ -1600,14 +1611,14 @@
context "when using the trigram ranking algorithm" do
it "sorts results by the trigram rank" do
ModelWithPgSearch.pg_search_scope :search_content_ranked_by_trigram,
- using: :trigram,
- against: :content,
- ranked_by: ":trigram"
+ using: :trigram,
+ against: :content,
+ ranked_by: ":trigram"
- close = ModelWithPgSearch.create!(content: 'abcdef')
- exact = ModelWithPgSearch.create!(content: 'abc')
+ close = ModelWithPgSearch.create!(content: "abcdef")
+ exact = ModelWithPgSearch.create!(content: "abc")
- results = ModelWithPgSearch.search_content_ranked_by_trigram('abc')
+ results = ModelWithPgSearch.search_content_ranked_by_trigram("abc")
expect(results.find_index(exact)).to be < results.find_index(close)
end
end
@@ -1615,14 +1626,14 @@
context "when using the dmetaphone ranking algorithm" do
it "sorts results by the dmetaphone rank" do
ModelWithPgSearch.pg_search_scope :search_content_ranked_by_dmetaphone,
- using: :dmetaphone,
- against: :content,
- ranked_by: ":dmetaphone"
+ using: :dmetaphone,
+ against: :content,
+ ranked_by: ":dmetaphone"
- once = ModelWithPgSearch.create!(content: 'Phoo Bar')
- twice = ModelWithPgSearch.create!(content: 'Phoo Fu')
+ once = ModelWithPgSearch.create!(content: "Phoo Bar")
+ twice = ModelWithPgSearch.create!(content: "Phoo Fu")
- results = ModelWithPgSearch.search_content_ranked_by_dmetaphone('foo')
+ results = ModelWithPgSearch.search_content_ranked_by_dmetaphone("foo")
expect(results.find_index(twice)).to be < results.find_index(once)
end
end
@@ -1631,17 +1642,17 @@
context "when there is a sort only feature" do
it "excludes that feature from the conditions, but uses it in the sorting" do
ModelWithPgSearch.pg_search_scope :search_content_ranked_by_dmetaphone,
- against: :content,
- using: {
- tsearch: { any_word: true, prefix: true },
- dmetaphone: { any_word: true, prefix: true, sort_only: true }
- },
- ranked_by: ":tsearch + (0.5 * :dmetaphone)"
+ against: :content,
+ using: {
+ tsearch: {any_word: true, prefix: true},
+ dmetaphone: {any_word: true, prefix: true, sort_only: true}
+ },
+ ranked_by: ":tsearch + (0.5 * :dmetaphone)"
exact = ModelWithPgSearch.create!(content: "ash hines")
one_exact_one_close = ModelWithPgSearch.create!(content: "ash heinz")
one_exact = ModelWithPgSearch.create!(content: "ash smith")
- one_close = ModelWithPgSearch.create!(content: "leigh heinz")
+ ModelWithPgSearch.create!(content: "leigh heinz")
results = ModelWithPgSearch.search_content_ranked_by_dmetaphone("ash hines")
expect(results).to eq [exact, one_exact_one_close, one_exact]
diff --git a/spec/integration/single_table_inheritance_spec.rb b/spec/integration/single_table_inheritance_spec.rb
index 8a068152..e4cb8668 100644
--- a/spec/integration/single_table_inheritance_spec.rb
+++ b/spec/integration/single_table_inheritance_spec.rb
@@ -6,8 +6,8 @@
context "with the standard type column" do
with_model :SuperclassModel do
table do |t|
- t.text 'content'
- t.string 'type'
+ t.text "content"
+ t.string "type"
end
model do
@@ -46,13 +46,13 @@
context "with a custom type column" do
with_model :SuperclassModel do
table do |t|
- t.text 'content'
- t.string 'custom_type'
+ t.text "content"
+ t.string "custom_type"
end
model do
include PgSearch::Model
- self.inheritance_column = 'custom_type'
+ self.inheritance_column = "custom_type"
pg_search_scope :search_content, against: :content
end
end
diff --git a/spec/lib/pg_search/configuration/association_spec.rb b/spec/lib/pg_search/configuration/association_spec.rb
index 6c7d79d2..05725e69 100644
--- a/spec/lib/pg_search/configuration/association_spec.rb
+++ b/spec/lib/pg_search/configuration/association_spec.rb
@@ -1,222 +1,143 @@
+# frozen_string_literal: true
+
require "spec_helper"
+# rubocop:disable RSpec/NestedGroups
describe PgSearch::Configuration::Association do
- context "through a belongs_to association" do
- with_model :AssociatedModel do
- table do |t|
- t.string "title"
- end
+ with_model :Avatar do
+ table do |t|
+ t.string :url
+ t.references :user
end
+ end
- with_model :Model do
- table do |t|
- t.string "title"
- t.belongs_to :another_model
- end
-
- model do
- include PgSearch
- belongs_to :another_model, :class_name => 'AssociatedModel'
-
- pg_search_scope :with_another, :associated_against => {:another_model => :title}
- end
- end
-
- let(:association) { described_class.new(Model, :another_model, :title) }
-
- describe "#table_name" do
- it "returns the table name for the associated model" do
- expect(association.table_name).to eq AssociatedModel.table_name
- end
+ with_model :User do
+ table do |t|
+ t.string :name
+ t.belongs_to :site
end
- describe "#join" do
- context "given any postgresql_version" do
- let(:column_select) do
- "\"#{association.table_name}\".\"title\""
- end
-
- let(:expected_sql) do
- <<-EOS.gsub(/\s+/, ' ').strip
- LEFT OUTER JOIN
- (SELECT model_id AS id,
- #{column_select} AS #{association.columns.first.alias}
- FROM \"#{Model.table_name}\"
- INNER JOIN \"#{association.table_name}\"
- ON \"#{association.table_name}\".\"id\" = \"#{Model.table_name}\".\"another_model_id\") #{association.subselect_alias}
- ON #{association.subselect_alias}.id = model_id
- EOS
- end
-
- it "returns the correct SQL join (v1)" do
- allow(Model.connection).to receive(:postgresql_version).and_return(1)
- expect(association.join("model_id")).to eq(expected_sql)
- end
+ model do
+ include PgSearch::Model
+ has_one :avatar, class_name: "Avatar"
+ belongs_to :site
- it "returns the correct SQL join (v100)" do
- allow(Model.connection).to receive(:postgresql_version).and_return(100_000)
- expect(association.join("model_id")).to eq(expected_sql)
- end
- end
- end
-
- describe "#subselect_alias" do
- it "returns a consistent string" do
- subselect_alias = association.subselect_alias
- expect(subselect_alias).to be_a String
- expect(association.subselect_alias).to eq subselect_alias
- end
+ pg_search_scope :with_avatar, associated_against: {avatar: :url}
+ pg_search_scope :with_site, associated_against: {site: :title}
end
end
- context "through a has_one association" do
- with_model :Model do
- table do |t|
- t.string "title"
- end
-
- model do
- include PgSearch
- has_one :another_model, :class_name => 'AssociatedModel', foreign_key: 'primary_model_id'
-
- pg_search_scope :with_another, :associated_against => {:another_model => :title}
- end
+ with_model :Site do
+ table do |t|
+ t.string :title
end
- with_model :AssociatedModel do
- table do |t|
- t.string "title"
- t.belongs_to :primary_model
- end
+ model do
+ include PgSearch::Model
+ has_many :users, class_name: "User"
- model do
- belongs_to :primary_model, :class_name => 'Model'
- end
+ pg_search_scope :with_users, associated_against: {users: :name}
end
+ end
- let(:association) { described_class.new(Model, :another_model, :title) }
+ context "with has_one" do
+ let(:association) { described_class.new(User, :avatar, :url) }
describe "#table_name" do
it "returns the table name for the associated model" do
- expect(association.table_name).to eq AssociatedModel.table_name
+ expect(association.table_name).to eq Avatar.table_name
end
end
describe "#join" do
- context "given any postgresql_version" do
- let(:column_select) do
- "\"#{association.table_name}\".\"title\""
- end
-
- let(:expected_sql) do
- <<-EOS.gsub(/\s+/, ' ').strip
- LEFT OUTER JOIN
- (SELECT model_id AS id,
- #{column_select} AS #{association.columns.first.alias}
- FROM \"#{Model.table_name}\"
- INNER JOIN \"#{association.table_name}\"
- ON \"#{association.table_name}\".\"primary_model_id\" = \"#{Model.table_name}\".\"id\") #{association.subselect_alias}
- ON #{association.subselect_alias}.id = model_id
- EOS
- end
-
- it "returns the correct SQL join (v1)" do
- allow(Model.connection).to receive(:postgresql_version).and_return(1)
- expect(association.join("model_id")).to eq(expected_sql)
- end
-
- it "returns the correct SQL join (v100)" do
- allow(Model.connection).to receive(:postgresql_version).and_return(100_000)
- expect(association.join("model_id")).to eq(expected_sql)
- end
+ let(:expected_sql) do
+ <<~SQL.squish
+ LEFT OUTER JOIN
+ (SELECT model_id AS id,
+ #{column_select} AS #{association.columns.first.alias}
+ FROM "#{User.table_name}"
+ INNER JOIN "#{association.table_name}"
+ ON "#{association.table_name}"."user_id" = "#{User.table_name}"."id") #{association.subselect_alias}
+ ON #{association.subselect_alias}.id = model_id
+ SQL
+ end
+ let(:column_select) do
+ "\"#{association.table_name}\".\"url\"::text"
end
- end
- describe "#subselect_alias" do
- it "returns a consistent string" do
- subselect_alias = association.subselect_alias
- expect(subselect_alias).to be_a String
- expect(association.subselect_alias).to eq subselect_alias
+ it "returns the correct SQL join" do
+ expect(association.join("model_id")).to eq(expected_sql)
end
end
end
- context "through a has_many association" do
- with_model :Model do
- table do |t|
- t.string "title"
- end
-
- model do
- include PgSearch
- has_many :associated_models, :class_name => 'AssociatedModel', foreign_key: 'primary_model_id'
+ context "with belongs_to" do
+ let(:association) { described_class.new(User, :site, :title) }
- pg_search_scope :with_another, :associated_against => {:another_model => :title}
+ describe "#table_name" do
+ it "returns the table name for the associated model" do
+ expect(association.table_name).to eq Site.table_name
end
end
- with_model :AssociatedModel do
- table do |t|
- t.string "title"
- t.belongs_to :primary_model
+ describe "#join" do
+ let(:expected_sql) do
+ <<~SQL.squish
+ LEFT OUTER JOIN
+ (SELECT model_id AS id,
+ #{column_select} AS #{association.columns.first.alias}
+ FROM "#{User.table_name}"
+ INNER JOIN "#{association.table_name}"
+ ON "#{association.table_name}"."id" = "#{User.table_name}"."site_id") #{association.subselect_alias}
+ ON #{association.subselect_alias}.id = model_id
+ SQL
+ end
+ let(:column_select) do
+ "\"#{association.table_name}\".\"title\"::text"
end
- model do
- belongs_to :primary_model, :class_name => 'Model'
+ it "returns the correct SQL join" do
+ expect(association.join("model_id")).to eq(expected_sql)
end
end
+ end
- let(:association) { described_class.new(Model, :associated_models, :title) }
+ context "with has_many" do
+ let(:association) { described_class.new(Site, :users, :name) }
describe "#table_name" do
it "returns the table name for the associated model" do
- expect(association.table_name).to eq AssociatedModel.table_name
+ expect(association.table_name).to eq User.table_name
end
end
describe "#join" do
let(:expected_sql) do
- <<-EOS.gsub(/\s+/, ' ').strip
+ <<~SQL.squish
LEFT OUTER JOIN
(SELECT model_id AS id,
- #{column_select} AS #{association.columns.first.alias}
- FROM \"#{Model.table_name}\"
- INNER JOIN \"#{association.table_name}\"
- ON \"#{association.table_name}\".\"primary_model_id\" = \"#{Model.table_name}\".\"id\"
+ string_agg("#{association.table_name}"."name"::text, ' ') AS #{association.columns.first.alias}
+ FROM "#{Site.table_name}"
+ INNER JOIN "#{association.table_name}"
+ ON "#{association.table_name}"."site_id" = "#{Site.table_name}"."id"
GROUP BY model_id) #{association.subselect_alias}
ON #{association.subselect_alias}.id = model_id
- EOS
+ SQL
end
- context "given postgresql_version 0..90_000" do
- let(:column_select) do
- "array_to_string(array_agg(\"#{association.table_name}\".\"title\"::text), ' ')"
- end
-
- it "returns the correct SQL join" do
- allow(Model.connection).to receive(:postgresql_version).and_return(1)
- expect(association.join("model_id")).to eq(expected_sql)
- end
+ it "returns the correct SQL join" do
+ expect(association.join("model_id")).to eq(expected_sql)
end
- context "given any other postgresql_version" do
- let(:column_select) do
- "string_agg(\"#{association.table_name}\".\"title\"::text, ' ')"
- end
-
- it "returns the correct SQL join" do
- allow(Model.connection).to receive(:postgresql_version).and_return(100_000)
- expect(association.join("model_id")).to eq(expected_sql)
+ describe "#subselect_alias" do
+ it "returns a consistent string" do
+ subselect_alias = association.subselect_alias
+ expect(subselect_alias).to be_a String
+ expect(association.subselect_alias).to eq subselect_alias
end
end
end
-
- describe "#subselect_alias" do
- it "returns a consistent string" do
- subselect_alias = association.subselect_alias
- expect(subselect_alias).to be_a String
- expect(association.subselect_alias).to eq subselect_alias
- end
- end
end
end
+
+# rubocop:enable RSpec/NestedGroups
diff --git a/spec/lib/pg_search/configuration/column_spec.rb b/spec/lib/pg_search/configuration/column_spec.rb
index 50f56b1e..9e933713 100644
--- a/spec/lib/pg_search/configuration/column_spec.rb
+++ b/spec/lib/pg_search/configuration/column_spec.rb
@@ -7,6 +7,7 @@
with_model :Model do
table do |t|
t.string :name
+ t.json :object
end
end
@@ -14,18 +15,29 @@
column = described_class.new("name", nil, Model)
expect(column.full_name).to eq(%(#{Model.quoted_table_name}."name"))
end
+
+ it "returns nested json attributes" do
+ column = described_class.new(Arel.sql("object->>'name'"), nil, Model)
+ expect(column.full_name).to eq(%(object->>'name'))
+ end
end
describe "#to_sql" do
with_model :Model do
table do |t|
t.string :name
+ t.json :object
end
end
it "returns an expression that casts the column to text and coalesces it with an empty string" do
column = described_class.new("name", nil, Model)
- expect(column.to_sql).to eq(%{coalesce(#{Model.quoted_table_name}."name"::text, '')})
+ expect(column.to_sql).to eq(%{coalesce((#{Model.quoted_table_name}."name")::text, '')})
+ end
+
+ it "returns an expression that casts the nested json attribute to text and coalesces it with an empty string" do
+ column = described_class.new(Arel.sql("object->>'name'"), nil, Model)
+ expect(column.to_sql).to eq(%{coalesce((object->>'name')::text, '')})
end
end
end
diff --git a/spec/lib/pg_search/configuration/foreign_column_spec.rb b/spec/lib/pg_search/configuration/foreign_column_spec.rb
index 105b8eba..3537a203 100644
--- a/spec/lib/pg_search/configuration/foreign_column_spec.rb
+++ b/spec/lib/pg_search/configuration/foreign_column_spec.rb
@@ -18,16 +18,16 @@
model do
include PgSearch::Model
- belongs_to :another_model, class_name: 'AssociatedModel'
+ belongs_to :another_model, class_name: "AssociatedModel"
- pg_search_scope :with_another, associated_against: { another_model: :title }
+ pg_search_scope :with_another, associated_against: {another_model: :title}
end
end
it "returns a consistent string" do
association = PgSearch::Configuration::Association.new(Model,
- :another_model,
- :title)
+ :another_model,
+ :title)
foreign_column = described_class.new("title", nil, Model, association)
column_alias = foreign_column.alias
diff --git a/spec/lib/pg_search/features/dmetaphone_spec.rb b/spec/lib/pg_search/features/dmetaphone_spec.rb
index 995f76b4..8327619d 100644
--- a/spec/lib/pg_search/features/dmetaphone_spec.rb
+++ b/spec/lib/pg_search/features/dmetaphone_spec.rb
@@ -23,7 +23,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.rank.to_sql).to eq(
- %{(ts_rank((to_tsvector('simple', pg_search_dmetaphone(coalesce(#{Model.quoted_table_name}."name"::text, ''))) || to_tsvector('simple', pg_search_dmetaphone(coalesce(#{Model.quoted_table_name}."content"::text, '')))), (to_tsquery('simple', ''' ' || pg_search_dmetaphone('query') || ' ''')), 0))}
+ %{(ts_rank((to_tsvector('simple', pg_search_dmetaphone(coalesce((#{Model.quoted_table_name}."name")::text, ''))) || to_tsvector('simple', pg_search_dmetaphone(coalesce((#{Model.quoted_table_name}."content")::text, '')))), (to_tsquery('simple', ''' ' || pg_search_dmetaphone('query') || ' ''')), 0))}
)
end
end
@@ -48,7 +48,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %{((to_tsvector('simple', pg_search_dmetaphone(coalesce(#{Model.quoted_table_name}."name"::text, ''))) || to_tsvector('simple', pg_search_dmetaphone(coalesce(#{Model.quoted_table_name}."content"::text, '')))) @@ (to_tsquery('simple', ''' ' || pg_search_dmetaphone('query') || ' ''')))}
+ %{((to_tsvector('simple', pg_search_dmetaphone(coalesce((#{Model.quoted_table_name}."name")::text, ''))) || to_tsvector('simple', pg_search_dmetaphone(coalesce((#{Model.quoted_table_name}."content")::text, '')))) @@ (to_tsquery('simple', ''' ' || pg_search_dmetaphone('query') || ' ''')))}
)
end
end
diff --git a/spec/lib/pg_search/features/trigram_spec.rb b/spec/lib/pg_search/features/trigram_spec.rb
index 995c5dae..5686043a 100644
--- a/spec/lib/pg_search/features/trigram_spec.rb
+++ b/spec/lib/pg_search/features/trigram_spec.rb
@@ -1,13 +1,13 @@
# frozen_string_literal: true
-require 'spec_helper'
-require 'ostruct'
+require "spec_helper"
+require "ostruct"
# rubocop:disable RSpec/MultipleMemoizedHelpers, RSpec/NestedGroups
describe PgSearch::Features::Trigram do
subject(:feature) { described_class.new(query, options, columns, Model, normalizer) }
- let(:query) { 'lolwut' }
+ let(:query) { "lolwut" }
let(:options) { {} }
let(:columns) {
[
@@ -16,13 +16,13 @@
]
}
let(:normalizer) { PgSearch::Normalizer.new(config) }
- let(:config) { OpenStruct.new(ignore: []) } # rubocop:disable Style/OpenStructUse
+ let(:config) { OpenStruct.new(ignore: []) }
let(:coalesced_columns) do
<<~SQL.squish
- coalesce(#{Model.quoted_table_name}."name"::text, '')
+ coalesce((#{Model.quoted_table_name}."name")::text, '')
|| ' '
- || coalesce(#{Model.quoted_table_name}."content"::text, '')
+ || coalesce((#{Model.quoted_table_name}."content")::text, '')
SQL
end
@@ -33,15 +33,15 @@
end
end
- describe 'conditions' do
- it 'escapes the search document and query' do
+ describe "conditions" do
+ it "escapes the search document and query" do
config.ignore = []
expect(feature.conditions.to_sql).to eq("('#{query}' % (#{coalesced_columns}))")
end
- context 'when searching by word_similarity' do
+ context "when searching by word_similarity" do
let(:options) do
- { word_similarity: true }
+ {word_similarity: true}
end
it 'uses the "<%" operator when searching by word_similarity' do
@@ -50,17 +50,17 @@
end
end
- context 'when ignoring accents' do
- it 'escapes the search document and query, but not the accent function' do
+ context "when ignoring accents" do
+ it "escapes the search document and query, but not the accent function" do
config.ignore = [:accents]
expect(feature.conditions.to_sql).to eq("(unaccent('#{query}') % (unaccent(#{coalesced_columns})))")
end
end
- context 'when a threshold is specified' do
- context 'when searching by similarity' do
+ context "when a threshold is specified" do
+ context "when searching by similarity" do
let(:options) do
- { threshold: 0.5 }
+ {threshold: 0.5}
end
it 'uses a minimum similarity expression instead of the "%" operator' do
@@ -70,9 +70,9 @@
end
end
- context 'when searching by word_similarity' do
+ context "when searching by word_similarity" do
let(:options) do
- { threshold: 0.5, word_similarity: true }
+ {threshold: 0.5, word_similarity: true}
end
it 'uses a minimum similarity expression instead of the "<%" operator' do
@@ -83,28 +83,28 @@
end
end
- context 'when only certain columns are selected' do
- context 'with one column' do
- let(:options) { { only: :name } }
+ context "when only certain columns are selected" do
+ context "with one column" do
+ let(:options) { {only: :name} }
- it 'only searches against the select column' do
- coalesced_column = "coalesce(#{Model.quoted_table_name}.\"name\"::text, '')"
+ it "only searches against the select column" do
+ coalesced_column = "coalesce((#{Model.quoted_table_name}.\"name\")::text, '')"
expect(feature.conditions.to_sql).to eq("('#{query}' % (#{coalesced_column}))")
end
end
- context 'with multiple columns' do
- let(:options) { { only: %i[name content] } }
+ context "with multiple columns" do
+ let(:options) { {only: %i[name content]} }
- it 'concatenates when multiples columns are selected' do
+ it "concatenates when multiples columns are selected" do
expect(feature.conditions.to_sql).to eq("('#{query}' % (#{coalesced_columns}))")
end
end
end
end
- describe '#rank' do
- it 'returns an expression using the similarity() function' do
+ describe "#rank" do
+ it "returns an expression using the similarity() function" do
expect(feature.rank.to_sql).to eq("(similarity('#{query}', (#{coalesced_columns})))")
end
end
diff --git a/spec/lib/pg_search/features/tsearch_spec.rb b/spec/lib/pg_search/features/tsearch_spec.rb
index d702d69c..3ce34077 100644
--- a/spec/lib/pg_search/features/tsearch_spec.rb
+++ b/spec/lib/pg_search/features/tsearch_spec.rb
@@ -24,7 +24,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.rank.to_sql).to eq(
- %{(ts_rank((to_tsvector('simple', coalesce(#{Model.quoted_table_name}."name"::text, '')) || to_tsvector('simple', coalesce(#{Model.quoted_table_name}."content"::text, ''))), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 0))}
+ %{(ts_rank((to_tsvector('simple', coalesce((#{Model.quoted_table_name}."name")::text, '')) || to_tsvector('simple', coalesce((#{Model.quoted_table_name}."content")::text, ''))), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 0))}
)
end
@@ -35,7 +35,7 @@
PgSearch::Configuration::Column.new(:name, nil, Model),
PgSearch::Configuration::Column.new(:content, nil, Model)
]
- options = { tsvector_column: :my_tsvector, normalization: 2 }
+ options = {tsvector_column: :my_tsvector, normalization: 2}
config = instance_double(PgSearch::Configuration, :config, ignore: [])
normalizer = PgSearch::Normalizer.new(config)
@@ -67,7 +67,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %{((to_tsvector('simple', coalesce(#{Model.quoted_table_name}."name"::text, '')) || to_tsvector('simple', coalesce(#{Model.quoted_table_name}."content"::text, ''))) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
+ %{((to_tsvector('simple', coalesce((#{Model.quoted_table_name}."name")::text, '')) || to_tsvector('simple', coalesce((#{Model.quoted_table_name}."content")::text, ''))) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
)
end
@@ -78,13 +78,13 @@
PgSearch::Configuration::Column.new(:name, nil, Model),
PgSearch::Configuration::Column.new(:content, nil, Model)
]
- options = { negation: true }
+ options = {negation: true}
config = instance_double(PgSearch::Configuration, :config, ignore: [])
normalizer = PgSearch::Normalizer.new(config)
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %{((to_tsvector('simple', coalesce(#{Model.quoted_table_name}."name"::text, '')) || to_tsvector('simple', coalesce(#{Model.quoted_table_name}."content"::text, ''))) @@ (to_tsquery('simple', '!' || ''' ' || 'query' || ' ''')))}
+ %{((to_tsvector('simple', coalesce((#{Model.quoted_table_name}."name")::text, '')) || to_tsvector('simple', coalesce((#{Model.quoted_table_name}."content")::text, ''))) @@ (to_tsquery('simple', '!' || ''' ' || 'query' || ' ''')))}
)
end
end
@@ -96,25 +96,25 @@
PgSearch::Configuration::Column.new(:name, nil, Model),
PgSearch::Configuration::Column.new(:content, nil, Model)
]
- options = { negation: false }
+ options = {negation: false}
config = instance_double(PgSearch::Configuration, :config, ignore: [])
normalizer = PgSearch::Normalizer.new(config)
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %{((to_tsvector('simple', coalesce(#{Model.quoted_table_name}."name"::text, '')) || to_tsvector('simple', coalesce(#{Model.quoted_table_name}."content"::text, ''))) @@ (to_tsquery('simple', ''' ' || '!query' || ' ''')))}
+ %{((to_tsvector('simple', coalesce((#{Model.quoted_table_name}."name")::text, '')) || to_tsvector('simple', coalesce((#{Model.quoted_table_name}."content")::text, ''))) @@ (to_tsquery('simple', ''' ' || '!query' || ' ''')))}
)
end
end
context "when options[:tsvector_column] is a string" do
- it 'uses the tsvector column' do
+ it "uses the tsvector column" do
query = "query"
columns = [
PgSearch::Configuration::Column.new(:name, nil, Model),
PgSearch::Configuration::Column.new(:content, nil, Model)
]
- options = { tsvector_column: "my_tsvector" }
+ options = {tsvector_column: "my_tsvector"}
config = instance_double(PgSearch::Configuration, :config, ignore: [])
normalizer = PgSearch::Normalizer.new(config)
@@ -126,13 +126,13 @@
end
context "when options[:tsvector_column] is an array of strings" do
- it 'uses the tsvector column' do
+ it "uses the tsvector column" do
query = "query"
columns = [
PgSearch::Configuration::Column.new(:name, nil, Model),
PgSearch::Configuration::Column.new(:content, nil, Model)
]
- options = { tsvector_column: ["tsvector1", "tsvector2"] }
+ options = {tsvector_column: ["tsvector1", "tsvector2"]}
config = instance_double(PgSearch::Configuration, :config, ignore: [])
normalizer = PgSearch::Normalizer.new(config)
@@ -155,7 +155,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %Q{((coalesce(#{Model.quoted_table_name}.\"my_tsvector\", '')) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
+ %Q{((coalesce((#{Model.quoted_table_name}.\"my_tsvector\")::tsvector, '')) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
)
end
end
@@ -181,13 +181,13 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.highlight.to_sql).to eq(
- "(ts_headline('simple', (coalesce(#{Model.quoted_table_name}.\"name\"::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), ''))"
+ "(ts_headline('simple', (coalesce((#{Model.quoted_table_name}.\"name\")::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), ''))"
)
end
context "when options[:dictionary] is passed" do
# rubocop:disable RSpec/ExampleLength
- it 'uses the provided dictionary' do
+ it "uses the provided dictionary" do
query = "query"
columns = [
PgSearch::Configuration::Column.new(:name, nil, Model),
@@ -206,7 +206,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
- expected_sql = %{(ts_headline('spanish', (coalesce(#{Model.quoted_table_name}."name"::text, '') || ' ' || coalesce(#{Model.quoted_table_name}."content"::text, '')), (to_tsquery('spanish', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = ""'))}
+ expected_sql = %{(ts_headline('spanish', (coalesce((#{Model.quoted_table_name}."name")::text, '') || ' ' || coalesce((#{Model.quoted_table_name}."content")::text, '')), (to_tsquery('spanish', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = ""'))}
expect(feature.highlight.to_sql).to eq(expected_sql)
end
@@ -223,13 +223,13 @@
options = {
highlight: {
StartSel: '',
- StopSel: '',
+ StopSel: "",
MaxWords: 123,
MinWords: 456,
ShortWord: 4,
HighlightAll: true,
MaxFragments: 3,
- FragmentDelimiter: '…'
+ FragmentDelimiter: "…"
}
}
@@ -238,7 +238,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
- expected_sql = %{(ts_headline('simple', (coalesce(#{Model.quoted_table_name}."name"::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = "", MaxFragments = 3, MaxWords = 123, MinWords = 456, ShortWord = 4, FragmentDelimiter = "…", HighlightAll = TRUE'))}
+ expected_sql = %{(ts_headline('simple', (coalesce((#{Model.quoted_table_name}."name")::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = "", MaxFragments = 3, MaxWords = 123, MinWords = 456, ShortWord = 4, FragmentDelimiter = "…", HighlightAll = TRUE'))}
expect(feature.highlight.to_sql).to eq(expected_sql)
end
@@ -253,13 +253,13 @@
options = {
highlight: {
start_sel: '',
- stop_sel: '',
+ stop_sel: "",
max_words: 123,
min_words: 456,
short_word: 4,
highlight_all: false,
max_fragments: 3,
- fragment_delimiter: '…'
+ fragment_delimiter: "…"
}
}
@@ -269,7 +269,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
highlight_sql = ActiveSupport::Deprecation.silence { feature.highlight.to_sql }
- expected_sql = %{(ts_headline('simple', (coalesce(#{Model.quoted_table_name}."name"::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = "", MaxFragments = 3, MaxWords = 123, MinWords = 456, ShortWord = 4, FragmentDelimiter = "…", HighlightAll = FALSE'))}
+ expected_sql = %{(ts_headline('simple', (coalesce((#{Model.quoted_table_name}."name")::text, '')), (to_tsquery('simple', ''' ' || 'query' || ' ''')), 'StartSel = "", StopSel = "", MaxFragments = 3, MaxWords = 123, MinWords = 456, ShortWord = 4, FragmentDelimiter = "…", HighlightAll = FALSE'))}
expect(highlight_sql).to eq(expected_sql)
end
@@ -288,7 +288,7 @@
feature = described_class.new(query, options, columns, Model, normalizer)
expect(feature.conditions.to_sql).to eq(
- %{((coalesce(#{Model.quoted_table_name}.\"my_tsvector\", '')) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
+ %{((coalesce((#{Model.quoted_table_name}.\"my_tsvector\")::tsvector, '')) @@ (to_tsquery('simple', ''' ' || 'query' || ' ''')))}
)
end
end
diff --git a/spec/lib/pg_search/multisearch/rebuilder_spec.rb b/spec/lib/pg_search/multisearch/rebuilder_spec.rb
index 5a462644..58a48c02 100644
--- a/spec/lib/pg_search/multisearch/rebuilder_spec.rb
+++ b/spec/lib/pg_search/multisearch/rebuilder_spec.rb
@@ -6,10 +6,10 @@
describe PgSearch::Multisearch::Rebuilder do
with_table "pg_search_documents", &DOCUMENTS_SCHEMA
- describe 'when initialized with a model that is not multisearchable' do
+ describe "when initialized with a model that is not multisearchable" do
with_model :not_multisearchable
- it 'raises an exception' do
+ it "raises an exception" do
expect {
described_class.new(NotMultisearchable)
}.to raise_exception(
@@ -253,7 +253,7 @@ def foo
model do
include PgSearch::Model
multisearchable against: :name,
- additional_attributes: ->(obj) { { additional_attribute_column: "#{obj.class}::#{obj.id}" } }
+ additional_attributes: ->(obj) { {additional_attribute_column: "#{obj.class}::#{obj.id}"} }
end
end
diff --git a/spec/lib/pg_search/multisearchable_spec.rb b/spec/lib/pg_search/multisearchable_spec.rb
index c0814e58..c155741b 100644
--- a/spec/lib/pg_search/multisearchable_spec.rb
+++ b/spec/lib/pg_search/multisearchable_spec.rb
@@ -149,7 +149,7 @@
end
context "when searching against a single column" do
- let(:multisearchable_options) { { against: :some_content } }
+ let(:multisearchable_options) { {against: :some_content} }
let(:text) { "foo bar" }
before do
@@ -159,7 +159,7 @@
record.save
end
- describe '#content' do
+ describe "#content" do
subject { super().pg_search_document.content }
it { is_expected.to eq(text) }
@@ -167,17 +167,17 @@
end
context "when searching against multiple columns" do
- let(:multisearchable_options) { { against: %i[attr_1 attr_2] } }
+ let(:multisearchable_options) { {against: %i[attr_1 attr_2]} }
before do
without_partial_double_verification do
- allow(record).to receive(:attr_1).and_return('1')
- allow(record).to receive(:attr_2).and_return('2')
+ allow(record).to receive(:attr_1).and_return("1")
+ allow(record).to receive(:attr_2).and_return("2")
end
record.save
end
- describe '#content' do
+ describe "#content" do
subject { super().pg_search_document.content }
it { is_expected.to eq("1 2") }
@@ -195,7 +195,7 @@
end
context "when searching against a single column" do
- let(:multisearchable_options) { { against: :some_content } }
+ let(:multisearchable_options) { {against: :some_content} }
let(:text) { "foo bar" }
before do
@@ -205,7 +205,7 @@
record.save
end
- describe '#content' do
+ describe "#content" do
subject { super().pg_search_document.content }
it { is_expected.to eq(text) }
@@ -213,17 +213,17 @@
end
context "when searching against multiple columns" do
- let(:multisearchable_options) { { against: %i[attr_1 attr_2] } }
+ let(:multisearchable_options) { {against: %i[attr_1 attr_2]} }
before do
without_partial_double_verification do
- allow(record).to receive(:attr_1).and_return('1')
- allow(record).to receive(:attr_2).and_return('2')
+ allow(record).to receive(:attr_1).and_return("1")
+ allow(record).to receive(:attr_2).and_return("2")
end
record.save
end
- describe '#content' do
+ describe "#content" do
subject { super().pg_search_document.content }
it { is_expected.to eq("1 2") }
@@ -234,7 +234,7 @@
let(:multisearchable_options) do
{
additional_attributes: lambda do |record|
- { foo: record.bar }
+ {foo: record.bar}
end
}
end
@@ -247,7 +247,7 @@
record.save
expect(record)
.to have_received(:create_pg_search_document)
- .with(content: '', foo: text)
+ .with(content: "", foo: text)
end
end
end
@@ -269,7 +269,7 @@
record.save
expect(record)
.to have_received(:create_pg_search_document)
- .with(content: '')
+ .with(content: "")
end
end
diff --git a/spec/lib/pg_search_spec.rb b/spec/lib/pg_search_spec.rb
index 80cca230..6a178f2d 100644
--- a/spec/lib/pg_search_spec.rb
+++ b/spec/lib/pg_search_spec.rb
@@ -49,7 +49,7 @@ def clear_searchable_cache
end
end
- let!(:soundalike_record) { MultisearchableModel.create!(title: 'foning') }
+ let!(:soundalike_record) { MultisearchableModel.create!(title: "foning") }
let(:query) { "Phoning" }
it { is_expected.to include(soundalike_record) }
@@ -65,9 +65,9 @@ def clear_searchable_cache
allow(described_class).to receive(:multisearch_options) do
lambda do |query, soundalike|
if soundalike
- { using: :dmetaphone, query: query }
+ {using: :dmetaphone, query: query}
else
- { query: query }
+ {query: query}
end
end
end
@@ -83,7 +83,7 @@ def clear_searchable_cache
end
end
- let!(:soundalike_record) { MultisearchableModel.create!(title: 'foning') }
+ let!(:soundalike_record) { MultisearchableModel.create!(title: "foning") }
let(:query) { "Phoning" }
context "with soundalike true" do
@@ -103,8 +103,8 @@ def clear_searchable_cache
context "with standard type column" do
with_model :SuperclassModel do
table do |t|
- t.text 'content'
- t.string 'type'
+ t.text "content"
+ t.string "type"
end
end
@@ -191,12 +191,12 @@ def clear_searchable_cache
context "with custom type column" do
with_model :SuperclassModel do
table do |t|
- t.text 'content'
- t.string 'inherit'
+ t.text "content"
+ t.string "inherit"
end
model do
- self.inheritance_column = 'inherit'
+ self.inheritance_column = "inherit"
end
end
@@ -254,7 +254,7 @@ def clear_searchable_cache
multisearch_enabled_inside = described_class.multisearch_enabled?
raise
end
- rescue StandardError
+ rescue
end
multisearch_enabled_after = described_class.multisearch_enabled?
diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb
index e25191aa..0498959b 100644
--- a/spec/spec_helper.rb
+++ b/spec/spec_helper.rb
@@ -1,19 +1,19 @@
# frozen_string_literal: true
-require 'warning'
+require "warning"
# Ignore Ruby 2.7 warnings from Active Record
Warning.ignore :keyword_separation
# https://github.com/grodowski/undercover#setting-up-required-lcov-reporting
-require 'simplecov'
-require 'simplecov-lcov'
+require "simplecov"
+require "simplecov-lcov"
SimpleCov::Formatter::LcovFormatter.config.report_with_single_file = true
SimpleCov.formatter = SimpleCov::Formatter::LcovFormatter
SimpleCov.start do
add_filter(%r{^/spec/})
enable_coverage(:branch)
end
-require 'undercover'
+require "undercover"
require "bundler/setup"
require "pg_search"
@@ -29,11 +29,11 @@
mocks.verify_partial_doubles = true
end
- config.example_status_persistence_file_path = 'tmp/examples.txt'
+ config.example_status_persistence_file_path = "tmp/examples.txt"
end
-require 'support/database'
-require 'support/with_model'
+require "support/database"
+require "support/with_model"
DOCUMENTS_SCHEMA = lambda do |t|
t.belongs_to :searchable, polymorphic: true, index: true
diff --git a/spec/support/database.rb b/spec/support/database.rb
index 11d3ef00..23ada4e5 100644
--- a/spec/support/database.rb
+++ b/spec/support/database.rb
@@ -10,10 +10,10 @@
end
begin
- connection_options = { adapter: 'postgresql', database: 'pg_search_test', min_messages: 'warning' }
+ connection_options = {adapter: "postgresql", database: "pg_search_test", min_messages: "warning"}
if ENV["CI"]
- connection_options[:username] = 'postgres'
- connection_options[:password] = 'postgres'
+ connection_options[:username] = "postgres"
+ connection_options[:password] = "postgres"
end
ActiveRecord::Base.establish_connection(connection_options)
connection = ActiveRecord::Base.connection
@@ -40,7 +40,7 @@ def install_extension(name)
return unless extension.none?
connection.execute "CREATE EXTENSION #{name};"
-rescue StandardError => e
+rescue => e
at_exit do
puts "-" * 80
puts "Please install the #{name} extension"
@@ -52,7 +52,7 @@ def install_extension(name)
def install_extension_if_missing(name, query, expected_result)
result = ActiveRecord::Base.connection.select_value(query)
raise "Unexpected output for #{query}: #{result.inspect}" unless result.casecmp(expected_result).zero?
-rescue StandardError
+rescue
install_extension(name)
end
@@ -62,10 +62,9 @@ def install_extension_if_missing(name, query, expected_result)
def load_sql(filename)
connection = ActiveRecord::Base.connection
- file_contents = File.read(File.join(File.dirname(__FILE__), '..', '..', 'sql', filename))
+ file_contents = File.read(File.join(File.dirname(__FILE__), "..", "..", "sql", filename))
connection.execute(file_contents)
end
load_sql("dmetaphone.sql")
-
-load_sql("tsvector_agg.sql") unless connection.select_value("SELECT 1 FROM pg_catalog.pg_aggregate WHERE aggfnoid = 'tsvector_agg'::REGPROC") == "1"
+load_sql("tsvector_agg.sql")
diff --git a/sql/tsvector_agg.sql b/sql/tsvector_agg.sql
index 6848d975..c5221685 100644
--- a/sql/tsvector_agg.sql
+++ b/sql/tsvector_agg.sql
@@ -4,7 +4,7 @@ $function$ BEGIN
END; $function$
LANGUAGE plpgsql;
-CREATE AGGREGATE tsvector_agg(tsvector) (
+CREATE OR REPLACE AGGREGATE tsvector_agg(tsvector) (
SFUNC=concat_tsvectors,
STYPE=tsvector,
INITCOND=''