geokit-rails 1.1.4 → 2.0.0.rc1
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.
Potentially problematic release.
This version of geokit-rails might be problematic. Click here for more details.
- checksums.yaml +15 -0
- data/lib/geokit-rails.rb +10 -26
- data/lib/geokit-rails/acts_as_mappable.rb +269 -369
- data/lib/geokit-rails/adapters/mysql2.rb +22 -0
- data/lib/geokit-rails/adapters/sqlite.rb +53 -0
- data/lib/geokit-rails/core_extensions.rb +10 -0
- data/lib/geokit-rails/defaults.rb +0 -1
- data/lib/geokit-rails/geocoder_control.rb +11 -9
- data/lib/geokit-rails/ip_geocode_lookup.rb +2 -4
- data/lib/geokit-rails/railtie.rb +38 -0
- data/lib/geokit-rails/version.rb +3 -0
- data/test/acts_as_mappable_test.rb +236 -281
- data/test/boot.rb +20 -13
- data/test/database.yml +10 -6
- data/test/{ip_geocode_lookup_test.rb → ip_geocode_lookup_test.disabled.rb} +7 -2
- data/test/models/mock_family.rb +2 -0
- data/test/models/mock_organization.rb +2 -0
- data/test/models/mock_person.rb +3 -0
- data/test/sqlite-debug.log +1293 -0
- data/test/tasks.rake +16 -10
- data/test/test.sqlite3 +0 -0
- data/test/test_helper.rb +26 -5
- metadata +223 -126
- data/CHANGELOG.rdoc +0 -46
- data/MIT-LICENSE +0 -20
- data/Manifest.txt +0 -44
- data/README.markdown +0 -561
- data/Rakefile +0 -54
- data/about.yml +0 -9
- data/assets/api_keys_template +0 -61
- data/geokit-rails.gemspec +0 -39
- data/init.rb +0 -1
- data/install.rb +0 -14
checksums.yaml
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
---
|
2
|
+
!binary "U0hBMQ==":
|
3
|
+
metadata.gz: !binary |-
|
4
|
+
ZjMwNjUzNmI4NzQ5M2Q2NTZiZGEwZTFkZDFlNjQ0MWE5Mzk1OGJkMQ==
|
5
|
+
data.tar.gz: !binary |-
|
6
|
+
N2E4MTgzZDk4YmVkNzIyMDg0N2RiOTMyMzdhYzgxNGVkMDZmZDgyOQ==
|
7
|
+
!binary "U0hBNTEy":
|
8
|
+
metadata.gz: !binary |-
|
9
|
+
NjE3ZWE2ZGY4MzkxOWFmOTc3N2RiM2U3ZTVmODdjOTQwNDgyYTQ4YjRkM2Y4
|
10
|
+
NmJkOWViMjUyN2I2NDcwMzBmOWUwYWE4ZmIzMGVhMGViNDFkMTRkZWI5YjNh
|
11
|
+
YTI4NjAyYzc4MGEwOGY2NWI1MjJhMDBkOTZiMjJkOWE4NWYxZDg=
|
12
|
+
data.tar.gz: !binary |-
|
13
|
+
MDQ1MDY2N2JjNjhkMzRiZjdhMDljMzUyNGI4NDU5YzNjMDdkNjcyNzcyN2Q2
|
14
|
+
YTBmODg3NWU3YzhiNWFkMjA1YzU3N2VmOTQ1MzdmNTE4M2RjZjljY2MwOTMz
|
15
|
+
YTQzOWY0ZGUwNDI5NWVmMjBjNjJiZmIwNjQ2MzZjNzk5NmM5NmY=
|
data/lib/geokit-rails.rb
CHANGED
@@ -1,26 +1,10 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
require 'geokit-rails/acts_as_mappable'
|
12
|
-
require 'geokit-rails/ip_geocode_lookup'
|
13
|
-
|
14
|
-
# Automatically mix in distance finder support into ActiveRecord classes.
|
15
|
-
ActiveRecord::Base.send :include, GeoKit::ActsAsMappable
|
16
|
-
|
17
|
-
# Automatically mix in ip geocoding helpers into ActionController classes.
|
18
|
-
ActionController::Base.send :include, GeoKit::IpGeocodeLookup
|
19
|
-
else
|
20
|
-
message=%q(WARNING: geokit-rails requires the Geokit gem. You either don't have the gem installed,
|
21
|
-
or you haven't told Rails to require it. If you're using a recent version of Rails:
|
22
|
-
config.gem "geokit" # in config/environment.rb
|
23
|
-
and of course install the gem: sudo gem install geokit)
|
24
|
-
puts message
|
25
|
-
Rails.logger.error message
|
26
|
-
end
|
1
|
+
require 'geokit'
|
2
|
+
|
3
|
+
require 'geokit-rails/railtie'
|
4
|
+
require 'geokit-rails/core_extensions'
|
5
|
+
|
6
|
+
require 'geokit-rails/defaults'
|
7
|
+
require 'geokit-rails/adapters/abstract'
|
8
|
+
require 'geokit-rails/acts_as_mappable'
|
9
|
+
require 'geokit-rails/geocoder_control'
|
10
|
+
require 'geokit-rails/ip_geocode_lookup'
|
@@ -1,136 +1,95 @@
|
|
1
|
+
require 'active_record'
|
2
|
+
require 'active_support/concern'
|
3
|
+
|
1
4
|
module Geokit
|
2
|
-
# Contains the class method acts_as_mappable targeted to be mixed into ActiveRecord.
|
3
|
-
# When mixed in, augments find services such that they provide distance calculation
|
4
|
-
# query services. The find method accepts additional options:
|
5
|
-
#
|
6
|
-
# * :origin - can be
|
7
|
-
# 1. a two-element array of latititude/longitude -- :origin=>[37.792,-122.393]
|
8
|
-
# 2. a geocodeable string -- :origin=>'100 Spear st, San Francisco, CA'
|
9
|
-
# 3. an object which responds to lat and lng methods, or latitude and longitude methods,
|
10
|
-
# or whatever methods you have specified for lng_column_name and lat_column_name
|
11
|
-
#
|
12
|
-
# Other finder methods are provided for specific queries. These are:
|
13
|
-
#
|
14
|
-
# * find_within (alias: find_inside)
|
15
|
-
# * find_beyond (alias: find_outside)
|
16
|
-
# * find_closest (alias: find_nearest)
|
17
|
-
# * find_farthest
|
18
|
-
#
|
19
|
-
# Counter methods are available and work similarly to finders.
|
20
|
-
#
|
21
|
-
# If raw SQL is desired, the distance_sql method can be used to obtain SQL appropriate
|
22
|
-
# to use in a find_by_sql call.
|
23
5
|
module ActsAsMappable
|
6
|
+
|
24
7
|
class UnsupportedAdapter < StandardError ; end
|
25
|
-
|
26
|
-
#
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
# By default, it tries to geocode the "address" field. Or, for more customized behavior:
|
46
|
-
#
|
47
|
-
# acts_as_mappable :auto_geocode=>{:field=>:address,:error_message=>'bad address'}
|
48
|
-
#
|
49
|
-
# In both cases, it creates a before_validation_on_create callback to geocode the given column.
|
50
|
-
# For anything more customized, we recommend you forgo the auto_geocode option
|
51
|
-
# and create your own AR callback to handle geocoding.
|
52
|
-
def acts_as_mappable(options = {})
|
53
|
-
metaclass = (class << self; self; end)
|
54
|
-
|
55
|
-
# Mix in the module, but ensure to do so just once.
|
56
|
-
return if !defined?(Geokit::Mappable) || metaclass.included_modules.include?(Geokit::ActsAsMappable::SingletonMethods)
|
57
|
-
|
58
|
-
send :extend, Geokit::ActsAsMappable::SingletonMethods
|
59
|
-
send :include, Geokit::Mappable
|
60
|
-
|
61
|
-
cattr_accessor :through
|
62
|
-
self.through = options[:through]
|
63
|
-
|
64
|
-
if reflection = Geokit::ActsAsMappable.end_of_reflection_chain(self.through, self)
|
65
|
-
metaclass.instance_eval do
|
66
|
-
[ :distance_column_name, :default_units, :default_formula, :lat_column_name, :lng_column_name, :qualified_lat_column_name, :qualified_lng_column_name ].each do |method_name|
|
67
|
-
define_method method_name do
|
68
|
-
reflection.klass.send(method_name)
|
8
|
+
|
9
|
+
# Add the +acts_as_mappable+ method into ActiveRecord subclasses
|
10
|
+
module Glue # :nodoc:
|
11
|
+
extend ActiveSupport::Concern
|
12
|
+
|
13
|
+
module ClassMethods # :nodoc:
|
14
|
+
def acts_as_mappable(options = {})
|
15
|
+
metaclass = (class << self; self; end)
|
16
|
+
|
17
|
+
include Geokit::ActsAsMappable
|
18
|
+
|
19
|
+
cattr_accessor :through
|
20
|
+
self.through = options[:through]
|
21
|
+
|
22
|
+
if reflection = Geokit::ActsAsMappable.end_of_reflection_chain(self.through, self)
|
23
|
+
metaclass.instance_eval do
|
24
|
+
[ :distance_column_name, :default_units, :default_formula, :lat_column_name, :lng_column_name, :qualified_lat_column_name, :qualified_lng_column_name ].each do |method_name|
|
25
|
+
define_method method_name do
|
26
|
+
reflection.klass.send(method_name)
|
27
|
+
end
|
69
28
|
end
|
70
29
|
end
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
30
|
+
else
|
31
|
+
cattr_accessor :distance_column_name, :default_units, :default_formula, :lat_column_name, :lng_column_name, :qualified_lat_column_name, :qualified_lng_column_name
|
32
|
+
|
33
|
+
self.distance_column_name = options[:distance_column_name] || 'distance'
|
34
|
+
self.default_units = options[:default_units] || Geokit::default_units
|
35
|
+
self.default_formula = options[:default_formula] || Geokit::default_formula
|
36
|
+
self.lat_column_name = options[:lat_column_name] || 'lat'
|
37
|
+
self.lng_column_name = options[:lng_column_name] || 'lng'
|
38
|
+
self.qualified_lat_column_name = "#{table_name}.#{lat_column_name}"
|
39
|
+
self.qualified_lng_column_name = "#{table_name}.#{lng_column_name}"
|
40
|
+
|
41
|
+
if options.include?(:auto_geocode) && options[:auto_geocode]
|
42
|
+
# if the form auto_geocode=>true is used, let the defaults take over by suppling an empty hash
|
43
|
+
options[:auto_geocode] = {} if options[:auto_geocode] == true
|
44
|
+
cattr_accessor :auto_geocode_field, :auto_geocode_error_message
|
45
|
+
self.auto_geocode_field = options[:auto_geocode][:field] || 'address'
|
46
|
+
self.auto_geocode_error_message = options[:auto_geocode][:error_message] || 'could not locate address'
|
47
|
+
|
48
|
+
# set the actual callback here
|
49
|
+
before_validation :auto_geocode_address, :on => :create
|
50
|
+
end
|
92
51
|
end
|
93
52
|
end
|
94
53
|
end
|
95
|
-
end
|
54
|
+
end # Glue
|
96
55
|
|
97
|
-
|
98
|
-
|
99
|
-
address=self.send(auto_geocode_field).to_s
|
100
|
-
geo=Geokit::Geocoders::MultiGeocoder.geocode(address)
|
56
|
+
class Relation < ActiveRecord::Relation
|
57
|
+
attr_accessor :distance_formula
|
101
58
|
|
102
|
-
|
103
|
-
self
|
104
|
-
|
105
|
-
|
106
|
-
|
59
|
+
def where(opts, *rest)
|
60
|
+
return self if opts.blank?
|
61
|
+
relation = clone
|
62
|
+
where_values = build_where(opts, rest)
|
63
|
+
relation.where_values += substitute_distance_in_values(where_values)
|
64
|
+
relation
|
107
65
|
end
|
108
66
|
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
association, through = through.to_a.first
|
117
|
-
else
|
118
|
-
association, through = through, nil
|
119
|
-
end
|
67
|
+
def order(*args)
|
68
|
+
return self if args.blank?
|
69
|
+
relation = clone
|
70
|
+
order_values = args.flatten
|
71
|
+
relation.order_values += substitute_distance_in_values(order_values)
|
72
|
+
relation
|
73
|
+
end
|
120
74
|
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
75
|
+
private
|
76
|
+
def substitute_distance_in_values(values)
|
77
|
+
return values unless @distance_formula
|
78
|
+
# substitute distance with the actual distance equation
|
79
|
+
pattern = Regexp.new("\\b#{@klass.distance_column_name}\\b")
|
80
|
+
values.map {|value| value.is_a?(String) ? value.gsub(pattern, @distance_formula) : value }
|
126
81
|
end
|
82
|
+
end
|
127
83
|
|
128
|
-
|
84
|
+
extend ActiveSupport::Concern
|
85
|
+
|
86
|
+
included do
|
87
|
+
include Geokit::Mappable
|
129
88
|
end
|
130
89
|
|
131
|
-
#
|
132
|
-
module
|
133
|
-
|
90
|
+
# Class methods included in models when +acts_as_mappable+ is called
|
91
|
+
module ClassMethods
|
92
|
+
|
134
93
|
# A proxy to an instance of a finder adapter, inferred from the connection's adapter.
|
135
94
|
def adapter
|
136
95
|
@adapter ||= begin
|
@@ -142,87 +101,76 @@ module Geokit
|
|
142
101
|
raise UnsupportedAdapter, "`#{connection.adapter_name.downcase}` is not a supported adapter."
|
143
102
|
end
|
144
103
|
end
|
145
|
-
|
146
|
-
# Extends the existing find method in potentially two ways:
|
147
|
-
# - If a mappable instance exists in the options, adds a distance column.
|
148
|
-
# - If a mappable instance exists in the options and the distance column exists in the
|
149
|
-
# conditions, substitutes the distance sql for the distance column -- this saves
|
150
|
-
# having to write the gory SQL.
|
151
|
-
def find(*args)
|
152
|
-
prepare_for_find_or_count(:find, args)
|
153
|
-
super(*args)
|
154
|
-
end
|
155
104
|
|
156
|
-
|
157
|
-
# - If a mappable instance exists in the options and the distance column exists in the
|
158
|
-
# conditions, substitutes the distance sql for the distance column -- this saves
|
159
|
-
# having to write the gory SQL.
|
160
|
-
def count(*args)
|
161
|
-
prepare_for_find_or_count(:count, args)
|
162
|
-
super(*args)
|
163
|
-
end
|
164
|
-
|
165
|
-
# Finds within a distance radius.
|
166
|
-
def find_within(distance, options={})
|
105
|
+
def within(distance, options = {})
|
167
106
|
options[:within] = distance
|
168
|
-
|
107
|
+
geo_scope(options)
|
169
108
|
end
|
170
|
-
alias
|
109
|
+
alias inside within
|
171
110
|
|
172
|
-
|
173
|
-
def find_beyond(distance, options={})
|
111
|
+
def beyond(distance, options = {})
|
174
112
|
options[:beyond] = distance
|
175
|
-
|
113
|
+
geo_scope(options)
|
176
114
|
end
|
177
|
-
alias
|
115
|
+
alias outside beyond
|
178
116
|
|
179
|
-
|
180
|
-
def find_by_range(range, options={})
|
117
|
+
def in_range(range, options = {})
|
181
118
|
options[:range] = range
|
182
|
-
|
119
|
+
geo_scope(options)
|
183
120
|
end
|
184
121
|
|
185
|
-
|
186
|
-
|
187
|
-
|
122
|
+
def in_bounds(bounds, options = {})
|
123
|
+
options[:bounds] = bounds
|
124
|
+
geo_scope(options)
|
188
125
|
end
|
189
|
-
alias find_nearest find_closest
|
190
126
|
|
191
|
-
|
192
|
-
|
193
|
-
find(:farthest, options)
|
127
|
+
def by_distance(options = {})
|
128
|
+
geo_scope(options).order("#{distance_column_name} asc")
|
194
129
|
end
|
195
130
|
|
196
|
-
|
197
|
-
|
198
|
-
options[:bounds] = bounds
|
199
|
-
find(:all, options)
|
131
|
+
def closest(options = {})
|
132
|
+
by_distance(options).first(1)
|
200
133
|
end
|
134
|
+
alias nearest closest
|
201
135
|
|
202
|
-
|
203
|
-
|
204
|
-
options[:within] = distance
|
205
|
-
count(options)
|
136
|
+
def farthest(options = {})
|
137
|
+
by_distance(options).last(1)
|
206
138
|
end
|
207
|
-
alias count_inside count_within
|
208
139
|
|
209
|
-
|
210
|
-
|
211
|
-
options[:beyond] = distance
|
212
|
-
count(options)
|
213
|
-
end
|
214
|
-
alias count_outside count_beyond
|
140
|
+
def geo_scope(options = {})
|
141
|
+
arel = self.is_a?(ActiveRecord::Relation) ? self : self.scoped
|
215
142
|
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
end
|
143
|
+
origin = extract_origin_from_options(options)
|
144
|
+
units = extract_units_from_options(options)
|
145
|
+
formula = extract_formula_from_options(options)
|
146
|
+
bounds = extract_bounds_from_options(options)
|
221
147
|
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
148
|
+
if origin || bounds
|
149
|
+
bounds = formulate_bounds_from_distance(options, origin, units) unless bounds
|
150
|
+
|
151
|
+
if origin
|
152
|
+
arel.distance_formula = distance_sql(origin, units, formula)
|
153
|
+
|
154
|
+
if arel.select_values.blank?
|
155
|
+
star_select = Arel::SqlLiteral.new(arel.quoted_table_name + '.*')
|
156
|
+
arel = arel.select(star_select)
|
157
|
+
end
|
158
|
+
end
|
159
|
+
|
160
|
+
if bounds
|
161
|
+
bound_conditions = bound_conditions(bounds)
|
162
|
+
arel = arel.where(bound_conditions) if bound_conditions
|
163
|
+
end
|
164
|
+
|
165
|
+
distance_conditions = distance_conditions(options)
|
166
|
+
arel = arel.where(distance_conditions) if distance_conditions
|
167
|
+
|
168
|
+
if self.through
|
169
|
+
arel = arel.includes(self.through)
|
170
|
+
end
|
171
|
+
end
|
172
|
+
|
173
|
+
arel
|
226
174
|
end
|
227
175
|
|
228
176
|
# Returns the distance calculation to be used as a display column or a condition. This
|
@@ -239,218 +187,170 @@ module Geokit
|
|
239
187
|
|
240
188
|
private
|
241
189
|
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
formula = extract_formula_from_options(options)
|
251
|
-
bounds = extract_bounds_from_options(options)
|
252
|
-
|
253
|
-
# Only proceed if this is a geokit-related query
|
254
|
-
if origin || bounds
|
255
|
-
# if no explicit bounds were given, try formulating them from the point and distance given
|
256
|
-
bounds = formulate_bounds_from_distance(options, origin, units) unless bounds
|
257
|
-
# Apply select adjustments based upon action.
|
258
|
-
add_distance_to_select(options, origin, units, formula) if origin && action == :find
|
259
|
-
# Apply the conditions for a bounding rectangle if applicable
|
260
|
-
apply_bounds_conditions(options,bounds) if bounds
|
261
|
-
# Apply distance scoping and perform substitutions.
|
262
|
-
apply_distance_scope(options)
|
263
|
-
substitute_distance_in_conditions(options, origin, units, formula) if origin && options.has_key?(:conditions)
|
264
|
-
# Order by scoping for find action.
|
265
|
-
apply_find_scope(args, options) if action == :find
|
266
|
-
# Handle :through
|
267
|
-
apply_include_for_through(options)
|
268
|
-
# Unfortunatley, we need to do extra work if you use an :include. See the method for more info.
|
269
|
-
handle_order_with_include(options,origin,units,formula) if options.include?(:include) && options.include?(:order) && origin
|
270
|
-
end
|
190
|
+
# Override ActiveRecord::Base.relation to return an instance of Geokit::ActsAsMappable::Relation.
|
191
|
+
# TODO: Do we need to override JoinDependency#relation too?
|
192
|
+
def relation
|
193
|
+
# NOTE: This cannot be @relation as ActiveRecord already uses this to
|
194
|
+
# cache *its* Relation object
|
195
|
+
@_geokit_relation ||= Relation.new(self, arel_table)
|
196
|
+
finder_needs_type_condition? ? @_geokit_relation.where(type_condition) : @_geokit_relation
|
197
|
+
end
|
271
198
|
|
272
|
-
|
273
|
-
|
274
|
-
|
199
|
+
# If it's a :within query, add a bounding box to improve performance.
|
200
|
+
# This only gets called if a :bounds argument is not otherwise supplied.
|
201
|
+
def formulate_bounds_from_distance(options, origin, units)
|
202
|
+
distance = options[:within] if options.has_key?(:within)
|
203
|
+
distance = options[:range].last-(options[:range].exclude_end?? 1 : 0) if options.has_key?(:range)
|
204
|
+
if distance
|
205
|
+
res=Geokit::Bounds.from_point_and_radius(origin,distance,:units=>units)
|
206
|
+
else
|
207
|
+
nil
|
275
208
|
end
|
209
|
+
end
|
276
210
|
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
|
284
|
-
else
|
285
|
-
options[:include] = [ self.through ]
|
286
|
-
end
|
287
|
-
end
|
211
|
+
def distance_conditions(options)
|
212
|
+
res = if options.has_key?(:within)
|
213
|
+
"#{distance_column_name} <= #{options[:within]}"
|
214
|
+
elsif options.has_key?(:beyond)
|
215
|
+
"#{distance_column_name} > #{options[:beyond]}"
|
216
|
+
elsif options.has_key?(:range)
|
217
|
+
"#{distance_column_name} >= #{options[:range].first} AND #{distance_column_name} <#{'=' unless options[:range].exclude_end?} #{options[:range].last}"
|
288
218
|
end
|
219
|
+
Arel::SqlLiteral.new("(#{res})") if res.present?
|
220
|
+
end
|
289
221
|
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
options[:order].sub!(distance_column_name, distance_sql(origin, units, formula))
|
297
|
-
end
|
222
|
+
def bound_conditions(bounds)
|
223
|
+
sw,ne = bounds.sw, bounds.ne
|
224
|
+
lng_sql = bounds.crosses_meridian? ? "(#{qualified_lng_column_name}<#{ne.lng} OR #{qualified_lng_column_name}>#{sw.lng})" : "#{qualified_lng_column_name}>#{sw.lng} AND #{qualified_lng_column_name}<#{ne.lng}"
|
225
|
+
res = "#{qualified_lat_column_name}>#{sw.lat} AND #{qualified_lat_column_name}<#{ne.lat} AND #{lng_sql}"
|
226
|
+
Arel::SqlLiteral.new("(#{res})") if res.present?
|
227
|
+
end
|
298
228
|
|
299
|
-
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
304
|
-
|
305
|
-
|
306
|
-
|
307
|
-
|
308
|
-
when :farthest
|
309
|
-
args[0] = :first
|
310
|
-
options[:limit] = 1
|
311
|
-
options[:order] = "#{distance_column_name} DESC"
|
312
|
-
end
|
313
|
-
end
|
229
|
+
# Extracts the origin instance out of the options if it exists and returns
|
230
|
+
# it. If there is no origin, looks for latitude and longitude values to
|
231
|
+
# create an origin. The side-effect of the method is to remove these
|
232
|
+
# option keys from the hash.
|
233
|
+
def extract_origin_from_options(options)
|
234
|
+
origin = options.delete(:origin)
|
235
|
+
res = normalize_point_to_lat_lng(origin) if origin
|
236
|
+
res
|
237
|
+
end
|
314
238
|
|
315
|
-
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
nil
|
324
|
-
end
|
325
|
-
end
|
239
|
+
# Extract the units out of the options if it exists and returns it. If
|
240
|
+
# there is no :units key, it uses the default. The side effect of the
|
241
|
+
# method is to remove the :units key from the options hash.
|
242
|
+
def extract_units_from_options(options)
|
243
|
+
units = options[:units] || default_units
|
244
|
+
options.delete(:units)
|
245
|
+
units
|
246
|
+
end
|
326
247
|
|
327
|
-
|
328
|
-
|
329
|
-
|
330
|
-
|
331
|
-
|
332
|
-
|
333
|
-
|
334
|
-
|
335
|
-
"#{distance_column_name} >= #{options[:range].first} AND #{distance_column_name} <#{'=' unless options[:range].exclude_end?} #{options[:range].last}"
|
336
|
-
end
|
248
|
+
# Extract the formula out of the options if it exists and returns it. If
|
249
|
+
# there is no :formula key, it uses the default. The side effect of the
|
250
|
+
# method is to remove the :formula key from the options hash.
|
251
|
+
def extract_formula_from_options(options)
|
252
|
+
formula = options[:formula] || default_formula
|
253
|
+
options.delete(:formula)
|
254
|
+
formula
|
255
|
+
end
|
337
256
|
|
338
|
-
|
339
|
-
|
340
|
-
|
341
|
-
|
342
|
-
end
|
257
|
+
def extract_bounds_from_options(options)
|
258
|
+
bounds = options.delete(:bounds)
|
259
|
+
bounds = Geokit::Bounds.normalize(bounds) if bounds
|
260
|
+
end
|
343
261
|
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
348
|
-
|
349
|
-
|
350
|
-
end
|
262
|
+
# Geocode IP address.
|
263
|
+
def geocode_ip_address(origin)
|
264
|
+
geo_location = Geokit::Geocoders::MultiGeocoder.geocode(origin)
|
265
|
+
return geo_location if geo_location.success
|
266
|
+
raise Geokit::Geocoders::GeocodeError
|
267
|
+
end
|
351
268
|
|
352
|
-
|
353
|
-
|
354
|
-
|
355
|
-
|
356
|
-
|
357
|
-
|
358
|
-
|
359
|
-
|
360
|
-
|
361
|
-
|
362
|
-
# Extract the units out of the options if it exists and returns it. If
|
363
|
-
# there is no :units key, it uses the default. The side effect of the
|
364
|
-
# method is to remove the :units key from the options hash.
|
365
|
-
def extract_units_from_options(options)
|
366
|
-
units = options[:units] || default_units
|
367
|
-
options.delete(:units)
|
368
|
-
units
|
369
|
-
end
|
269
|
+
# Given a point in a variety of (an address to geocode,
|
270
|
+
# an array of [lat,lng], or an object with appropriate lat/lng methods, an IP addres)
|
271
|
+
# this method will normalize it into a Geokit::LatLng instance. The only thing this
|
272
|
+
# method adds on top of LatLng#normalize is handling of IP addresses
|
273
|
+
def normalize_point_to_lat_lng(point)
|
274
|
+
res = geocode_ip_address(point) if point.is_a?(String) && /^(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})?$/.match(point)
|
275
|
+
res = Geokit::LatLng.normalize(point) unless res
|
276
|
+
res
|
277
|
+
end
|
370
278
|
|
371
|
-
|
372
|
-
|
373
|
-
|
374
|
-
|
375
|
-
|
376
|
-
|
377
|
-
|
279
|
+
# Looks for the distance column and replaces it with the distance sql. If an origin was not
|
280
|
+
# passed in and the distance column exists, we leave it to be flagged as bad SQL by the database.
|
281
|
+
# Conditions are either a string or an array. In the case of an array, the first entry contains
|
282
|
+
# the condition.
|
283
|
+
def substitute_distance_in_where_values(arel, origin, units=default_units, formula=default_formula)
|
284
|
+
pattern = Regexp.new("\\b#{distance_column_name}\\b")
|
285
|
+
value = distance_sql(origin, units, formula)
|
286
|
+
arel.where_values.map! do |where_value|
|
287
|
+
if where_value.is_a?(String)
|
288
|
+
where_value.gsub(pattern, value)
|
289
|
+
else
|
290
|
+
where_value
|
291
|
+
end
|
378
292
|
end
|
293
|
+
arel
|
294
|
+
end
|
379
295
|
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
296
|
+
# Returns the distance SQL using the spherical world formula (Haversine). The SQL is tuned
|
297
|
+
# to the database in use.
|
298
|
+
def sphere_distance_sql(origin, units)
|
299
|
+
lat = deg2rad(origin.lat)
|
300
|
+
lng = deg2rad(origin.lng)
|
301
|
+
multiplier = units_sphere_multiplier(units)
|
384
302
|
|
385
|
-
|
386
|
-
|
387
|
-
geo_location = Geokit::Geocoders::MultiGeocoder.geocode(origin)
|
388
|
-
return geo_location if geo_location.success
|
389
|
-
raise Geokit::Geocoders::GeocodeError
|
390
|
-
end
|
303
|
+
adapter.sphere_distance_sql(lat, lng, multiplier) if adapter
|
304
|
+
end
|
391
305
|
|
392
|
-
|
393
|
-
|
394
|
-
|
395
|
-
|
396
|
-
|
397
|
-
res = geocode_ip_address(point) if point.is_a?(String) && /^(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})?$/.match(point)
|
398
|
-
res = Geokit::LatLng.normalize(point) unless res
|
399
|
-
res
|
400
|
-
end
|
306
|
+
# Returns the distance SQL using the flat-world formula (Phythagorean Theory). The SQL is tuned
|
307
|
+
# to the database in use.
|
308
|
+
def flat_distance_sql(origin, units)
|
309
|
+
lat_degree_units = units_per_latitude_degree(units)
|
310
|
+
lng_degree_units = units_per_longitude_degree(origin.lat, units)
|
401
311
|
|
402
|
-
|
403
|
-
|
404
|
-
if origin
|
405
|
-
distance_selector = distance_sql(origin, units, formula) + " AS #{distance_column_name}"
|
406
|
-
selector = options.has_key?(:select) && options[:select] ? options[:select] : "*"
|
407
|
-
options[:select] = "#{selector}, #{distance_selector}"
|
408
|
-
end
|
409
|
-
end
|
312
|
+
adapter.flat_distance_sql(origin, lat_degree_units, lng_degree_units)
|
313
|
+
end
|
410
314
|
|
411
|
-
|
412
|
-
# passed in and the distance column exists, we leave it to be flagged as bad SQL by the database.
|
413
|
-
# Conditions are either a string or an array. In the case of an array, the first entry contains
|
414
|
-
# the condition.
|
415
|
-
def substitute_distance_in_conditions(options, origin, units=default_units, formula=default_formula)
|
416
|
-
condition = options[:conditions].is_a?(String) ? options[:conditions] : options[:conditions].first
|
417
|
-
pattern = Regexp.new("\\b#{distance_column_name}\\b")
|
418
|
-
condition.gsub!(pattern, distance_sql(origin, units, formula))
|
419
|
-
end
|
315
|
+
end # ClassMethods
|
420
316
|
|
421
|
-
|
422
|
-
|
423
|
-
|
424
|
-
|
425
|
-
lng = deg2rad(origin.lng)
|
426
|
-
multiplier = units_sphere_multiplier(units)
|
317
|
+
# this is the callback for auto_geocoding
|
318
|
+
def auto_geocode_address
|
319
|
+
address=self.send(auto_geocode_field).to_s
|
320
|
+
geo=Geokit::Geocoders::MultiGeocoder.geocode(address)
|
427
321
|
|
428
|
-
|
322
|
+
if geo.success
|
323
|
+
self.send("#{lat_column_name}=", geo.lat)
|
324
|
+
self.send("#{lng_column_name}=", geo.lng)
|
325
|
+
else
|
326
|
+
errors.add(auto_geocode_field, auto_geocode_error_message)
|
327
|
+
end
|
328
|
+
|
329
|
+
geo.success
|
330
|
+
end
|
331
|
+
|
332
|
+
def self.end_of_reflection_chain(through, klass)
|
333
|
+
while through
|
334
|
+
reflection = nil
|
335
|
+
if through.is_a?(Hash)
|
336
|
+
association, through = through.to_a.first
|
337
|
+
else
|
338
|
+
association, through = through, nil
|
429
339
|
end
|
430
|
-
|
431
|
-
|
432
|
-
|
433
|
-
|
434
|
-
|
435
|
-
lng_degree_units = units_per_longitude_degree(origin.lat, units)
|
436
|
-
|
437
|
-
adapter.flat_distance_sql(origin, lat_degree_units, lng_degree_units)
|
340
|
+
|
341
|
+
if reflection = klass.reflect_on_association(association)
|
342
|
+
klass = reflection.klass
|
343
|
+
else
|
344
|
+
raise ArgumentError, "You gave #{association} in :through, but I could not find it on #{klass}."
|
438
345
|
end
|
346
|
+
end
|
347
|
+
|
348
|
+
reflection
|
439
349
|
end
|
440
|
-
|
441
|
-
end
|
442
|
-
|
443
|
-
|
444
|
-
|
445
|
-
|
446
|
-
|
447
|
-
# resulting distance.
|
448
|
-
def sort_by_distance_from(origin, opts={})
|
449
|
-
distance_attribute_name = opts.delete(:distance_attribute_name) || 'distance'
|
450
|
-
self.each do |e|
|
451
|
-
e.class.send(:attr_accessor, distance_attribute_name) if !e.respond_to?("#{distance_attribute_name}=")
|
452
|
-
e.send("#{distance_attribute_name}=", e.distance_to(origin,opts))
|
453
|
-
end
|
454
|
-
self.sort!{|a,b|a.send(distance_attribute_name) <=> b.send(distance_attribute_name)}
|
455
|
-
end
|
456
|
-
end
|
350
|
+
|
351
|
+
end # ActsAsMappable
|
352
|
+
end # Geokit
|
353
|
+
|
354
|
+
|
355
|
+
|
356
|
+
# ActiveRecord::Base.extend Geokit::ActsAsMappable
|