gummi 0.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
data/.gitignore ADDED
@@ -0,0 +1,18 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ *.DS_Store
7
+ Gemfile.lock
8
+ InstalledFiles
9
+ _yardoc
10
+ coverage
11
+ doc/
12
+ lib/bundler/man
13
+ pkg
14
+ rdoc
15
+ spec/reports
16
+ test/tmp
17
+ test/version_tmp
18
+ tmp
data/Gemfile ADDED
@@ -0,0 +1,6 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in gummi.gemspec
4
+ gemspec
5
+
6
+ gem "codeclimate-test-reporter", group: :test, require: nil
data/LICENSE.txt ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2013 Jens Norrgrann
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,31 @@
1
+ [![Code Climate](https://codeclimate.com/github/bukowskis/gummi.png)](https://codeclimate.com/github/bukowskis/gummi)
2
+ # Gummi
3
+
4
+ A minimal wrapper around elasticsearch-ruby using a repository pattern.
5
+ Still very much alpha. Use at your own risk...
6
+
7
+ ## Installation
8
+
9
+ Add this line to your application's Gemfile:
10
+
11
+ gem 'gummi'
12
+
13
+ And then execute:
14
+
15
+ $ bundle
16
+
17
+ Or install it yourself as:
18
+
19
+ $ gem install gummi
20
+
21
+ ## Usage
22
+
23
+ TODO: Write usage instructions here
24
+
25
+ ## Contributing
26
+
27
+ 1. Fork it
28
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
29
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
30
+ 4. Push to the branch (`git push origin my-new-feature`)
31
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1 @@
1
+ require "bundler/gem_tasks"
data/gummi.gemspec ADDED
@@ -0,0 +1,30 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'gummi/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "gummi"
8
+ spec.version = Gummi::VERSION
9
+ spec.authors = ["bukowskis"]
10
+ spec.description = %q{A small wrapper around Elasticsearch}
11
+ spec.summary = %q{A small wrapper around Elasticsearch}
12
+ spec.homepage = ""
13
+ spec.license = "MIT"
14
+
15
+ spec.files = `git ls-files`.split($/) - ['.travis.yml']
16
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
17
+ spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
18
+ spec.require_paths = ["lib"]
19
+
20
+ spec.add_dependency('virtus', '~> 1.0.0')
21
+ spec.add_dependency('elasticsearch', '~> 0.4.0')
22
+ spec.add_dependency('activesupport', '>= 3.0')
23
+ spec.add_dependency('activemodel', '>= 3.0')
24
+ spec.add_dependency('hooks', '~>0.3.3')
25
+ spec.add_dependency('leaflet')
26
+
27
+ spec.add_development_dependency('bundler', '~> 1.3')
28
+ spec.add_development_dependency('rake')
29
+ spec.add_development_dependency('rspec')
30
+ end
data/lib/gummi.rb ADDED
@@ -0,0 +1,42 @@
1
+ require 'virtus'
2
+ require 'elasticsearch'
3
+ require 'active_support/core_ext'
4
+ require 'active_model'
5
+ require 'hooks'
6
+ require 'leaflet'
7
+
8
+ require "repobahn/repository"
9
+ require "repobahn/entity"
10
+
11
+ require "gummi/version"
12
+ require "gummi/api"
13
+ require "gummi/attributes"
14
+ require "gummi/document"
15
+ require "gummi/entity"
16
+ require "gummi/index"
17
+ require "gummi/object"
18
+ require "gummi/repository"
19
+ require "gummi/fields/boolean"
20
+ require "gummi/fields/time"
21
+ require "gummi/fields/integer"
22
+ require "gummi/fields/positive_integer"
23
+ require "gummi/fields/keyword"
24
+ require "gummi/fields/ngram_and_plain"
25
+ require "gummi/fields/path_hierarchy"
26
+ require "gummi/fields/string"
27
+ require "gummi/fields/sanitized_string"
28
+ require "gummi/default_index"
29
+ require "gummi/search/searching"
30
+ require "gummi/search/filtered"
31
+ require "gummi/search/raw"
32
+ require "gummi/search/result"
33
+
34
+ module Gummi
35
+ def self.env
36
+ if defined? Rails
37
+ Rails.env
38
+ else
39
+ RAILS_ENV || "development"
40
+ end
41
+ end
42
+ end
data/lib/gummi/api.rb ADDED
@@ -0,0 +1,7 @@
1
+ module Gummi
2
+ module API
3
+ def self.client
4
+ @client ||= ::Elasticsearch::Client.new log: true
5
+ end
6
+ end
7
+ end
@@ -0,0 +1,26 @@
1
+ module Gummi
2
+ module Attributes
3
+ extend ActiveSupport::Concern
4
+
5
+ module ClassMethods
6
+
7
+ def mapping_for_attribute(attribute)
8
+ if attribute.is_a? Virtus::Attribute::EmbeddedValue
9
+ {properties: attribute.primitive.mapping}
10
+ elsif attribute.is_a? Virtus::Attribute::Collection
11
+ mapping_for_attribute(attribute.member_type)
12
+ else
13
+ attribute.mapping
14
+ end
15
+ end
16
+
17
+ def mapping
18
+ result = {}
19
+ attribute_set.each do |attribute|
20
+ result.merge!({ attribute.name => mapping_for_attribute(attribute)})
21
+ end
22
+ result
23
+ end
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,13 @@
1
+ module Gummi
2
+ class DefaultIndex
3
+ include Gummi::Index
4
+
5
+ def self.name
6
+ if defined? Rails
7
+ "#{Rails.application.class.name.deconstantize.underscore}_#{Rails.env}"
8
+ else
9
+ "gummi_#{Gummi.env}"
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,118 @@
1
+ module Gummi
2
+ module Document
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ include Virtus.model
7
+ include Gummi::Attributes
8
+ end
9
+
10
+ attr_accessor :id
11
+ attr_accessor :version
12
+
13
+ def overwrite
14
+ response = client.index index: index.name, type: document_type, id: id, body: attributes
15
+ if response["ok"]
16
+ self.version = response["_version"]
17
+ self.id = response["_id"]
18
+ true
19
+ else
20
+ false
21
+ end
22
+ end
23
+
24
+ def update
25
+ response = client.update index: index.name, type: document_type, id: id, retry_on_conflict: 0, version: version, body: { doc: attributes.as_json }
26
+ if response["ok"]
27
+ self.version = response["_version"]
28
+ true
29
+ else
30
+ false
31
+ end
32
+ end
33
+
34
+ private
35
+
36
+ def client
37
+ Gummi::API.client
38
+ end
39
+
40
+ def document_type
41
+ self.class.document_type
42
+ end
43
+
44
+ def index
45
+ self.class.index
46
+ end
47
+
48
+ module ClassMethods
49
+
50
+ def get!(id)
51
+ response = client.get index: index.name, type: document_type, id: id
52
+ doc_hash = {id: response["_id"], version: response["_version"]}.merge(response["_source"])
53
+ lot = self.new(doc_hash)
54
+ lot.version = response["_version"]
55
+ lot
56
+ end
57
+
58
+ def get(id)
59
+ get!(id)
60
+ rescue ::Elasticsearch::Transport::Transport::Errors::NotFound
61
+ nil
62
+ end
63
+
64
+ def document_type(type_name)
65
+ @document_type = type_name
66
+ end
67
+
68
+ def document_type
69
+ @document_type || name.split('::').last.underscore
70
+ end
71
+
72
+ def index
73
+ @index || Gummi::DefaultIndex
74
+ end
75
+
76
+ def index=(index)
77
+ @index = index
78
+ end
79
+
80
+ def parent_document_type
81
+ nil
82
+ end
83
+
84
+ def sync_mapping!
85
+ client.indices.put_mapping creation_options
86
+ end
87
+
88
+ def new_filtered_search(options = {})
89
+ args = {}
90
+ args[:index] = index.name
91
+ args[:type] = document_type
92
+ args.merge! options
93
+
94
+ Gummi::Search::Filtered.new args
95
+ end
96
+
97
+ def creation_options
98
+ result = {
99
+ index: index.name,
100
+ type: document_type,
101
+ body: {
102
+ document_type => {
103
+ properties: mapping,
104
+ }
105
+ }
106
+ }
107
+ result[:body][document_type].merge!(_parent: { type: parent_document_type }) if parent_document_type.present?
108
+ result
109
+ end
110
+
111
+ private
112
+
113
+ def client
114
+ Gummi::API.client
115
+ end
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,12 @@
1
+ module Gummi
2
+ module Entity
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ include Repobahn::Entity
7
+ end
8
+
9
+ attr_accessor :id
10
+ attr_accessor :version
11
+ end
12
+ end
@@ -0,0 +1,10 @@
1
+ module Gummi
2
+ module Fields
3
+ class Boolean < Virtus::Attribute::Boolean
4
+
5
+ def mapping
6
+ { type: 'boolean' }
7
+ end
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,14 @@
1
+ module Gummi
2
+ module Fields
3
+ class Integer < Virtus::Attribute
4
+
5
+ def coerce(value)
6
+ value.to_i if value.present?
7
+ end
8
+
9
+ def mapping
10
+ { type: 'integer' }
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,13 @@
1
+ module Gummi
2
+ module Fields
3
+ class Keyword < Virtus::Attribute
4
+ def coerce(value)
5
+ value
6
+ end
7
+
8
+ def mapping
9
+ { type: 'string', index_analyzer: 'keyword_index_analyzer', search_analyzer: 'keyword_search_analyzer' }
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,18 @@
1
+ module Gummi
2
+ module Fields
3
+ class NgramAndPlain < Virtus::Attribute
4
+ def coerce(value)
5
+ value
6
+ end
7
+
8
+ def mapping
9
+ { type: 'multi_field',
10
+ fields: {
11
+ name => { type: 'string', index_analyzer: 'text_index_analyzer', search_analyzer: 'text_search_analyzer' },
12
+ :plain => { type: 'string', index_analyzer: 'string_index_analyzer', search_analyzer: 'text_search_analyzer' },
13
+ }
14
+ }
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,13 @@
1
+ module Gummi
2
+ module Fields
3
+ class PathHierarchy < Virtus::Attribute
4
+ def coerce(value)
5
+ value
6
+ end
7
+
8
+ def mapping
9
+ {type: 'string', index_analyzer: 'path_hierarchy_analyzer' }
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,19 @@
1
+ module Gummi
2
+ module Fields
3
+ class PositiveInteger < Virtus::Attribute
4
+
5
+ def coerce(value)
6
+ coerced = value.to_i
7
+ if coerced > 0
8
+ coerced
9
+ else
10
+ default_value.value
11
+ end
12
+ end
13
+
14
+ def mapping
15
+ { type: 'integer' }
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,28 @@
1
+ module Gummi
2
+ module Fields
3
+ class SanitizedString < Virtus::Attribute
4
+
5
+ def coerce(value)
6
+ return nil if value.blank?
7
+ sanitize_string_for_query(value.to_s)
8
+ end
9
+
10
+ def mapping
11
+ { type: 'string' }
12
+ end
13
+
14
+ def sanitize_string_for_query(str)
15
+ # Escape special characters
16
+ escaped_characters = Regexp.escape('\/\\+-&|!(){}[]^~*?:')
17
+ str = str.gsub(/([#{escaped_characters}])/) do |match|
18
+ '\\'+match
19
+ end
20
+
21
+ # Escape odd quotes
22
+ quote_count = str.count '"'
23
+ str = str.gsub(/(.*)"(.*)/, '\1\"\3') if quote_count % 2 == 1
24
+ str
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,15 @@
1
+ module Gummi
2
+ module Fields
3
+ class String < Virtus::Attribute
4
+
5
+
6
+ def coerce(value)
7
+ value
8
+ end
9
+
10
+ def mapping
11
+ { type: 'string' }
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,15 @@
1
+ module Gummi
2
+ module Fields
3
+ class Time < Virtus::Attribute
4
+
5
+ def coerce(value)
6
+ return nil unless value.respond_to? :in_time_zone
7
+ value.in_time_zone 'UTC'
8
+ end
9
+
10
+ def mapping
11
+ {type: "date"}
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,146 @@
1
+ module Gummi
2
+ module Index
3
+ extend ActiveSupport::Concern
4
+
5
+ module ClassMethods
6
+
7
+ # Return true if created or false if already created.
8
+ #
9
+ def setup
10
+ created_settings = client.indices.create index: name, body: { settings: settings }
11
+ created_settings.present?
12
+ refresh
13
+ rescue ::Elasticsearch::Transport::Transport::Errors::BadRequest => exception
14
+ false
15
+ end
16
+
17
+ # Return true if successful or already teared down.
18
+ #
19
+ # Raises NotImplementedError in production.
20
+ #
21
+ def teardown
22
+ raise NotImplementedError unless Gummi.env == 'development' || Gummi.env == 'test'
23
+ response = client.indices.delete index: name
24
+ response.present?
25
+ rescue ::Elasticsearch::Transport::Transport::Errors::NotFound
26
+ true
27
+ end
28
+
29
+ def name
30
+ raise "Implement me"
31
+ end
32
+
33
+ def refresh
34
+ client.indices.refresh
35
+ client.cluster.health wait_for_status: :yellow
36
+ end
37
+
38
+ def settings
39
+ default_settings
40
+ end
41
+
42
+ def default_settings
43
+ {
44
+ index: {
45
+ # Main Settings
46
+ number_of_shards: '3',
47
+ number_of_replicas: (Gummi.env == 'production' ? '2' : '0'),
48
+ refresh_interval: '1s',
49
+ store: { type: (Gummi.env == 'test' ? :memory : :niofs) },
50
+ mapper: { dynamic: false },
51
+
52
+ analysis: {
53
+
54
+ # Tokenizers are just some sort of "tool" or "module" that can be applied to analyzers.
55
+ tokenizer: {
56
+ # This one is a little bit more general and is able to chop any word into all of its components.
57
+ ngram_tokenizer: {
58
+ type: 'nGram',
59
+ min_gram: 1,
60
+ max_gram: 7,
61
+ token_chars: [ 'letter', 'digit' ],
62
+ }
63
+
64
+ },
65
+
66
+ # Now we are ready to use our tokenizers.
67
+ # Let's create the most important thing: Analyzers.
68
+ analyzer: {
69
+
70
+ path_hierarchy_analyzer: {
71
+ type: 'custom',
72
+ tokenizer: 'path_hierarchy',
73
+ },
74
+ # When adding long text to Elastic, we most likely are going to use this
75
+ # analyzer. This is commonly used for titles and descriptions.
76
+ text_index_analyzer: {
77
+ type: 'custom',
78
+ tokenizer: 'ngram_tokenizer', # Chopping every word up into tokens
79
+ filter: {
80
+ 0 => 'standard', # Some default transformations
81
+ 1 => 'lowercase', # Make everything lowercase
82
+ 2 => 'word_delimiter', # E.g. "O'Neil" -> "O Neil", "Victoria's" -> "Victoria"
83
+ 2 => 'asciifolding', # Transform everything into ASCII
84
+ },
85
+ },
86
+
87
+ # For smaller texts, such as the city "stockholm", we don't want any
88
+ # tokenizing. It's enough to explicitly save the word as it is.
89
+ # As a matter of fact, if we would tokenize the city, then the facets
90
+ # would report that we have Transports in "st" "sto" "stoc" etc.
91
+ string_index_analyzer: {
92
+ type: 'custom',
93
+ tokenizer: 'standard',
94
+ filter: {
95
+ # The filters, however, are identical to the other analyzer.
96
+ 0 => 'standard',
97
+ 1 => 'lowercase',
98
+ 2 => 'word_delimiter',
99
+ 3 => 'asciifolding',
100
+ },
101
+ },
102
+
103
+ # For finding Slugs
104
+ keyword_index_analyzer: {
105
+ type: 'custom',
106
+ tokenizer: 'keyword',
107
+ filter: {
108
+ 0 => 'lowercase',
109
+ 1 => 'asciifolding',
110
+ },
111
+ },
112
+
113
+ # This is an analyzer that we apply to the search query itself.
114
+ text_search_analyzer: {
115
+ type: 'custom',
116
+ tokenizer: 'standard',
117
+ filter: {
118
+ 0 => 'standard',
119
+ 1 => 'lowercase',
120
+ 2 => 'word_delimiter',
121
+ 3 => 'asciifolding',
122
+ },
123
+ },
124
+
125
+ # This is an analyzer that we apply to the search query itself.
126
+ keyword_search_analyzer: {
127
+ type: 'custom',
128
+ tokenizer: 'keyword',
129
+ filter: {
130
+ 0 => 'lowercase',
131
+ 1 => 'asciifolding',
132
+ },
133
+ },
134
+
135
+ }
136
+ }
137
+ }
138
+ }
139
+ end
140
+
141
+ def client
142
+ Gummi::API.client
143
+ end
144
+ end
145
+ end
146
+ end
@@ -0,0 +1,10 @@
1
+ module Gummi
2
+ module Object
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ include Virtus.model
7
+ include Gummi::Attributes
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,31 @@
1
+
2
+ module Gummi
3
+ module Repository
4
+ extend ActiveSupport::Concern
5
+
6
+ included do
7
+ include Repobahn::Repository
8
+ after_conversion :set_id_and_version
9
+ end
10
+
11
+
12
+ module ClassMethods
13
+
14
+ def get(id)
15
+ record = db_model.get id
16
+ to_entity_from_db record if record
17
+ end
18
+
19
+ def overwrite(entity)
20
+ return false unless entity.valid?
21
+ db_record = db_model.new(entity.attributes)
22
+ db_record.overwrite
23
+ end
24
+
25
+ def set_id_and_version(entity, db)
26
+ entity.id = db.id
27
+ entity.version = db.version
28
+ end
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,36 @@
1
+ module Gummi
2
+ module Search
3
+ class Filtered
4
+ include Gummi::Search::Searching
5
+
6
+ attribute :query_string, Gummi::Fields::SanitizedString
7
+ attribute :query_filters, Array[Hash], default: []
8
+ attribute :facets, Hash, default: {}
9
+
10
+ def to_client_args
11
+ args = {}
12
+ args[:index] = index
13
+ args[:type] = type if type
14
+ args[:from] = from
15
+ args[:body] = {query: filtered, facets: facets }
16
+ args
17
+ end
18
+
19
+ def query
20
+ {query_string: { query: query_string}} if query_string.present?
21
+ end
22
+
23
+ def filtered
24
+ { 'filtered' => { 'query' => query, 'filter' => process_query_filters }}
25
+ end
26
+
27
+ def process_query_filters
28
+ if query_filters.length > 1
29
+ {and: query_filters}
30
+ else
31
+ query_filters.first
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,7 @@
1
+ module Gummi
2
+ module Search
3
+ class Raw
4
+ include Gummi::Search::Searching
5
+ end
6
+ end
7
+ end
@@ -0,0 +1,23 @@
1
+ module Gummi
2
+ module Search
3
+ class Result
4
+
5
+ attr_reader :took, :total, :hits, :facets
6
+
7
+ def initialize(result)
8
+ @took = result["took"]
9
+ @total = result["hits"]["total"]
10
+ @hits = result["hits"]["hits"]
11
+ @facets = result["facets"]
12
+ end
13
+
14
+ def records
15
+ hits.map do |hit|
16
+ model = "DB::#{hit["_type"].humanize}".constantize
17
+ doc_hash = {id: hit["_id"]}.merge(hit["_source"])
18
+ model.new(doc_hash)
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,43 @@
1
+ module Gummi
2
+ module Search
3
+ module Searching
4
+ extend ActiveSupport::Concern
5
+
6
+ included do
7
+ include Virtus.model
8
+
9
+ attribute :type, String
10
+ attribute :index, String, default: lambda {|search, attr| Gummi::DefaultIndex.name}
11
+ attribute :page, Gummi::Fields::PositiveInteger, default: 1
12
+ attribute :per_page, Gummi::Fields::PositiveInteger, default: 300
13
+ attribute :options, Hash, default: {}
14
+ end
15
+
16
+ def size
17
+ per_page
18
+ end
19
+
20
+ def from
21
+ per_page * (page - 1)
22
+ end
23
+
24
+ def execute
25
+ Gummi::Search::Result.new client.search(to_client_args)
26
+ end
27
+
28
+ def to_client_args
29
+ args = {}
30
+ args[:index] = index
31
+ args[:type] = type if type
32
+ args[:from] = from
33
+ args[:size] = size
34
+ args.merge options
35
+ end
36
+
37
+ private
38
+ def client
39
+ Gummi::API.client
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,3 @@
1
+ module Gummi
2
+ VERSION = "0.0.6"
3
+ end
@@ -0,0 +1,12 @@
1
+ module Repobahn
2
+ module Entity
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ include Virtus.model
7
+ include ActiveModel::Conversion
8
+ extend ActiveModel::Naming
9
+ end
10
+
11
+ end
12
+ end
@@ -0,0 +1,57 @@
1
+ module Repobahn
2
+ module Repository
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ include Hooks
7
+ define_hook :after_conversion
8
+ end
9
+
10
+
11
+ module ClassMethods
12
+
13
+ def entity_model
14
+ @entity_model || default_entity_model
15
+ end
16
+
17
+ def entity_model=(klass)
18
+ @entity_model = klass
19
+ end
20
+
21
+ def db_model
22
+ @db_model || default_db_model
23
+ end
24
+
25
+ def db_model=(klass)
26
+ @db_model = klass
27
+ end
28
+
29
+ def find(id)
30
+ record = db_model.find id
31
+ to_entity_from_db record if record
32
+ end
33
+
34
+ def to_entity_from_db(records)
35
+ entities = Array(records).map do |record|
36
+ entity = entity_model.new(record.attributes)
37
+ run_hook :after_conversion, entity, record
38
+ entity
39
+ end
40
+ entities.length > 1 ? entities : entities.first
41
+ end
42
+
43
+ private
44
+
45
+ def default_entity_model
46
+ full_name = name.split('::')
47
+ model_name = full_name.pop.singularize
48
+ full_name << model_name
49
+ full_name.join('::').constantize
50
+ end
51
+
52
+ def default_db_model
53
+ "DB::#{name.split('::').last.singularize}".constantize
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,74 @@
1
+ require 'spec_helper'
2
+
3
+ class ExampleModel
4
+ include Gummi::Document
5
+
6
+ attribute :test, String
7
+ end
8
+
9
+ describe Gummi::Document do
10
+
11
+ context "included" do
12
+ it "should add accessors for id and version" do
13
+ m = ExampleModel.new
14
+ m.respond_to?(:version).should be_true
15
+ m.respond_to?(:id).should be_true
16
+ end
17
+
18
+ it "should add attributes methods" do
19
+ m = ExampleModel.new(test: 'hello')
20
+ m.test.should == 'hello'
21
+ end
22
+ end
23
+
24
+ context "attributes" do
25
+
26
+ context "date times" do
27
+ it "should coerce from elastics strings to real Time" do
28
+ time = Time.now
29
+ person = DB::Person.new(born_at: time)
30
+ person.overwrite
31
+ person_from_es = DB::Person.get person.id
32
+ person_from_es.born_at.should be_a Time
33
+ end
34
+
35
+ it "should always store time in UTC" do
36
+ time = Time.now.in_time_zone 'CET'
37
+
38
+ person = DB::Person.new(born_at: time)
39
+ person.born_at.zone.should == 'UTC'
40
+ end
41
+ end
42
+
43
+ context "computed_attributes" do
44
+
45
+ let(:person) { DB::Person.new}
46
+
47
+ it "should add them to the attributes hash" do
48
+ person.name = "olof palme"
49
+ person.attributes[:computed_name].should == 'OLOF PALME'
50
+ end
51
+
52
+ it "should compute every time" do
53
+ person.name = "olof palme"
54
+ person.computed_name.should == person.name.upcase
55
+ person.name = "carl bildt"
56
+ person.computed_name.should == person.name.upcase
57
+ end
58
+
59
+ it "should provide a mapping" do
60
+ DB::Person.mapping.should include(:computed_name => {:type=>"string"})
61
+ end
62
+ end
63
+ end
64
+
65
+ context 'getting from elastic' do
66
+ let (:person) { DB::Person.new(name: 'Buzz Lightyear') }
67
+
68
+ it "should return an instance of the db_model" do
69
+ person.overwrite
70
+ person_from_es = DB::Person.get(person.id)
71
+ person_from_es.should be_a DB::Person
72
+ end
73
+ end
74
+ end
@@ -0,0 +1,57 @@
1
+ require 'spec_helper'
2
+
3
+ describe Gummi::Repository do
4
+
5
+ let(:repository_model) { People }
6
+ let(:entity_model) { Person }
7
+ let(:db_model) { DB::Person }
8
+
9
+ describe ".entity_model" do
10
+ it "should default to singular version of repository_model" do
11
+ People.entity_model.should == Person
12
+ end
13
+ end
14
+
15
+ describe ".db_model" do
16
+ it "should default to singular version of repository_model in the DB namespace" do
17
+ People.db_model.should == DB::Person
18
+ end
19
+ end
20
+
21
+ context "converting from db to entity" do
22
+
23
+ let (:db_person) { DB::Person.new(name: 'Buzz Lightyear') }
24
+
25
+ it "should map the attributes from db to entity" do
26
+ person = People.to_entity_from_db(db_person)
27
+ person.name.should == 'Buzz Lightyear'
28
+ end
29
+
30
+ it "should run hook for after_conversion" do
31
+ person = People.to_entity_from_db(db_person)
32
+ person.converted_name.should == db_person.name.reverse
33
+ end
34
+ end
35
+
36
+ describe ".get" do
37
+ context "existing record" do
38
+
39
+ let (:db_person) { DB::Person.new(name: 'Buzz Lightyear') }
40
+
41
+ it "should return an entity" do
42
+ db_person.overwrite
43
+ person = People.get(db_person.id)
44
+ person.id.should == db_person.id
45
+ end
46
+ end
47
+
48
+ context "missing record" do
49
+ it "returns nil" do
50
+ person = People.get('missing_id')
51
+ person.should be_nil
52
+ end
53
+ end
54
+ end
55
+
56
+
57
+ end
@@ -0,0 +1,15 @@
1
+ # example db model
2
+
3
+ module DB
4
+ class Person
5
+ include Gummi::Document
6
+
7
+ attribute :name, Gummi::Fields::String
8
+ attribute :computed_name, Gummi::Fields::String
9
+ attribute :born_at, Gummi::Fields::Time
10
+
11
+ def computed_name
12
+ name.upcase if name
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,9 @@
1
+ class People
2
+ include Gummi::Repository
3
+
4
+ after_conversion :convert_name
5
+
6
+ def self.convert_name(entity, db)
7
+ entity.converted_name = db.name.reverse
8
+ end
9
+ end
@@ -0,0 +1,7 @@
1
+ # example entity model
2
+ class Person
3
+ include Gummi::Entity
4
+
5
+ attribute :name, String
6
+ attribute :converted_name, String
7
+ end
@@ -0,0 +1,19 @@
1
+ require "codeclimate-test-reporter"
2
+ CodeClimate::TestReporter.start
3
+
4
+ RAILS_ENV = 'test'
5
+ require 'gummi'
6
+ require_relative 'models/people'
7
+ require_relative 'models/person'
8
+ require_relative 'models/db/person'
9
+
10
+ RSpec.configure do |config|
11
+ config.before(:suite) do
12
+ Gummi::DefaultIndex.setup
13
+ DB::Person.sync_mapping!
14
+ end
15
+
16
+ config.after(:suite) do
17
+ Gummi::DefaultIndex.teardown
18
+ end
19
+ end
metadata ADDED
@@ -0,0 +1,232 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: gummi
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.6
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - bukowskis
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2013-11-18 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: virtus
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ~>
20
+ - !ruby/object:Gem::Version
21
+ version: 1.0.0
22
+ type: :runtime
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ~>
28
+ - !ruby/object:Gem::Version
29
+ version: 1.0.0
30
+ - !ruby/object:Gem::Dependency
31
+ name: elasticsearch
32
+ requirement: !ruby/object:Gem::Requirement
33
+ none: false
34
+ requirements:
35
+ - - ~>
36
+ - !ruby/object:Gem::Version
37
+ version: 0.4.0
38
+ type: :runtime
39
+ prerelease: false
40
+ version_requirements: !ruby/object:Gem::Requirement
41
+ none: false
42
+ requirements:
43
+ - - ~>
44
+ - !ruby/object:Gem::Version
45
+ version: 0.4.0
46
+ - !ruby/object:Gem::Dependency
47
+ name: activesupport
48
+ requirement: !ruby/object:Gem::Requirement
49
+ none: false
50
+ requirements:
51
+ - - ! '>='
52
+ - !ruby/object:Gem::Version
53
+ version: '3.0'
54
+ type: :runtime
55
+ prerelease: false
56
+ version_requirements: !ruby/object:Gem::Requirement
57
+ none: false
58
+ requirements:
59
+ - - ! '>='
60
+ - !ruby/object:Gem::Version
61
+ version: '3.0'
62
+ - !ruby/object:Gem::Dependency
63
+ name: activemodel
64
+ requirement: !ruby/object:Gem::Requirement
65
+ none: false
66
+ requirements:
67
+ - - ! '>='
68
+ - !ruby/object:Gem::Version
69
+ version: '3.0'
70
+ type: :runtime
71
+ prerelease: false
72
+ version_requirements: !ruby/object:Gem::Requirement
73
+ none: false
74
+ requirements:
75
+ - - ! '>='
76
+ - !ruby/object:Gem::Version
77
+ version: '3.0'
78
+ - !ruby/object:Gem::Dependency
79
+ name: hooks
80
+ requirement: !ruby/object:Gem::Requirement
81
+ none: false
82
+ requirements:
83
+ - - ~>
84
+ - !ruby/object:Gem::Version
85
+ version: 0.3.3
86
+ type: :runtime
87
+ prerelease: false
88
+ version_requirements: !ruby/object:Gem::Requirement
89
+ none: false
90
+ requirements:
91
+ - - ~>
92
+ - !ruby/object:Gem::Version
93
+ version: 0.3.3
94
+ - !ruby/object:Gem::Dependency
95
+ name: leaflet
96
+ requirement: !ruby/object:Gem::Requirement
97
+ none: false
98
+ requirements:
99
+ - - ! '>='
100
+ - !ruby/object:Gem::Version
101
+ version: '0'
102
+ type: :runtime
103
+ prerelease: false
104
+ version_requirements: !ruby/object:Gem::Requirement
105
+ none: false
106
+ requirements:
107
+ - - ! '>='
108
+ - !ruby/object:Gem::Version
109
+ version: '0'
110
+ - !ruby/object:Gem::Dependency
111
+ name: bundler
112
+ requirement: !ruby/object:Gem::Requirement
113
+ none: false
114
+ requirements:
115
+ - - ~>
116
+ - !ruby/object:Gem::Version
117
+ version: '1.3'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ none: false
122
+ requirements:
123
+ - - ~>
124
+ - !ruby/object:Gem::Version
125
+ version: '1.3'
126
+ - !ruby/object:Gem::Dependency
127
+ name: rake
128
+ requirement: !ruby/object:Gem::Requirement
129
+ none: false
130
+ requirements:
131
+ - - ! '>='
132
+ - !ruby/object:Gem::Version
133
+ version: '0'
134
+ type: :development
135
+ prerelease: false
136
+ version_requirements: !ruby/object:Gem::Requirement
137
+ none: false
138
+ requirements:
139
+ - - ! '>='
140
+ - !ruby/object:Gem::Version
141
+ version: '0'
142
+ - !ruby/object:Gem::Dependency
143
+ name: rspec
144
+ requirement: !ruby/object:Gem::Requirement
145
+ none: false
146
+ requirements:
147
+ - - ! '>='
148
+ - !ruby/object:Gem::Version
149
+ version: '0'
150
+ type: :development
151
+ prerelease: false
152
+ version_requirements: !ruby/object:Gem::Requirement
153
+ none: false
154
+ requirements:
155
+ - - ! '>='
156
+ - !ruby/object:Gem::Version
157
+ version: '0'
158
+ description: A small wrapper around Elasticsearch
159
+ email:
160
+ executables: []
161
+ extensions: []
162
+ extra_rdoc_files: []
163
+ files:
164
+ - .gitignore
165
+ - Gemfile
166
+ - LICENSE.txt
167
+ - README.md
168
+ - Rakefile
169
+ - gummi.gemspec
170
+ - lib/gummi.rb
171
+ - lib/gummi/api.rb
172
+ - lib/gummi/attributes.rb
173
+ - lib/gummi/default_index.rb
174
+ - lib/gummi/document.rb
175
+ - lib/gummi/entity.rb
176
+ - lib/gummi/fields/boolean.rb
177
+ - lib/gummi/fields/integer.rb
178
+ - lib/gummi/fields/keyword.rb
179
+ - lib/gummi/fields/ngram_and_plain.rb
180
+ - lib/gummi/fields/path_hierarchy.rb
181
+ - lib/gummi/fields/positive_integer.rb
182
+ - lib/gummi/fields/sanitized_string.rb
183
+ - lib/gummi/fields/string.rb
184
+ - lib/gummi/fields/time.rb
185
+ - lib/gummi/index.rb
186
+ - lib/gummi/object.rb
187
+ - lib/gummi/repository.rb
188
+ - lib/gummi/search/filtered.rb
189
+ - lib/gummi/search/raw.rb
190
+ - lib/gummi/search/result.rb
191
+ - lib/gummi/search/searching.rb
192
+ - lib/gummi/version.rb
193
+ - lib/repobahn/entity.rb
194
+ - lib/repobahn/repository.rb
195
+ - spec/lib/gummi/document_spec.rb
196
+ - spec/lib/gummi/repository_spec.rb
197
+ - spec/models/db/person.rb
198
+ - spec/models/people.rb
199
+ - spec/models/person.rb
200
+ - spec/spec_helper.rb
201
+ homepage: ''
202
+ licenses:
203
+ - MIT
204
+ post_install_message:
205
+ rdoc_options: []
206
+ require_paths:
207
+ - lib
208
+ required_ruby_version: !ruby/object:Gem::Requirement
209
+ none: false
210
+ requirements:
211
+ - - ! '>='
212
+ - !ruby/object:Gem::Version
213
+ version: '0'
214
+ required_rubygems_version: !ruby/object:Gem::Requirement
215
+ none: false
216
+ requirements:
217
+ - - ! '>='
218
+ - !ruby/object:Gem::Version
219
+ version: '0'
220
+ requirements: []
221
+ rubyforge_project:
222
+ rubygems_version: 1.8.23
223
+ signing_key:
224
+ specification_version: 3
225
+ summary: A small wrapper around Elasticsearch
226
+ test_files:
227
+ - spec/lib/gummi/document_spec.rb
228
+ - spec/lib/gummi/repository_spec.rb
229
+ - spec/models/db/person.rb
230
+ - spec/models/people.rb
231
+ - spec/models/person.rb
232
+ - spec/spec_helper.rb