ts-xml 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (91) hide show
  1. data/LICENSE +20 -0
  2. data/README.textile +30 -0
  3. data/features/alternate_primary_key.feature +27 -0
  4. data/features/attribute_transformation.feature +22 -0
  5. data/features/attribute_updates.feature +39 -0
  6. data/features/deleting_instances.feature +67 -0
  7. data/features/direct_attributes.feature +11 -0
  8. data/features/excerpts.feature +13 -0
  9. data/features/extensible_delta_indexing.feature +9 -0
  10. data/features/facets.feature +82 -0
  11. data/features/facets_across_model.feature +29 -0
  12. data/features/handling_edits.feature +92 -0
  13. data/features/retry_stale_indexes.feature +24 -0
  14. data/features/searching_across_models.feature +20 -0
  15. data/features/searching_by_index.feature +40 -0
  16. data/features/searching_by_model.feature +175 -0
  17. data/features/searching_with_find_arguments.feature +56 -0
  18. data/features/sphinx_detection.feature +25 -0
  19. data/features/sphinx_scopes.feature +42 -0
  20. data/features/step_definitions/alpha_steps.rb +7 -0
  21. data/features/step_definitions/beta_steps.rb +7 -0
  22. data/features/step_definitions/common_steps.rb +188 -0
  23. data/features/step_definitions/extensible_delta_indexing_steps.rb +7 -0
  24. data/features/step_definitions/facet_steps.rb +96 -0
  25. data/features/step_definitions/find_arguments_steps.rb +36 -0
  26. data/features/step_definitions/gamma_steps.rb +15 -0
  27. data/features/step_definitions/scope_steps.rb +15 -0
  28. data/features/step_definitions/search_steps.rb +89 -0
  29. data/features/step_definitions/sphinx_steps.rb +35 -0
  30. data/features/sti_searching.feature +19 -0
  31. data/features/support/database.example.yml +3 -0
  32. data/features/support/db/fixtures/alphas.rb +10 -0
  33. data/features/support/db/fixtures/authors.rb +1 -0
  34. data/features/support/db/fixtures/betas.rb +10 -0
  35. data/features/support/db/fixtures/boxes.rb +9 -0
  36. data/features/support/db/fixtures/categories.rb +1 -0
  37. data/features/support/db/fixtures/cats.rb +3 -0
  38. data/features/support/db/fixtures/comments.rb +24 -0
  39. data/features/support/db/fixtures/developers.rb +29 -0
  40. data/features/support/db/fixtures/dogs.rb +3 -0
  41. data/features/support/db/fixtures/extensible_betas.rb +10 -0
  42. data/features/support/db/fixtures/foxes.rb +3 -0
  43. data/features/support/db/fixtures/gammas.rb +10 -0
  44. data/features/support/db/fixtures/people.rb +1001 -0
  45. data/features/support/db/fixtures/posts.rb +6 -0
  46. data/features/support/db/fixtures/robots.rb +14 -0
  47. data/features/support/db/fixtures/tags.rb +27 -0
  48. data/features/support/db/migrations/create_alphas.rb +7 -0
  49. data/features/support/db/migrations/create_animals.rb +5 -0
  50. data/features/support/db/migrations/create_authors.rb +3 -0
  51. data/features/support/db/migrations/create_authors_posts.rb +6 -0
  52. data/features/support/db/migrations/create_betas.rb +5 -0
  53. data/features/support/db/migrations/create_boxes.rb +5 -0
  54. data/features/support/db/migrations/create_categories.rb +3 -0
  55. data/features/support/db/migrations/create_comments.rb +10 -0
  56. data/features/support/db/migrations/create_developers.rb +9 -0
  57. data/features/support/db/migrations/create_extensible_betas.rb +5 -0
  58. data/features/support/db/migrations/create_gammas.rb +3 -0
  59. data/features/support/db/migrations/create_people.rb +13 -0
  60. data/features/support/db/migrations/create_posts.rb +5 -0
  61. data/features/support/db/migrations/create_robots.rb +4 -0
  62. data/features/support/db/migrations/create_taggings.rb +5 -0
  63. data/features/support/db/migrations/create_tags.rb +4 -0
  64. data/features/support/env.rb +21 -0
  65. data/features/support/lib/generic_delta_handler.rb +8 -0
  66. data/features/support/models/alpha.rb +21 -0
  67. data/features/support/models/animal.rb +5 -0
  68. data/features/support/models/author.rb +3 -0
  69. data/features/support/models/beta.rb +8 -0
  70. data/features/support/models/box.rb +8 -0
  71. data/features/support/models/cat.rb +3 -0
  72. data/features/support/models/category.rb +4 -0
  73. data/features/support/models/comment.rb +10 -0
  74. data/features/support/models/developer.rb +16 -0
  75. data/features/support/models/dog.rb +3 -0
  76. data/features/support/models/extensible_beta.rb +9 -0
  77. data/features/support/models/fox.rb +5 -0
  78. data/features/support/models/gamma.rb +5 -0
  79. data/features/support/models/person.rb +23 -0
  80. data/features/support/models/post.rb +21 -0
  81. data/features/support/models/robot.rb +12 -0
  82. data/features/support/models/tag.rb +3 -0
  83. data/features/support/models/tagging.rb +4 -0
  84. data/lib/thinking_sphinx/xml.rb +5 -0
  85. data/lib/thinking_sphinx/xml/adapters/abstract_adapter.rb +34 -0
  86. data/lib/thinking_sphinx/xml/adapters/oracle_adapter.rb +123 -0
  87. data/lib/thinking_sphinx/xml/adapters/sqlite3_adapter.rb +95 -0
  88. data/lib/thinking_sphinx/xml/source.rb +46 -0
  89. data/lib/thinking_sphinx/xml/tasks.rb +95 -0
  90. data/spec/cucumber_env.rb +22 -0
  91. metadata +183 -0
