textacular 5.5.1 → 5.7.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +9 -0
- data/Gemfile +1 -0
- data/README.md +0 -5
- data/Rakefile +25 -14
- data/lib/textacular/rails.rb +3 -1
- data/lib/textacular/version.rb +1 -1
- metadata +37 -57
- data/spec/config.github.yml +0 -8
- data/spec/config.yml.example +0 -5
- data/spec/spec_helper.rb +0 -104
- data/spec/support/ar_stand_in.rb +0 -4
- data/spec/support/character.rb +0 -7
- data/spec/support/game.rb +0 -5
- data/spec/support/game_extended_with_textacular.rb +0 -5
- data/spec/support/game_extended_with_textacular_and_custom_language.rb +0 -7
- data/spec/support/game_fail.rb +0 -3
- data/spec/support/game_fail_extended_with_textacular.rb +0 -5
- data/spec/support/not_there.rb +0 -3
- data/spec/support/textacular_web_comic.rb +0 -7
- data/spec/support/web_comic.rb +0 -7
- data/spec/support/web_comic_with_searchable.rb +0 -6
- data/spec/support/web_comic_with_searchable_name.rb +0 -6
- data/spec/support/web_comic_with_searchable_name_and_author.rb +0 -6
- data/spec/textacular/full_text_indexer_spec.rb +0 -69
- data/spec/textacular/migration_generator_spec.rb +0 -67
- data/spec/textacular/searchable_spec.rb +0 -244
- data/spec/textacular/trigram_installer_spec.rb +0 -24
- data/spec/textacular_spec.rb +0 -294
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a50c0b7cd47b58a8bc97486e1cb01ca2f2f987c0d3634180ebb02c521585fdf0
|
4
|
+
data.tar.gz: c6b71a207530b8bed1ff5b261b59a90493fde59a295e3f8c13585f0f1e551e8f
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 7ceb9851288d634618d85ab654b5faec8f3957e13a1bf5b93191107911bdcc536d4abe164fb4f2323cdd393a07f54c2789ee7bb7df0d6744a6af7e6ca984c158
|
7
|
+
data.tar.gz: f33e73a565c3f84fdb9184c8143857707cb8b7dc46a88b6fc3b6c916f0d9123e1d625a4b7eab00ed35352dbeb9c390d6839b1dea8f028b267c2bc381d691f0d3
|
data/CHANGELOG.md
CHANGED
@@ -2,6 +2,15 @@
|
|
2
2
|
|
3
3
|
## Unreleased
|
4
4
|
|
5
|
+
## 5.7.0
|
6
|
+
|
7
|
+
* ActiveRecord 7.2 compatibility
|
8
|
+
* Defer extending ActiveRecord::Base until Active Record is loaded
|
9
|
+
|
10
|
+
## 5.6.0
|
11
|
+
|
12
|
+
* ActiveRecord 7.1 compatibility
|
13
|
+
|
5
14
|
## 5.5.1
|
6
15
|
|
7
16
|
* Revert breaking change to fuzzy_search introduced in 49f09b389d2f2b18cbe9de565b7ac5fcac14d7c6
|
data/Gemfile
CHANGED
data/README.md
CHANGED
@@ -1,13 +1,8 @@
|
|
1
1
|
# textacular
|
2
2
|
[![Gem Version](http://img.shields.io/gem/v/textacular.svg)][rubygems]
|
3
3
|
[![Build Status](https://github.com/textacular/textacular/actions/workflows/main.yml/badge.svg)](https://github.com/textacular/textacular/actions/workflows/main.yml)
|
4
|
-
[![Code Climate](https://img.shields.io/codeclimate/github/textacular/textacular.svg)][codeclimate]
|
5
4
|
|
6
5
|
[rubygems]: http://rubygems.org/gems/textacular
|
7
|
-
[codeclimate]: https://codeclimate.com/github/textacular/textacular
|
8
|
-
|
9
|
-
Further documentation available at http://textacular.github.com/textacular.
|
10
|
-
|
11
6
|
|
12
7
|
## DESCRIPTION:
|
13
8
|
|
data/Rakefile
CHANGED
@@ -6,8 +6,15 @@ require 'pry'
|
|
6
6
|
require 'rspec/core/rake_task'
|
7
7
|
|
8
8
|
RSpec::Core::RakeTask.new(:spec)
|
9
|
+
require "rake/testtask"
|
9
10
|
|
10
|
-
|
11
|
+
Rake::TestTask.new(:test) do |t|
|
12
|
+
t.libs << "test"
|
13
|
+
t.libs << "lib"
|
14
|
+
t.test_files = FileList["test/**/*_test.rb"]
|
15
|
+
end
|
16
|
+
|
17
|
+
task :default => :test
|
11
18
|
|
12
19
|
file 'spec/config.yml' do |t|
|
13
20
|
sh 'erb spec/config.yml.example > spec/config.yml'
|
@@ -20,9 +27,9 @@ end
|
|
20
27
|
|
21
28
|
namespace :db do
|
22
29
|
|
23
|
-
task :connect => '
|
30
|
+
task :connect => 'test/config.yml' do |t|
|
24
31
|
ActiveRecord::Base.establish_connection \
|
25
|
-
YAML.load_file '
|
32
|
+
YAML.load_file 'test/config.yml'
|
26
33
|
end
|
27
34
|
|
28
35
|
task :disconnect do
|
@@ -43,28 +50,32 @@ namespace :db do
|
|
43
50
|
|
44
51
|
desc 'Run the test database migrations'
|
45
52
|
task :up => :'db:connect' do
|
46
|
-
if ActiveRecord.version >= Gem::Version.new('
|
47
|
-
|
48
|
-
|
49
|
-
|
53
|
+
if ActiveRecord.version >= Gem::Version.new('7.2.0')
|
54
|
+
ActiveRecord::Base.connection_pool.migration_context.up
|
55
|
+
elsif ActiveRecord.version >= Gem::Version.new('6.0.0')
|
56
|
+
ActiveRecord::Migration.new.migration_context.up
|
50
57
|
elsif ActiveRecord.version >= Gem::Version.new('5.2')
|
51
58
|
migrations = ActiveRecord::Migration.new.migration_context.migrations
|
52
|
-
|
59
|
+
ActiveRecord::Migrator.new(:up, migrations, nil).migrate
|
53
60
|
else
|
54
61
|
migrations = ActiveRecord::Migrator.migrations('db/migrate')
|
55
|
-
|
62
|
+
ActiveRecord::Migrator.new(:up, migrations, nil).migrate
|
56
63
|
end
|
57
|
-
ActiveRecord::Migrator.new(:up, migrations, schema_migration).migrate
|
58
64
|
end
|
59
65
|
|
60
66
|
desc 'Reverse the test database migrations'
|
61
67
|
task :down => :'db:connect' do
|
62
|
-
|
63
|
-
ActiveRecord::
|
68
|
+
if ActiveRecord.version >= Gem::Version.new('7.2.0')
|
69
|
+
ActiveRecord::Base.connection_pool.migration_context.down
|
70
|
+
elsif ActiveRecord.version >= Gem::Version.new('6.0.0')
|
71
|
+
ActiveRecord::Migration.new.migration_context.down
|
72
|
+
elsif ActiveRecord.version >= Gem::Version.new('5.2')
|
73
|
+
migrations = ActiveRecord::Migration.new.migration_context.migrations
|
74
|
+
ActiveRecord::Migrator.new(:down, migrations, nil).migrate
|
64
75
|
else
|
65
|
-
ActiveRecord::Migrator.migrations('db/migrate')
|
76
|
+
migrations = ActiveRecord::Migrator.migrations('db/migrate')
|
77
|
+
ActiveRecord::Migrator.new(:down, migrations, nil).migrate
|
66
78
|
end
|
67
|
-
ActiveRecord::Migrator.new(:down, migrations, nil).migrate
|
68
79
|
end
|
69
80
|
end
|
70
81
|
task :migrate => :'migrate:up'
|
data/lib/textacular/rails.rb
CHANGED
@@ -4,7 +4,9 @@ require File.expand_path(File.dirname(__FILE__) + '/../textacular')
|
|
4
4
|
module Textacular
|
5
5
|
class Railtie < Rails::Railtie
|
6
6
|
initializer "textacular.configure_rails_initialization" do
|
7
|
-
|
7
|
+
ActiveSupport.on_load(:active_record) do
|
8
|
+
ActiveRecord::Base.extend(Textacular)
|
9
|
+
end
|
8
10
|
end
|
9
11
|
|
10
12
|
rake_tasks do
|
data/lib/textacular/version.rb
CHANGED
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: textacular
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 5.
|
4
|
+
version: 5.7.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Ben Hamill
|
@@ -11,10 +11,10 @@ authors:
|
|
11
11
|
autorequire:
|
12
12
|
bindir: bin
|
13
13
|
cert_chain: []
|
14
|
-
date:
|
14
|
+
date: 2024-08-16 00:00:00.000000000 Z
|
15
15
|
dependencies:
|
16
16
|
- !ruby/object:Gem::Dependency
|
17
|
-
name:
|
17
|
+
name: activesupport
|
18
18
|
requirement: !ruby/object:Gem::Requirement
|
19
19
|
requirements:
|
20
20
|
- - ">="
|
@@ -28,7 +28,7 @@ dependencies:
|
|
28
28
|
- !ruby/object:Gem::Version
|
29
29
|
version: '0'
|
30
30
|
- !ruby/object:Gem::Dependency
|
31
|
-
name:
|
31
|
+
name: byebug
|
32
32
|
requirement: !ruby/object:Gem::Requirement
|
33
33
|
requirements:
|
34
34
|
- - ">="
|
@@ -42,7 +42,7 @@ dependencies:
|
|
42
42
|
- !ruby/object:Gem::Version
|
43
43
|
version: '0'
|
44
44
|
- !ruby/object:Gem::Dependency
|
45
|
-
name: database_cleaner
|
45
|
+
name: database_cleaner-active_record
|
46
46
|
requirement: !ruby/object:Gem::Requirement
|
47
47
|
requirements:
|
48
48
|
- - ">="
|
@@ -56,7 +56,21 @@ dependencies:
|
|
56
56
|
- !ruby/object:Gem::Version
|
57
57
|
version: '0'
|
58
58
|
- !ruby/object:Gem::Dependency
|
59
|
-
name:
|
59
|
+
name: minitest
|
60
|
+
requirement: !ruby/object:Gem::Requirement
|
61
|
+
requirements:
|
62
|
+
- - ">="
|
63
|
+
- !ruby/object:Gem::Version
|
64
|
+
version: '0'
|
65
|
+
type: :development
|
66
|
+
prerelease: false
|
67
|
+
version_requirements: !ruby/object:Gem::Requirement
|
68
|
+
requirements:
|
69
|
+
- - ">="
|
70
|
+
- !ruby/object:Gem::Version
|
71
|
+
version: '0'
|
72
|
+
- !ruby/object:Gem::Dependency
|
73
|
+
name: pg
|
60
74
|
requirement: !ruby/object:Gem::Requirement
|
61
75
|
requirements:
|
62
76
|
- - ">="
|
@@ -98,7 +112,21 @@ dependencies:
|
|
98
112
|
- !ruby/object:Gem::Version
|
99
113
|
version: '0'
|
100
114
|
- !ruby/object:Gem::Dependency
|
101
|
-
name:
|
115
|
+
name: rake
|
116
|
+
requirement: !ruby/object:Gem::Requirement
|
117
|
+
requirements:
|
118
|
+
- - ">="
|
119
|
+
- !ruby/object:Gem::Version
|
120
|
+
version: '0'
|
121
|
+
type: :development
|
122
|
+
prerelease: false
|
123
|
+
version_requirements: !ruby/object:Gem::Requirement
|
124
|
+
requirements:
|
125
|
+
- - ">="
|
126
|
+
- !ruby/object:Gem::Version
|
127
|
+
version: '0'
|
128
|
+
- !ruby/object:Gem::Dependency
|
129
|
+
name: rspec
|
102
130
|
requirement: !ruby/object:Gem::Requirement
|
103
131
|
requirements:
|
104
132
|
- - ">="
|
@@ -118,9 +146,6 @@ dependencies:
|
|
118
146
|
- - ">="
|
119
147
|
- !ruby/object:Gem::Version
|
120
148
|
version: '5.0'
|
121
|
-
- - "<"
|
122
|
-
- !ruby/object:Gem::Version
|
123
|
-
version: '7.1'
|
124
149
|
type: :runtime
|
125
150
|
prerelease: false
|
126
151
|
version_requirements: !ruby/object:Gem::Requirement
|
@@ -128,9 +153,6 @@ dependencies:
|
|
128
153
|
- - ">="
|
129
154
|
- !ruby/object:Gem::Version
|
130
155
|
version: '5.0'
|
131
|
-
- - "<"
|
132
|
-
- !ruby/object:Gem::Version
|
133
|
-
version: '7.1'
|
134
156
|
description: |-
|
135
157
|
Textacular exposes full text search capabilities from PostgreSQL, extending
|
136
158
|
ActiveRecord with scopes making search easy and fun!
|
@@ -154,27 +176,6 @@ files:
|
|
154
176
|
- lib/textacular/tasks.rb
|
155
177
|
- lib/textacular/trigram_installer.rb
|
156
178
|
- lib/textacular/version.rb
|
157
|
-
- spec/config.github.yml
|
158
|
-
- spec/config.yml.example
|
159
|
-
- spec/spec_helper.rb
|
160
|
-
- spec/support/ar_stand_in.rb
|
161
|
-
- spec/support/character.rb
|
162
|
-
- spec/support/game.rb
|
163
|
-
- spec/support/game_extended_with_textacular.rb
|
164
|
-
- spec/support/game_extended_with_textacular_and_custom_language.rb
|
165
|
-
- spec/support/game_fail.rb
|
166
|
-
- spec/support/game_fail_extended_with_textacular.rb
|
167
|
-
- spec/support/not_there.rb
|
168
|
-
- spec/support/textacular_web_comic.rb
|
169
|
-
- spec/support/web_comic.rb
|
170
|
-
- spec/support/web_comic_with_searchable.rb
|
171
|
-
- spec/support/web_comic_with_searchable_name.rb
|
172
|
-
- spec/support/web_comic_with_searchable_name_and_author.rb
|
173
|
-
- spec/textacular/full_text_indexer_spec.rb
|
174
|
-
- spec/textacular/migration_generator_spec.rb
|
175
|
-
- spec/textacular/searchable_spec.rb
|
176
|
-
- spec/textacular/trigram_installer_spec.rb
|
177
|
-
- spec/textacular_spec.rb
|
178
179
|
homepage: http://textacular.github.com/textacular
|
179
180
|
licenses:
|
180
181
|
- MIT
|
@@ -194,29 +195,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
194
195
|
- !ruby/object:Gem::Version
|
195
196
|
version: '0'
|
196
197
|
requirements: []
|
197
|
-
rubygems_version: 3.
|
198
|
+
rubygems_version: 3.5.3
|
198
199
|
signing_key:
|
199
200
|
specification_version: 4
|
200
201
|
summary: Textacular exposes full text search capabilities from PostgreSQL
|
201
|
-
test_files:
|
202
|
-
- spec/config.yml.example
|
203
|
-
- spec/config.github.yml
|
204
|
-
- spec/spec_helper.rb
|
205
|
-
- spec/support/ar_stand_in.rb
|
206
|
-
- spec/support/character.rb
|
207
|
-
- spec/support/game.rb
|
208
|
-
- spec/support/game_extended_with_textacular.rb
|
209
|
-
- spec/support/game_extended_with_textacular_and_custom_language.rb
|
210
|
-
- spec/support/game_fail.rb
|
211
|
-
- spec/support/game_fail_extended_with_textacular.rb
|
212
|
-
- spec/support/not_there.rb
|
213
|
-
- spec/support/textacular_web_comic.rb
|
214
|
-
- spec/support/web_comic.rb
|
215
|
-
- spec/support/web_comic_with_searchable.rb
|
216
|
-
- spec/support/web_comic_with_searchable_name.rb
|
217
|
-
- spec/support/web_comic_with_searchable_name_and_author.rb
|
218
|
-
- spec/textacular_spec.rb
|
219
|
-
- spec/textacular/full_text_indexer_spec.rb
|
220
|
-
- spec/textacular/migration_generator_spec.rb
|
221
|
-
- spec/textacular/searchable_spec.rb
|
222
|
-
- spec/textacular/trigram_installer_spec.rb
|
202
|
+
test_files: []
|
data/spec/config.github.yml
DELETED
data/spec/config.yml.example
DELETED
data/spec/spec_helper.rb
DELETED
@@ -1,104 +0,0 @@
|
|
1
|
-
require 'pry'
|
2
|
-
require 'textacular'
|
3
|
-
require 'database_cleaner'
|
4
|
-
require 'yaml'
|
5
|
-
|
6
|
-
config = YAML.load_file File.expand_path(File.dirname(__FILE__) + '/config.yml')
|
7
|
-
ActiveRecord::Base.establish_connection config.merge(:adapter => :postgresql)
|
8
|
-
|
9
|
-
# This file was generated by the `rspec --init` command. Conventionally, all
|
10
|
-
# specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`.
|
11
|
-
# The generated `.rspec` file contains `--require spec_helper` which will cause this
|
12
|
-
# file to always be loaded, without a need to explicitly require it in any files.
|
13
|
-
#
|
14
|
-
# Given that it is always loaded, you are encouraged to keep this file as
|
15
|
-
# light-weight as possible. Requiring heavyweight dependencies from this file
|
16
|
-
# will add to the boot time of your test suite on EVERY test run, even for an
|
17
|
-
# individual file that may not need all of that loaded. Instead, consider making
|
18
|
-
# a separate helper file that requires the additional dependencies and performs
|
19
|
-
# the additional setup, and require it from the spec files that actually need it.
|
20
|
-
#
|
21
|
-
# The `.rspec` file also contains a few flags that are not defaults but that
|
22
|
-
# users commonly want.
|
23
|
-
#
|
24
|
-
# See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration
|
25
|
-
RSpec.configure do |config|
|
26
|
-
# rspec-expectations config goes here. You can use an alternate
|
27
|
-
# assertion/expectation library such as wrong or the stdlib/minitest
|
28
|
-
# assertions if you prefer.
|
29
|
-
config.expect_with :rspec do |expectations|
|
30
|
-
# This option will default to `true` in RSpec 4. It makes the `description`
|
31
|
-
# and `failure_message` of custom matchers include text for helper methods
|
32
|
-
# defined using `chain`, e.g.:
|
33
|
-
# be_bigger_than(2).and_smaller_than(4).description
|
34
|
-
# # => "be bigger than 2 and smaller than 4"
|
35
|
-
# ...rather than:
|
36
|
-
# # => "be bigger than 2"
|
37
|
-
expectations.include_chain_clauses_in_custom_matcher_descriptions = true
|
38
|
-
end
|
39
|
-
|
40
|
-
# rspec-mocks config goes here. You can use an alternate test double
|
41
|
-
# library (such as bogus or mocha) by changing the `mock_with` option here.
|
42
|
-
config.mock_with :rspec do |mocks|
|
43
|
-
# Prevents you from mocking or stubbing a method that does not exist on
|
44
|
-
# a real object. This is generally recommended, and will default to
|
45
|
-
# `true` in RSpec 4.
|
46
|
-
mocks.verify_partial_doubles = true
|
47
|
-
end
|
48
|
-
|
49
|
-
# These two settings work together to allow you to limit a spec run
|
50
|
-
# to individual examples or groups you care about by tagging them with
|
51
|
-
# `:focus` metadata. When nothing is tagged with `:focus`, all examples
|
52
|
-
# get run.
|
53
|
-
config.filter_run :focus
|
54
|
-
config.run_all_when_everything_filtered = true
|
55
|
-
|
56
|
-
# Limits the available syntax to the non-monkey patched syntax that is recommended.
|
57
|
-
# For more details, see:
|
58
|
-
# - http://myronmars.to/n/dev-blog/2012/06/rspecs-new-expectation-syntax
|
59
|
-
# - http://teaisaweso.me/blog/2013/05/27/rspecs-new-message-expectation-syntax/
|
60
|
-
# - http://myronmars.to/n/dev-blog/2014/05/notable-changes-in-rspec-3#new__config_option_to_disable_rspeccore_monkey_patching
|
61
|
-
config.disable_monkey_patching!
|
62
|
-
|
63
|
-
# This setting enables warnings. It's recommended, but in some cases may
|
64
|
-
# be too noisy due to issues in dependencies.
|
65
|
-
config.warnings = true
|
66
|
-
|
67
|
-
# Many RSpec users commonly either run the entire suite or an individual
|
68
|
-
# file, and it's useful to allow more verbose output when running an
|
69
|
-
# individual spec file.
|
70
|
-
# if config.files_to_run.one?
|
71
|
-
# Use the documentation formatter for detailed output,
|
72
|
-
# unless a formatter has already been configured
|
73
|
-
# (e.g. via a command-line flag).
|
74
|
-
# config.default_formatter = 'doc'
|
75
|
-
# end
|
76
|
-
|
77
|
-
# Print the 10 slowest examples and example groups at the
|
78
|
-
# end of the spec run, to help surface which specs are running
|
79
|
-
# particularly slow.
|
80
|
-
# config.profile_examples = 10
|
81
|
-
|
82
|
-
# Run specs in random order to surface order dependencies. If you find an
|
83
|
-
# order dependency and want to debug it, you can fix the order by providing
|
84
|
-
# the seed, which is printed after each run.
|
85
|
-
# --seed 1234
|
86
|
-
config.order = :random
|
87
|
-
|
88
|
-
# Seed global randomization in this process using the `--seed` CLI option.
|
89
|
-
# Setting this allows you to use `--seed` to deterministically reproduce
|
90
|
-
# test failures related to randomization by passing the same `--seed` value
|
91
|
-
# as the one that triggered the failure.
|
92
|
-
Kernel.srand config.seed
|
93
|
-
|
94
|
-
config.before(:suite) do
|
95
|
-
DatabaseCleaner.strategy = :transaction
|
96
|
-
DatabaseCleaner.clean_with(:truncation)
|
97
|
-
end
|
98
|
-
|
99
|
-
config.around(:each) do |example|
|
100
|
-
DatabaseCleaner.cleaning do
|
101
|
-
example.run
|
102
|
-
end
|
103
|
-
end
|
104
|
-
end
|
data/spec/support/ar_stand_in.rb
DELETED
data/spec/support/character.rb
DELETED
data/spec/support/game.rb
DELETED
data/spec/support/game_fail.rb
DELETED
data/spec/support/not_there.rb
DELETED
data/spec/support/web_comic.rb
DELETED
@@ -1,69 +0,0 @@
|
|
1
|
-
require 'support/web_comic_with_searchable_name'
|
2
|
-
require 'support/web_comic_with_searchable_name_and_author'
|
3
|
-
|
4
|
-
RSpec.describe Textacular::FullTextIndexer do
|
5
|
-
context "with one specific field in a Searchable call" do
|
6
|
-
it "generates the right SQL" do
|
7
|
-
file_name = "web_comic_with_searchable_name_full_text_search"
|
8
|
-
content = <<-MIGRATION
|
9
|
-
class WebComicWithSearchableNameFullTextSearch < ActiveRecord::Migration
|
10
|
-
def self.up
|
11
|
-
execute(<<-SQL.strip)
|
12
|
-
DROP index IF EXISTS web_comics_name_fts_idx;
|
13
|
-
CREATE index web_comics_name_fts_idx
|
14
|
-
ON web_comics
|
15
|
-
USING gin(to_tsvector('english', "web_comics"."name"::text));
|
16
|
-
SQL
|
17
|
-
end
|
18
|
-
|
19
|
-
def self.down
|
20
|
-
execute(<<-SQL.strip)
|
21
|
-
DROP index IF EXISTS web_comics_name_fts_idx;
|
22
|
-
SQL
|
23
|
-
end
|
24
|
-
end
|
25
|
-
MIGRATION
|
26
|
-
|
27
|
-
generator = double(:migration_generator)
|
28
|
-
expect(Textacular::MigrationGenerator).to receive(:new).with(file_name, content).and_return(generator)
|
29
|
-
expect(generator).to receive(:generate_migration)
|
30
|
-
|
31
|
-
Textacular::FullTextIndexer.new.generate_migration('WebComicWithSearchableName')
|
32
|
-
end
|
33
|
-
end
|
34
|
-
|
35
|
-
context "with two specific fields in a Searchable call" do
|
36
|
-
it "generates the right SQL" do
|
37
|
-
file_name = "web_comic_with_searchable_name_and_author_full_text_search"
|
38
|
-
content = <<-MIGRATION
|
39
|
-
class WebComicWithSearchableNameAndAuthorFullTextSearch < ActiveRecord::Migration
|
40
|
-
def self.up
|
41
|
-
execute(<<-SQL.strip)
|
42
|
-
DROP index IF EXISTS web_comics_name_fts_idx;
|
43
|
-
CREATE index web_comics_name_fts_idx
|
44
|
-
ON web_comics
|
45
|
-
USING gin(to_tsvector('english', "web_comics"."name"::text));
|
46
|
-
DROP index IF EXISTS web_comics_author_fts_idx;
|
47
|
-
CREATE index web_comics_author_fts_idx
|
48
|
-
ON web_comics
|
49
|
-
USING gin(to_tsvector('english', "web_comics"."author"::text));
|
50
|
-
SQL
|
51
|
-
end
|
52
|
-
|
53
|
-
def self.down
|
54
|
-
execute(<<-SQL.strip)
|
55
|
-
DROP index IF EXISTS web_comics_name_fts_idx;
|
56
|
-
DROP index IF EXISTS web_comics_author_fts_idx;
|
57
|
-
SQL
|
58
|
-
end
|
59
|
-
end
|
60
|
-
MIGRATION
|
61
|
-
|
62
|
-
generator = double(:migration_generator)
|
63
|
-
expect(Textacular::MigrationGenerator).to receive(:new).with(file_name, content).and_return(generator)
|
64
|
-
expect(generator).to receive(:generate_migration)
|
65
|
-
|
66
|
-
Textacular::FullTextIndexer.new.generate_migration('WebComicWithSearchableNameAndAuthor')
|
67
|
-
end
|
68
|
-
end
|
69
|
-
end
|
@@ -1,67 +0,0 @@
|
|
1
|
-
RSpec.describe Textacular::MigrationGenerator do
|
2
|
-
describe ".stream_output" do
|
3
|
-
context "when Rails is not defined" do
|
4
|
-
subject do
|
5
|
-
Textacular::MigrationGenerator.new('filename', 'content')
|
6
|
-
end
|
7
|
-
|
8
|
-
it "points to STDOUT" do
|
9
|
-
output_stream = nil
|
10
|
-
|
11
|
-
subject.stream_output do |io|
|
12
|
-
output_stream = io
|
13
|
-
end
|
14
|
-
|
15
|
-
expect(output_stream).to eq(STDOUT)
|
16
|
-
end
|
17
|
-
end
|
18
|
-
|
19
|
-
context "when Rails is defined" do
|
20
|
-
before do
|
21
|
-
module ::Rails
|
22
|
-
# Stub this out, sort of.
|
23
|
-
def self.root
|
24
|
-
File.join('.', 'fake_rails')
|
25
|
-
end
|
26
|
-
end
|
27
|
-
end
|
28
|
-
|
29
|
-
after do
|
30
|
-
Object.send(:remove_const, :Rails)
|
31
|
-
FileUtils.rm_rf(File.join('.', 'fake_rails'))
|
32
|
-
end
|
33
|
-
|
34
|
-
let(:now) do
|
35
|
-
Time.now
|
36
|
-
end
|
37
|
-
|
38
|
-
subject do
|
39
|
-
Textacular::MigrationGenerator.new('file_name', 'content')
|
40
|
-
end
|
41
|
-
|
42
|
-
it "points to a properly names migration file" do
|
43
|
-
expected_file_name = "./fake_rails/db/migrate/#{now.strftime('%Y%m%d%H%M%S')}_file_name.rb"
|
44
|
-
|
45
|
-
output_stream = nil
|
46
|
-
|
47
|
-
subject.stream_output(now) do |io|
|
48
|
-
output_stream = io
|
49
|
-
end
|
50
|
-
|
51
|
-
expect(output_stream.path).to eq(expected_file_name)
|
52
|
-
end
|
53
|
-
end
|
54
|
-
|
55
|
-
it "generates the right SQL" do
|
56
|
-
content = "content\n" #newline automatically added
|
57
|
-
output = StringIO.new
|
58
|
-
|
59
|
-
generator = Textacular::MigrationGenerator.new('file_name', content)
|
60
|
-
generator.instance_variable_set(:@output_stream, output)
|
61
|
-
|
62
|
-
generator.generate_migration
|
63
|
-
|
64
|
-
expect(output.string).to eq(content)
|
65
|
-
end
|
66
|
-
end
|
67
|
-
end
|
@@ -1,244 +0,0 @@
|
|
1
|
-
require 'support/web_comic_with_searchable'
|
2
|
-
require 'support/web_comic_with_searchable_name'
|
3
|
-
require 'support/web_comic_with_searchable_name_and_author'
|
4
|
-
require 'support/character'
|
5
|
-
|
6
|
-
RSpec.describe "Searchable" do
|
7
|
-
context "when extending an ActiveRecord::Base subclass" do
|
8
|
-
context "with no parameters" do
|
9
|
-
let!(:questionable_content) do
|
10
|
-
WebComicWithSearchable.create(
|
11
|
-
name: 'Questionable Content',
|
12
|
-
author: 'Jeph Jaques',
|
13
|
-
)
|
14
|
-
end
|
15
|
-
|
16
|
-
let!(:johnny_wander) do
|
17
|
-
WebComicWithSearchable.create(
|
18
|
-
name: 'Johnny Wander',
|
19
|
-
author: 'Ananth & Yuko',
|
20
|
-
)
|
21
|
-
end
|
22
|
-
|
23
|
-
let!(:dominic_deegan) do
|
24
|
-
WebComicWithSearchable.create(
|
25
|
-
name: 'Dominic Deegan',
|
26
|
-
author: 'Mookie',
|
27
|
-
)
|
28
|
-
end
|
29
|
-
|
30
|
-
let!(:penny_arcade) do
|
31
|
-
WebComicWithSearchable.create(
|
32
|
-
name: 'Penny Arcade',
|
33
|
-
author: 'Tycho & Gabe',
|
34
|
-
)
|
35
|
-
end
|
36
|
-
|
37
|
-
let!(:null) do
|
38
|
-
WebComicWithSearchable.create(
|
39
|
-
author: 'Foo',
|
40
|
-
)
|
41
|
-
end
|
42
|
-
|
43
|
-
it "searches across all columns" do
|
44
|
-
expect(
|
45
|
-
WebComicWithSearchable.advanced_search("Penny")
|
46
|
-
).to eq([penny_arcade])
|
47
|
-
expect(
|
48
|
-
WebComicWithSearchable.advanced_search("Dominic")
|
49
|
-
).to eq([dominic_deegan])
|
50
|
-
end
|
51
|
-
|
52
|
-
it "ranks results, egen with NULL columns" do
|
53
|
-
comic = WebComicWithSearchable.basic_search('Foo').first
|
54
|
-
rank = comic.attributes.find { |key, value| key.to_s =~ /\Arank\d+\z/ }.last
|
55
|
-
|
56
|
-
expect(rank).to be_present
|
57
|
-
end
|
58
|
-
end
|
59
|
-
|
60
|
-
context "with one column as a parameter" do
|
61
|
-
let!(:questionable_content) do
|
62
|
-
WebComicWithSearchableName.create(
|
63
|
-
name: 'Questionable Content',
|
64
|
-
author: nil,
|
65
|
-
)
|
66
|
-
end
|
67
|
-
|
68
|
-
let!(:johnny_wander) do
|
69
|
-
WebComicWithSearchableName.create(
|
70
|
-
name: 'Johnny Wander',
|
71
|
-
author: 'Ananth & Yuko',
|
72
|
-
)
|
73
|
-
end
|
74
|
-
|
75
|
-
let!(:dominic_deegan) do
|
76
|
-
WebComicWithSearchableName.create(
|
77
|
-
name: 'Dominic Deegan',
|
78
|
-
author: 'Mookie',
|
79
|
-
)
|
80
|
-
end
|
81
|
-
|
82
|
-
let!(:penny_arcade) do
|
83
|
-
WebComicWithSearchableName.create(
|
84
|
-
name: 'Penny Arcade',
|
85
|
-
author: 'Tycho & Gabe',
|
86
|
-
)
|
87
|
-
end
|
88
|
-
|
89
|
-
it "only searches across the given column" do
|
90
|
-
expect(WebComicWithSearchableName.advanced_search("Penny")).to eq([penny_arcade])
|
91
|
-
|
92
|
-
expect(WebComicWithSearchableName.advanced_search("Tycho")).to be_empty
|
93
|
-
end
|
94
|
-
|
95
|
-
describe "basic search" do # Uses plainto_tsquery
|
96
|
-
["hello \\", "tebow!" , "food &"].each do |search_term|
|
97
|
-
it "works with interesting term \"#{search_term}\"" do
|
98
|
-
expect(WebComicWithSearchableName.basic_search(search_term)).to be_empty
|
99
|
-
end
|
100
|
-
end
|
101
|
-
end
|
102
|
-
|
103
|
-
describe "advanced_search" do # Uses to_tsquery
|
104
|
-
["hello \\", "tebow!" , "food &"].each do |search_term|
|
105
|
-
it "fails with interesting term \"#{search_term}\"" do
|
106
|
-
expect {
|
107
|
-
WebComicWithSearchableName.advanced_search(search_term).first
|
108
|
-
}.to raise_error(ActiveRecord::StatementInvalid)
|
109
|
-
end
|
110
|
-
end
|
111
|
-
it "searches with negation" do
|
112
|
-
expect(WebComicWithSearchableName.advanced_search('foo & ! bar')).to be_empty
|
113
|
-
end
|
114
|
-
end
|
115
|
-
|
116
|
-
describe "web search" do # Uses websearch_to_tsquery
|
117
|
-
["hello \\", "tebow!" , "food &"].each do |search_term|
|
118
|
-
it "works with interesting term \"#{search_term}\"" do
|
119
|
-
expect(WebComicWithSearchableName.web_search(search_term)).to be_empty
|
120
|
-
end
|
121
|
-
end
|
122
|
-
end
|
123
|
-
|
124
|
-
it "does fuzzy searching" do
|
125
|
-
expect(
|
126
|
-
WebComicWithSearchableName.fuzzy_search('Questio')
|
127
|
-
).to eq([questionable_content])
|
128
|
-
end
|
129
|
-
|
130
|
-
it "return a valid rank when fuzzy searching on NULL columns" do
|
131
|
-
qcont_with_author = questionable_content.becomes(WebComicWithSearchableNameAndAuthor)
|
132
|
-
search_result = WebComicWithSearchableNameAndAuthor.fuzzy_search('Questio')
|
133
|
-
expect([qcont_with_author]).to eq(search_result)
|
134
|
-
expect(search_result.first.attributes.find { |k, _| k[0..3] == 'rank' }.last).to be_truthy
|
135
|
-
end
|
136
|
-
|
137
|
-
it "defines :searchable_columns as private" do
|
138
|
-
expect { WebComicWithSearchableName.searchable_columns }.to raise_error(NoMethodError)
|
139
|
-
|
140
|
-
begin
|
141
|
-
WebComicWithSearchableName.searchable_columns
|
142
|
-
rescue NoMethodError => error
|
143
|
-
expect(error.message).to match(/private method/)
|
144
|
-
end
|
145
|
-
end
|
146
|
-
|
147
|
-
it "defines #indexable_columns which returns a write-proof Enumerable" do
|
148
|
-
expect(WebComicWithSearchableName.indexable_columns).to be_an(Enumerator)
|
149
|
-
|
150
|
-
expect {
|
151
|
-
WebComicWithSearchableName.indexable_columns[0] = 'foo'
|
152
|
-
}.to raise_error(NoMethodError)
|
153
|
-
end
|
154
|
-
end
|
155
|
-
|
156
|
-
context "with two columns as parameters" do
|
157
|
-
let!(:questionable_content) do
|
158
|
-
WebComicWithSearchableNameAndAuthor.create(
|
159
|
-
name: 'Questionable Content',
|
160
|
-
author: 'Jeph Jaques',
|
161
|
-
)
|
162
|
-
end
|
163
|
-
|
164
|
-
let!(:johnny_wander) do
|
165
|
-
WebComicWithSearchableNameAndAuthor.create(
|
166
|
-
name: 'Johnny Wander',
|
167
|
-
author: 'Ananth & Yuko',
|
168
|
-
)
|
169
|
-
end
|
170
|
-
|
171
|
-
let!(:dominic_deegan) do
|
172
|
-
WebComicWithSearchableNameAndAuthor.create(
|
173
|
-
name: 'Dominic Deegan',
|
174
|
-
author: 'Mookie',
|
175
|
-
)
|
176
|
-
end
|
177
|
-
|
178
|
-
let!(:penny_arcade) do
|
179
|
-
WebComicWithSearchableNameAndAuthor.create(
|
180
|
-
name: 'Penny Arcade',
|
181
|
-
author: 'Tycho & Gabe',
|
182
|
-
)
|
183
|
-
end
|
184
|
-
|
185
|
-
it "only searches across the given columns" do
|
186
|
-
expect(
|
187
|
-
WebComicWithSearchableNameAndAuthor.advanced_search("Penny")
|
188
|
-
).to eq([penny_arcade])
|
189
|
-
|
190
|
-
expect(
|
191
|
-
WebComicWithSearchableNameAndAuthor.advanced_search("Tycho")
|
192
|
-
).to eq([penny_arcade])
|
193
|
-
|
194
|
-
expect(
|
195
|
-
WebComicWithSearchableNameAndAuthor.web_search("Penny")
|
196
|
-
).to eq([penny_arcade])
|
197
|
-
|
198
|
-
expect(
|
199
|
-
WebComicWithSearchableNameAndAuthor.web_search("Tycho")
|
200
|
-
).to eq([penny_arcade])
|
201
|
-
end
|
202
|
-
|
203
|
-
it "allows includes" do
|
204
|
-
expect(
|
205
|
-
WebComicWithSearchableNameAndAuthor.includes(:characters).advanced_search("Penny")
|
206
|
-
).to eq([penny_arcade])
|
207
|
-
end
|
208
|
-
end
|
209
|
-
|
210
|
-
context 'custom rank' do
|
211
|
-
let!(:questionable_content) do
|
212
|
-
WebComicWithSearchableName.create(
|
213
|
-
name: 'Questionable Content',
|
214
|
-
author: nil,
|
215
|
-
)
|
216
|
-
end
|
217
|
-
|
218
|
-
it "is selected for search" do
|
219
|
-
search_result = WebComicWithSearchableNameAndAuthor.search('Questionable Content', true, 'my_rank')
|
220
|
-
expect(search_result.first.attributes['my_rank']).to be_truthy
|
221
|
-
end
|
222
|
-
|
223
|
-
it "is selected for basic_search" do
|
224
|
-
search_result = WebComicWithSearchableNameAndAuthor.basic_search('Questionable Content', true, 'my_rank')
|
225
|
-
expect(search_result.first.attributes['my_rank']).to be_truthy
|
226
|
-
end
|
227
|
-
|
228
|
-
it "is selected for advanced_search" do
|
229
|
-
search_result = WebComicWithSearchableNameAndAuthor.advanced_search('Questionable Content', true, 'my_rank')
|
230
|
-
expect(search_result.first.attributes['my_rank']).to be_truthy
|
231
|
-
end
|
232
|
-
|
233
|
-
it "is selected for fuzzy_search" do
|
234
|
-
search_result = WebComicWithSearchableNameAndAuthor.fuzzy_search('Questionable Content', true, 'my_rank')
|
235
|
-
expect(search_result.first.attributes['my_rank']).to be_truthy
|
236
|
-
end
|
237
|
-
|
238
|
-
it "is selected for web_search" do
|
239
|
-
search_result = WebComicWithSearchableNameAndAuthor.web_search('Questionable Content', true, 'my_rank')
|
240
|
-
expect(search_result.first.attributes['my_rank']).to be_truthy
|
241
|
-
end
|
242
|
-
end
|
243
|
-
end
|
244
|
-
end
|
@@ -1,24 +0,0 @@
|
|
1
|
-
RSpec.describe "Textacular::TrigramInstaller" do
|
2
|
-
let(:content) do
|
3
|
-
<<-MIGRATION
|
4
|
-
class InstallTrigram < ActiveRecord::Migration[5.0]
|
5
|
-
def self.up
|
6
|
-
ActiveRecord::Base.connection.execute("CREATE EXTENSION IF NOT EXISTS pg_trgm;")
|
7
|
-
end
|
8
|
-
|
9
|
-
def self.down
|
10
|
-
ActiveRecord::Base.connection.execute("DROP EXTENSION pg_trgm;")
|
11
|
-
end
|
12
|
-
end
|
13
|
-
MIGRATION
|
14
|
-
end
|
15
|
-
|
16
|
-
it "generates a migration" do
|
17
|
-
generator = double(:migration_generator)
|
18
|
-
|
19
|
-
expect(Textacular::MigrationGenerator).to receive(:new).with('install_trigram', content).and_return(generator)
|
20
|
-
expect(generator).to receive(:generate_migration)
|
21
|
-
|
22
|
-
Textacular::TrigramInstaller.new.generate_migration
|
23
|
-
end
|
24
|
-
end
|
data/spec/textacular_spec.rb
DELETED
@@ -1,294 +0,0 @@
|
|
1
|
-
# coding: utf-8
|
2
|
-
# Above for Ruby 1.9 tests
|
3
|
-
|
4
|
-
require 'support/ar_stand_in'
|
5
|
-
require 'support/not_there'
|
6
|
-
require 'support/textacular_web_comic'
|
7
|
-
require 'support/game_extended_with_textacular'
|
8
|
-
require 'support/game_extended_with_textacular_and_custom_language'
|
9
|
-
require 'support/game_fail_extended_with_textacular'
|
10
|
-
|
11
|
-
RSpec.describe Textacular do
|
12
|
-
context "after extending ActiveRecord::Base" do
|
13
|
-
it "doesn't break #respond_to?" do
|
14
|
-
expect{ ARStandIn.respond_to?(:abstract_class?) }.to_not raise_error
|
15
|
-
end
|
16
|
-
|
17
|
-
it "doesn't break #respond_to? for table-less classes" do
|
18
|
-
expect(NotThere.table_exists?).to be_falsey
|
19
|
-
expect { NotThere.respond_to? :system }.to_not raise_error
|
20
|
-
end
|
21
|
-
|
22
|
-
it "doesn't break #method_missing" do
|
23
|
-
expect { ARStandIn.random }.to raise_error(NoMethodError)
|
24
|
-
|
25
|
-
begin
|
26
|
-
ARStandIn.random
|
27
|
-
rescue NoMethodError => error
|
28
|
-
expect(error.message).to match(/undefined method `random'/)
|
29
|
-
end
|
30
|
-
end
|
31
|
-
|
32
|
-
it "doesn't break #method_missing for table-less classes" do
|
33
|
-
expect(NotThere.table_exists?).to be_falsey
|
34
|
-
|
35
|
-
expect { NotThere.random }.to raise_error(NoMethodError)
|
36
|
-
|
37
|
-
begin
|
38
|
-
NotThere.random
|
39
|
-
rescue NoMethodError => error
|
40
|
-
expect(error.message).to match(/undefined method `random'/)
|
41
|
-
end
|
42
|
-
end
|
43
|
-
|
44
|
-
context "when finding models based on searching a related model" do
|
45
|
-
let(:webcomics_with_tall_characters) do
|
46
|
-
[johnny_wander]
|
47
|
-
end
|
48
|
-
|
49
|
-
let(:webcomics_with_angry_characters) do
|
50
|
-
[johnny_wander, penny_arcade, questionable_content]
|
51
|
-
end
|
52
|
-
|
53
|
-
let(:webcomics_with_crude_characters) do
|
54
|
-
[penny_arcade, questionable_content]
|
55
|
-
end
|
56
|
-
|
57
|
-
let!(:johnny_wander) do
|
58
|
-
TextacularWebComic.create(:name => "Johnny Wander", :author => "Ananth & Yuko").tap do |comic|
|
59
|
-
comic.characters.create :name => 'Ananth', :description => 'Stubble! What is under that hat?!?'
|
60
|
-
comic.characters.create :name => 'Yuko', :description => 'So... small. Carl Sagan haircut.'
|
61
|
-
comic.characters.create :name => 'John', :description => 'Tall. Anger issues?'
|
62
|
-
comic.characters.create :name => 'Cricket', :description => 'Chirrup!'
|
63
|
-
end
|
64
|
-
end
|
65
|
-
|
66
|
-
let!(:questionable_content) do
|
67
|
-
TextacularWebComic.create(:name => "Questionable Content", :author => "Jeph Jaques").tap do |comic|
|
68
|
-
comic.characters.create :name => 'Martin', :description => 'the insecure protagonist'
|
69
|
-
comic.characters.create :name => 'Faye', :description => 'a sarcastic barrista with anger management issues'
|
70
|
-
comic.characters.create :name => 'Pintsize', :description => 'a crude AnthroPC'
|
71
|
-
end
|
72
|
-
end
|
73
|
-
|
74
|
-
let!(:penny_arcade) do
|
75
|
-
TextacularWebComic.create(:name => "Penny Arcade", :author => "Tycho & Gabe").tap do |comic|
|
76
|
-
comic.characters.create :name => 'Gabe', :description => 'the simple one'
|
77
|
-
comic.characters.create :name => 'Tycho', :description => 'the wordy one'
|
78
|
-
comic.characters.create :name => 'Div', :description => 'a crude divx player with anger management issues'
|
79
|
-
end
|
80
|
-
end
|
81
|
-
|
82
|
-
it "looks in the related model with nested searching syntax" do
|
83
|
-
expect(
|
84
|
-
TextacularWebComic.joins(:characters).advanced_search(
|
85
|
-
:characters => {:description => 'tall'}
|
86
|
-
)
|
87
|
-
).to eq(webcomics_with_tall_characters)
|
88
|
-
|
89
|
-
expect(
|
90
|
-
TextacularWebComic.joins(:characters).advanced_search(
|
91
|
-
:characters => {:description => 'anger'}
|
92
|
-
).sort
|
93
|
-
).to eq(webcomics_with_angry_characters.sort)
|
94
|
-
|
95
|
-
expect(
|
96
|
-
TextacularWebComic.joins(:characters).advanced_search(
|
97
|
-
:characters => {:description => 'crude'}
|
98
|
-
).sort
|
99
|
-
).to eq(webcomics_with_crude_characters.sort)
|
100
|
-
end
|
101
|
-
end
|
102
|
-
end
|
103
|
-
|
104
|
-
context "after extending an ActiveRecord::Base subclass" do
|
105
|
-
context "when the DB connection is unavailable" do
|
106
|
-
before do
|
107
|
-
GameFailExtendedWithTextacular.establish_connection({:adapter => :postgresql, :database =>'unavailable', :username=>'bad', :pool=>5, :timeout=>5000}) rescue nil
|
108
|
-
end
|
109
|
-
|
110
|
-
it "doesn't break respond_to?" do
|
111
|
-
expect { GameFailExtendedWithTextacular.respond_to?(:advanced_search) }.to_not raise_error
|
112
|
-
end
|
113
|
-
end
|
114
|
-
|
115
|
-
context "when the DB connection is available" do
|
116
|
-
let!(:zelda) do
|
117
|
-
GameExtendedWithTextacular.create(
|
118
|
-
:system => "NES",
|
119
|
-
:title => "Legend of Zelda",
|
120
|
-
:description => "A Link to the Past."
|
121
|
-
)
|
122
|
-
end
|
123
|
-
|
124
|
-
let!(:mario) do
|
125
|
-
GameExtendedWithTextacular.create(
|
126
|
-
:system => "NES",
|
127
|
-
:title => "Super Mario Bros.",
|
128
|
-
:description => "The original platformer."
|
129
|
-
)
|
130
|
-
end
|
131
|
-
|
132
|
-
let!(:sonic) do
|
133
|
-
GameExtendedWithTextacular.create(
|
134
|
-
:system => "Genesis",
|
135
|
-
:title => "Sonic the Hedgehog",
|
136
|
-
:description => "Spiky."
|
137
|
-
)
|
138
|
-
end
|
139
|
-
|
140
|
-
let!(:donkey_kong) do
|
141
|
-
GameExtendedWithTextacular.create(
|
142
|
-
:system => "SNES",
|
143
|
-
:title => "Diddy's Kong Quest",
|
144
|
-
:description => "Donkey Kong Country 2"
|
145
|
-
)
|
146
|
-
end
|
147
|
-
|
148
|
-
let!(:mega_man) do
|
149
|
-
GameExtendedWithTextacular.create(
|
150
|
-
:system => nil,
|
151
|
-
:title => "Mega Man",
|
152
|
-
:description => "Beware Dr. Brain"
|
153
|
-
)
|
154
|
-
end
|
155
|
-
|
156
|
-
let!(:sf_nes) do
|
157
|
-
GameExtendedWithTextacular.create(
|
158
|
-
:system => "SNES",
|
159
|
-
:title => "Street Fighter 2",
|
160
|
-
:description => "Yoga Flame!"
|
161
|
-
)
|
162
|
-
end
|
163
|
-
|
164
|
-
let!(:sf_genesis) do
|
165
|
-
GameExtendedWithTextacular.create(
|
166
|
-
:system => "Genesis",
|
167
|
-
:title => "Street Fighter 2",
|
168
|
-
:description => "Yoga Flame!"
|
169
|
-
)
|
170
|
-
end
|
171
|
-
|
172
|
-
let!(:takun) do
|
173
|
-
GameExtendedWithTextacular.create(
|
174
|
-
:system => "Saturn",
|
175
|
-
:title => "Magical Tarurūto-kun",
|
176
|
-
:description => "カッコイイ!"
|
177
|
-
)
|
178
|
-
end
|
179
|
-
|
180
|
-
it "defines a #search method" do
|
181
|
-
expect(GameExtendedWithTextacular).to respond_to(:search)
|
182
|
-
end
|
183
|
-
|
184
|
-
describe "#fuzzy_search" do
|
185
|
-
it 'searches non-text columns' do
|
186
|
-
expect(GameExtendedWithTextacular.fuzzy_search(id: mario.id)
|
187
|
-
).to eq([mario])
|
188
|
-
end
|
189
|
-
end
|
190
|
-
|
191
|
-
describe "#advanced_search" do
|
192
|
-
context "with a String argument" do
|
193
|
-
it "searches across all :string columns (if not indexes have been specified)" do
|
194
|
-
expect(
|
195
|
-
GameExtendedWithTextacular.advanced_search("Mario")
|
196
|
-
).to eq([mario])
|
197
|
-
|
198
|
-
expect(
|
199
|
-
GameExtendedWithTextacular.advanced_search("NES").to_set
|
200
|
-
).to eq(Set.new([mario, zelda]))
|
201
|
-
end
|
202
|
-
|
203
|
-
it "works if a query has an apostrophe" do
|
204
|
-
expect(GameExtendedWithTextacular.advanced_search("Diddy's")).to eq([donkey_kong])
|
205
|
-
end
|
206
|
-
|
207
|
-
it "works if the query contains whitespace" do
|
208
|
-
expect(GameExtendedWithTextacular.advanced_search("Mega Man")).to eq([mega_man])
|
209
|
-
end
|
210
|
-
|
211
|
-
it "works if the query contains an accent" do
|
212
|
-
expect(GameExtendedWithTextacular.advanced_search("Tarurūto-kun")).to eq([takun])
|
213
|
-
end
|
214
|
-
|
215
|
-
it "searches across records with NULL values" do
|
216
|
-
expect(GameExtendedWithTextacular.advanced_search("Mega")).to eq([mega_man])
|
217
|
-
end
|
218
|
-
|
219
|
-
it "scopes consecutively" do
|
220
|
-
expect(
|
221
|
-
GameExtendedWithTextacular.advanced_search("Genesis").advanced_search("Street Fighter")
|
222
|
-
).to eq([sf_genesis])
|
223
|
-
end
|
224
|
-
end
|
225
|
-
|
226
|
-
context "with a Hash argument" do
|
227
|
-
it "searches across the given columns" do
|
228
|
-
expect(
|
229
|
-
GameExtendedWithTextacular.advanced_search(:title => 'NES')
|
230
|
-
).to be_empty
|
231
|
-
expect(
|
232
|
-
GameExtendedWithTextacular.advanced_search(:system => "Mario")
|
233
|
-
).to be_empty
|
234
|
-
expect(
|
235
|
-
GameExtendedWithTextacular.advanced_search(:system => "NES", :title => "Sonic")
|
236
|
-
).to be_empty
|
237
|
-
|
238
|
-
expect(
|
239
|
-
GameExtendedWithTextacular.advanced_search(:title => "Mario")
|
240
|
-
).to eq([mario])
|
241
|
-
|
242
|
-
expect(
|
243
|
-
GameExtendedWithTextacular.advanced_search(:system => "NES").size
|
244
|
-
).to eq(2)
|
245
|
-
|
246
|
-
expect(
|
247
|
-
GameExtendedWithTextacular.advanced_search(:system => "NES", :title => "Zelda")
|
248
|
-
).to eq([zelda])
|
249
|
-
expect(
|
250
|
-
GameExtendedWithTextacular.advanced_search(:title => "Mega")
|
251
|
-
).to eq([mega_man])
|
252
|
-
end
|
253
|
-
|
254
|
-
it "scopes consecutively" do
|
255
|
-
expect(
|
256
|
-
GameExtendedWithTextacular
|
257
|
-
.advanced_search(:system => "Genesis")
|
258
|
-
.advanced_search(:title => "Street Fighter")
|
259
|
-
).to eq([sf_genesis])
|
260
|
-
end
|
261
|
-
|
262
|
-
it "casts non-string columns as text" do
|
263
|
-
expect(
|
264
|
-
GameExtendedWithTextacular.advanced_search(:id => mario.id)
|
265
|
-
).to eq([mario])
|
266
|
-
end
|
267
|
-
end
|
268
|
-
|
269
|
-
context "after selecting columns to return" do
|
270
|
-
it "doesn't fetch extra columns" do
|
271
|
-
expect {
|
272
|
-
GameExtendedWithTextacular.select(:title).advanced_search("Mario").first.system
|
273
|
-
}.to raise_error(ActiveModel::MissingAttributeError)
|
274
|
-
end
|
275
|
-
end
|
276
|
-
|
277
|
-
context "after setting a custom language" do
|
278
|
-
let!(:harry_potter_7) do
|
279
|
-
GameExtendedWithTextacularAndCustomLanguage.create(
|
280
|
-
:system => "PS3",
|
281
|
-
:title => "Harry Potter & the Deathly Hallows"
|
282
|
-
)
|
283
|
-
end
|
284
|
-
|
285
|
-
it "finds results" do
|
286
|
-
expect(
|
287
|
-
GameExtendedWithTextacularAndCustomLanguage.advanced_search(title: "harry")
|
288
|
-
).to eq([harry_potter_7])
|
289
|
-
end
|
290
|
-
end
|
291
|
-
end
|
292
|
-
end
|
293
|
-
end
|
294
|
-
end
|