@@ -0,0 +1,6 @@
1
+ post = Post.create(
2
+ :subject => "Hello World", :content => "Um Text", :id => 1, :category_id => 1
3
+ )
4
+
5
+ post.authors << Author.find(:first)
6
+ post.save
@@ -0,0 +1,14 @@
1
+ # Reset the primary key to allow us to create robots with specific internal_ids
2
+ class Robot < ActiveRecord::Base
3
+ set_primary_key :alternate_primary_key
4
+ end
5
+
6
+ Robot.create :name => 'Fritz', :internal_id => 'F0001'
7
+ Robot.create :name => 'Sizzle', :internal_id => 'S0001'
8
+ Robot.create :name => 'Sizzle Jr.', :internal_id => 'S0002'
9
+ Robot.create :name => 'Expendable', :internal_id => 'E0001'
10
+
11
+ # Annnnnnnnnnd we're back
12
+ class Robot < ActiveRecord::Base
13
+ set_primary_key :internal_id
14
+ end
@@ -0,0 +1,27 @@
1
+ post = Post.find(1)
2
+
3
+ [
4
+ 'Nunc in neque. Integer sed odio ac quam pellentesque suscipit. Cras posuere posuere est. Suspendisse est.',
5
+ 'In hac habitasse platea dictumst. Etiam eleifend est ac diam. Ut ac felis sit amet mi bibendum varius.',
6
+ 'Nullam sollicitudin tellus at ipsum. Duis mi eros, blandit sed, condimentum non, mattis vel, orci.',
7
+ 'Praesent auctor mollis leo. Nulla facilisi. Pellentesque habitant morbi tristique senectus et netus et',
8
+ 'malesuada fames ac turpis egestas. Donec non odio. Maecenas varius elit ut ante. Phasellus egestas, quam',
9
+ 'a congue euismod, diam urna gravida risus, at euismod lectus diam id sem. Vestibulum sed dolor et massa',
10
+ 'porta placerat. Etiam eget risus. Sed ornare. Vivamus in sapien. Maecenas non enim nec metus posuere',
11
+ 'vehicula. Donec rhoncus mauris at metus. Curabitur volutpat massa a metus. Aliquam ornare, neque ut',
12
+ 'tristique convallis, libero orci eleifend nibh, ac porta leo felis sed orci. Lorem ipsum dolor sit amet,',
13
+ 'Waffles' # This one is important, whereas the others are just padding
14
+ ].each do |text|
15
+ Tagging.create :taggable => post, :tag => Tag.create(:text => text)
16
+ end
17
+
18
+ Developer.find(:all).each do |developer|
19
+ [:country, :city, :state].each do |column|
20
+ Tagging.create(
21
+ :taggable => developer,
22
+ :tag => Tag.find_or_create_by_text(developer.send(column))
23
+ )
24
+ end
25
+ end
26
+
27
+ Tagging.create(:taggable => Developer.find(:all).last)
@@ -0,0 +1,7 @@
1
+ ActiveRecord::Base.connection.create_table :alphas, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :value, :integer, :null => false
4
+ t.column :cost, :decimal, :precision => 10, :scale => 6
5
+ t.column :created_on, :date
6
+ t.column :created_at, :timestamp
7
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :animals, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :type, :string
4
+ t.column :delta, :boolean, :null => false, :default => false
5
+ end
@@ -0,0 +1,3 @@
1
+ ActiveRecord::Base.connection.create_table :authors, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ end
@@ -0,0 +1,6 @@
1
+ ActiveRecord::Base.connection.create_table(
2
+ :authors_posts, :force => true, :id => false
3
+ ) do |t|
4
+ t.column :author_id, :integer, :null => false
5
+ t.column :post_id, :integer, :null => false
6
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :betas, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :value, :integer, :null => false
4
+ t.column :delta, :boolean, :null => false, :default => false
5
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :boxes, :force => true do |t|
2
+ t.column :width, :integer, :null => false
3
+ t.column :length, :integer, :null => false
4
+ t.column :depth, :integer
5
+ end
@@ -0,0 +1,3 @@
1
+ ActiveRecord::Base.connection.create_table :categories, :force => true do |t|
2
+ t.column :name, :string
3
+ end
@@ -0,0 +1,10 @@
1
+ ActiveRecord::Base.connection.create_table :comments, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :email, :string
4
+ t.column :url, :string
5
+ t.column :content, :text
6
+ t.column :post_id, :integer, :null => false
7
+ t.column :category_id, :integer, :null => false
8
+ t.column :created_at, :datetime
9
+ t.column :updated_at, :datetime
10
+ end
@@ -0,0 +1,9 @@
1
+ require 'faker'
2
+
3
+ ActiveRecord::Base.connection.create_table :developers, :force => true do |t|
4
+ t.column :name, :string, :null => false
5
+ t.column :city, :string
6
+ t.column :state, :string
7
+ t.column :country, :string
8
+ t.column :age, :integer
9
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :extensible_betas, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :delta, :boolean, :null => false, :default => false
4
+ t.column :changed_by_generic, :boolean, :null => false, :default => false
5
+ end
@@ -0,0 +1,3 @@
1
+ ActiveRecord::Base.connection.create_table :gammas, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ end
@@ -0,0 +1,13 @@
1
+ ActiveRecord::Base.connection.create_table :people, :force => true do |t|
2
+ t.column :first_name, :string
3
+ t.column :middle_initial, :string
4
+ t.column :last_name, :string
5
+ t.column :gender, :string
6
+ t.column :street_address, :string
7
+ t.column :city, :string
8
+ t.column :state, :string
9
+ t.column :postcode, :string
10
+ t.column :email, :string
11
+ t.column :birthday, :datetime
12
+ t.column :delta, :boolean, :null => false, :default => false
13
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :posts, :force => true do |t|
2
+ t.column :subject, :string, :null => false
3
+ t.column :content, :text
4
+ t.column :category_id, :integer, :null => false
5
+ end
@@ -0,0 +1,4 @@
1
+ ActiveRecord::Base.connection.create_table :robots, :primary_key => :alternate_primary_key, :force => true do |t|
2
+ t.column :name, :string, :null => false
3
+ t.column :internal_id, :string, :null => false
4
+ end
@@ -0,0 +1,5 @@
1
+ ActiveRecord::Base.connection.create_table :taggings, :force => true do |t|
2
+ t.column :tag_id, :integer
3
+ t.column :taggable_id, :integer, :null => false
4
+ t.column :taggable_type, :string, :null => false
5
+ end
@@ -0,0 +1,4 @@
1
+ ActiveRecord::Base.connection.create_table :tags, :force => true do |t|
2
+ # t.column :post_id, :integer, :null => false
3
+ t.column :text, :text, :null => true
4
+ end
@@ -0,0 +1,21 @@
1
+ require 'rubygems'
2
+ require 'cucumber'
3
+ require 'spec'
4
+ require 'fileutils'
5
+ require 'ginger'
6
+ require 'will_paginate'
7
+ require 'active_record'
8
+
9
+ $:.unshift File.dirname(__FILE__) + '/../../lib'
10
+ Dir[File.join(File.dirname(__FILE__), '../../vendor/*/lib')].each do |path|
11
+ $:.unshift path
12
+ end
13
+
14
+ require 'cucumber/thinking_sphinx/internal_world'
15
+
16
+ world = Cucumber::ThinkingSphinx::InternalWorld.new
17
+ world.configure_database
18
+
19
+ require "thinking_sphinx"
20
+
21
+ world.setup
@@ -0,0 +1,8 @@
1
+ class GenericDeltaHandler < ThinkingSphinx::Deltas::DefaultDelta
2
+
3
+ def index(model, instance = nil)
4
+ #do nothing but set a bit for every record
5
+ #this is just a demonstration of extensibility
6
+ model.update_all(:changed_by_generic => true)
7
+ end
8
+ end
@@ -0,0 +1,21 @@
1
+ class Alpha < ActiveRecord::Base
2
+ define_index do
3
+ indexes :name, :sortable => true
4
+
5
+ has value, created_at, created_on
6
+ has cost, :facet => true
7
+
8
+ set_property :field_weights => {"name" => 10}
9
+ end
10
+
11
+ define_index 'alternative' do
12
+ indexes :name, :as => :alternative_name, :sortable => true
13
+
14
+ has value, created_at, created_on
15
+ has cost, :facet => true
16
+
17
+ set_property :field_weights => {'alternative_name' => 10}
18
+
19
+ where "value > 3"
20
+ end
21
+ end
@@ -0,0 +1,5 @@
1
+ class Animal < ActiveRecord::Base
2
+ define_index do
3
+ indexes name, :sortable => true
4
+ end
5
+ end
@@ -0,0 +1,3 @@
1
+ class Author < ActiveRecord::Base
2
+ #
3
+ end
@@ -0,0 +1,8 @@
1
+ class Beta < ActiveRecord::Base
2
+ define_index do
3
+ indexes :name, :sortable => true
4
+ has value
5
+
6
+ set_property :delta => true
7
+ end
8
+ end
@@ -0,0 +1,8 @@
1
+ class Box < ActiveRecord::Base
2
+ define_index do
3
+ indexes width, :as => :width_field
4
+
5
+ has width, length, depth
6
+ has [width, length, depth], :as => :dimensions
7
+ end
8
+ end
@@ -0,0 +1,3 @@
1
+ class Cat < Animal
2
+ #
3
+ end
@@ -0,0 +1,4 @@
1
+ class Category < ActiveRecord::Base
2
+ has_many :posts
3
+ has_many :comments
4
+ end
@@ -0,0 +1,10 @@
1
+ class Comment < ActiveRecord::Base
2
+ define_index do
3
+ indexes :content
4
+
5
+ has category.name, :facet => true, :as => :category_name, :type => :string
6
+ end
7
+
8
+ belongs_to :post
9
+ belongs_to :category
10
+ end
@@ -0,0 +1,16 @@
1
+ require 'features/support/models/tag'
2
+ require 'features/support/models/tagging'
3
+
4
+ class Developer < ActiveRecord::Base
5
+ has_many :taggings, :as => :taggable
6
+ has_many :tags, :through => :taggings
7
+
8
+ define_index do
9
+ indexes country, :facet => true
10
+ indexes state, :facet => true
11
+ indexes tags.text, :as => :tags, :facet => true
12
+ has age, :facet => true
13
+ has tags(:id), :as => :tag_ids, :facet => true
14
+ facet city
15
+ end
16
+ end
@@ -0,0 +1,3 @@
1
+ class Dog < Animal
2
+ #
3
+ end
@@ -0,0 +1,9 @@
1
+ require File.join(File.dirname(__FILE__), "..", "lib", "generic_delta_handler")
2
+
3
+ class ExtensibleBeta < ActiveRecord::Base
4
+ define_index do
5
+ indexes :name, :sortable => true
6
+
7
+ set_property :delta => "GenericDeltaHandler"
8
+ end
9
+ end
@@ -0,0 +1,5 @@
1
+ class Fox < Animal
2
+ define_index do
3
+ indexes :name
4
+ end
5
+ end
@@ -0,0 +1,5 @@
1
+ class Gamma < ActiveRecord::Base
2
+ define_index do
3
+ indexes :name, :sortable => true
4
+ end
5
+ end
@@ -0,0 +1,23 @@
1
+ class Person < ActiveRecord::Base
2
+ define_index do
3
+ indexes first_name, last_name, :sortable => true
4
+
5
+ has [first_name, middle_initial, last_name], :as => :name_sort
6
+ has birthday
7
+ has gender, :facet => true
8
+
9
+ set_property :min_infix_len => 1
10
+ set_property :enable_star => true
11
+ end
12
+
13
+ sphinx_scope(:with_first_name) { |name|
14
+ { :conditions => {:first_name => name} }
15
+ }
16
+ sphinx_scope(:with_last_name) { |name|
17
+ { :conditions => {:last_name => name} }
18
+ }
19
+ sphinx_scope(:with_id) { |id|
20
+ { :with => {:sphinx_internal_id => id} }
21
+ }
22
+ sphinx_scope(:ids_only) { {:ids_only => true} }
23
+ end
@@ -0,0 +1,21 @@
1
+ class Post < ActiveRecord::Base
2
+ has_many :comments, :dependent => :destroy
3
+ has_many :taggings, :as => :taggable
4
+ has_many :tags, :through => :taggings
5
+ belongs_to :category
6
+ has_and_belongs_to_many :authors
7
+
8
+ define_index do
9
+ indexes subject
10
+ indexes content
11
+ indexes tags.text, :as => :tags
12
+ indexes comments.content, :as => :comments
13
+ indexes authors.name, :as => :authors
14
+
15
+ has comments(:id), :as => :comment_ids, :source => :ranged_query,
16
+ :facet => true
17
+ has category.name, :facet => true, :as => :category_name, :type => :string
18
+ has 'COUNT(DISTINCT comments.id)', :as => :comments_count, :type => :integer
19
+ has comments.created_at, :as => :comments_created_at
20
+ end
21
+ end
@@ -0,0 +1,12 @@
1
+ class Robot < ActiveRecord::Base
2
+ set_primary_key :internal_id
3
+ set_sphinx_primary_key :alternate_primary_key
4
+
5
+ define_index do
6
+ indexes :name
7
+ end
8
+
9
+ def id
10
+ internal_id
11
+ end
12
+ end
@@ -0,0 +1,3 @@
1
+ class Tag < ActiveRecord::Base
2
+ belongs_to :post
3
+ end
@@ -0,0 +1,4 @@
1
+ class Tagging < ActiveRecord::Base
2
+ belongs_to :tag
3
+ belongs_to :taggable, :polymorphic => true
4
+ end
@@ -0,0 +1,5 @@
1
+ require 'thinking_sphinx/xml/source'
2
+
3
+ require 'thinking_sphinx/xml/adapters/abstract_adapter'
4
+ require 'thinking_sphinx/xml/adapters/sqlite3_adapter'
5
+ require 'thinking_sphinx/xml/adapters/oracle_adapter'
@@ -0,0 +1,34 @@
1
+ module ThinkingSphinx
2
+ class AbstractAdapter
3
+ def self.detect(model)
4
+ case model.connection.class.name
5
+ when "ActiveRecord::ConnectionAdapters::MysqlAdapter",
6
+ "ActiveRecord::ConnectionAdapters::MysqlplusAdapter"
7
+ ThinkingSphinx::MysqlAdapter.new model
8
+ when "ActiveRecord::ConnectionAdapters::PostgreSQLAdapter"
9
+ ThinkingSphinx::PostgreSQLAdapter.new model
10
+ when "ActiveRecord::ConnectionAdapters::SQLite3Adapter"
11
+ ThinkingSphinx::SQLite3Adapter.new model
12
+ when "ActiveRecord::ConnectionAdapters::OracleAdapter",
13
+ "ActiveRecord::ConnectionAdapters::OracleEnhancedAdapter"
14
+ ThinkingSphinx::OracleAdapter.new model
15
+ when "ActiveRecord::ConnectionAdapters::JdbcAdapter"
16
+ if model.connection.config[:adapter] == "jdbcmysql"
17
+ ThinkingSphinx::MysqlAdapter.new model
18
+ elsif model.connection.config[:adapter] == "jdbcpostgresql"
19
+ ThinkingSphinx::PostgreSQLAdapter.new model
20
+ else
21
+ raise "Invalid Database Adapter: Sphinx only supports MySQL and PostgreSQL"
22
+ end
23
+ else
24
+ raise "Invalid Database Adapter: Sphinx only supports MySQL and PostgreSQL, not #{model.connection.class.name}"
25
+ end
26
+ end
27
+
28
+ def select_each(query)
29
+ connection.select_all(query).each do |values|
30
+ yield values
31
+ end
32
+ end
33
+ end
34
+ end