geocoder-kb 1.2.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.
- checksums.yaml +7 -0
- data/.gitignore +6 -0
- data/.travis.yml +31 -0
- data/CHANGELOG.md +384 -0
- data/LICENSE +20 -0
- data/README.md +1085 -0
- data/Rakefile +25 -0
- data/bin/geocode +5 -0
- data/examples/autoexpire_cache_dalli.rb +62 -0
- data/examples/autoexpire_cache_redis.rb +28 -0
- data/examples/cache_bypass.rb +48 -0
- data/examples/sidekiq_worker.rb +16 -0
- data/gemfiles/Gemfile.mongoid-2.4.x +16 -0
- data/lib/generators/geocoder/config/config_generator.rb +14 -0
- data/lib/generators/geocoder/config/templates/initializer.rb +21 -0
- data/lib/generators/geocoder/maxmind/geolite_city_generator.rb +28 -0
- data/lib/generators/geocoder/maxmind/geolite_country_generator.rb +28 -0
- data/lib/generators/geocoder/maxmind/templates/migration/geolite_city.rb +30 -0
- data/lib/generators/geocoder/maxmind/templates/migration/geolite_country.rb +17 -0
- data/lib/geocoder.rb +47 -0
- data/lib/geocoder/cache.rb +90 -0
- data/lib/geocoder/calculations.rb +428 -0
- data/lib/geocoder/cli.rb +121 -0
- data/lib/geocoder/configuration.rb +124 -0
- data/lib/geocoder/configuration_hash.rb +11 -0
- data/lib/geocoder/exceptions.rb +21 -0
- data/lib/geocoder/ip_address.rb +21 -0
- data/lib/geocoder/lookup.rb +102 -0
- data/lib/geocoder/lookups/amap.rb +55 -0
- data/lib/geocoder/lookups/baidu.rb +55 -0
- data/lib/geocoder/lookups/baidu_ip.rb +54 -0
- data/lib/geocoder/lookups/base.rb +302 -0
- data/lib/geocoder/lookups/bing.rb +59 -0
- data/lib/geocoder/lookups/dstk.rb +20 -0
- data/lib/geocoder/lookups/esri.rb +48 -0
- data/lib/geocoder/lookups/freegeoip.rb +47 -0
- data/lib/geocoder/lookups/geocoder_ca.rb +54 -0
- data/lib/geocoder/lookups/geocoder_us.rb +39 -0
- data/lib/geocoder/lookups/geocodio.rb +42 -0
- data/lib/geocoder/lookups/geoip2.rb +40 -0
- data/lib/geocoder/lookups/google.rb +67 -0
- data/lib/geocoder/lookups/google_places_details.rb +50 -0
- data/lib/geocoder/lookups/google_premier.rb +47 -0
- data/lib/geocoder/lookups/here.rb +62 -0
- data/lib/geocoder/lookups/mapquest.rb +60 -0
- data/lib/geocoder/lookups/maxmind.rb +90 -0
- data/lib/geocoder/lookups/maxmind_local.rb +58 -0
- data/lib/geocoder/lookups/nominatim.rb +52 -0
- data/lib/geocoder/lookups/okf.rb +43 -0
- data/lib/geocoder/lookups/opencagedata.rb +58 -0
- data/lib/geocoder/lookups/ovi.rb +62 -0
- data/lib/geocoder/lookups/pointpin.rb +68 -0
- data/lib/geocoder/lookups/postcode_anywhere_uk.rb +51 -0
- data/lib/geocoder/lookups/smarty_streets.rb +45 -0
- data/lib/geocoder/lookups/telize.rb +40 -0
- data/lib/geocoder/lookups/test.rb +44 -0
- data/lib/geocoder/lookups/yahoo.rb +88 -0
- data/lib/geocoder/lookups/yandex.rb +54 -0
- data/lib/geocoder/models/active_record.rb +50 -0
- data/lib/geocoder/models/base.rb +39 -0
- data/lib/geocoder/models/mongo_base.rb +64 -0
- data/lib/geocoder/models/mongo_mapper.rb +26 -0
- data/lib/geocoder/models/mongoid.rb +32 -0
- data/lib/geocoder/query.rb +111 -0
- data/lib/geocoder/railtie.rb +26 -0
- data/lib/geocoder/request.rb +25 -0
- data/lib/geocoder/results/amap.rb +85 -0
- data/lib/geocoder/results/baidu.rb +79 -0
- data/lib/geocoder/results/baidu_ip.rb +62 -0
- data/lib/geocoder/results/base.rb +67 -0
- data/lib/geocoder/results/bing.rb +48 -0
- data/lib/geocoder/results/dstk.rb +6 -0
- data/lib/geocoder/results/esri.rb +51 -0
- data/lib/geocoder/results/freegeoip.rb +45 -0
- data/lib/geocoder/results/geocoder_ca.rb +60 -0
- data/lib/geocoder/results/geocoder_us.rb +39 -0
- data/lib/geocoder/results/geocodio.rb +66 -0
- data/lib/geocoder/results/geoip2.rb +64 -0
- data/lib/geocoder/results/google.rb +124 -0
- data/lib/geocoder/results/google_places_details.rb +35 -0
- data/lib/geocoder/results/google_premier.rb +6 -0
- data/lib/geocoder/results/here.rb +62 -0
- data/lib/geocoder/results/mapquest.rb +51 -0
- data/lib/geocoder/results/maxmind.rb +135 -0
- data/lib/geocoder/results/maxmind_local.rb +49 -0
- data/lib/geocoder/results/nominatim.rb +94 -0
- data/lib/geocoder/results/okf.rb +106 -0
- data/lib/geocoder/results/opencagedata.rb +82 -0
- data/lib/geocoder/results/ovi.rb +62 -0
- data/lib/geocoder/results/pointpin.rb +44 -0
- data/lib/geocoder/results/postcode_anywhere_uk.rb +42 -0
- data/lib/geocoder/results/smarty_streets.rb +106 -0
- data/lib/geocoder/results/telize.rb +45 -0
- data/lib/geocoder/results/test.rb +33 -0
- data/lib/geocoder/results/yahoo.rb +55 -0
- data/lib/geocoder/results/yandex.rb +84 -0
- data/lib/geocoder/sql.rb +107 -0
- data/lib/geocoder/stores/active_record.rb +289 -0
- data/lib/geocoder/stores/base.rb +127 -0
- data/lib/geocoder/stores/mongo_base.rb +89 -0
- data/lib/geocoder/stores/mongo_mapper.rb +13 -0
- data/lib/geocoder/stores/mongoid.rb +13 -0
- data/lib/geocoder/version.rb +3 -0
- data/lib/hash_recursive_merge.rb +74 -0
- data/lib/maxmind_database.rb +109 -0
- data/lib/oauth_util.rb +112 -0
- data/lib/tasks/geocoder.rake +29 -0
- data/lib/tasks/maxmind.rake +73 -0
- data/test/fixtures/baidu_invalid_key +1 -0
- data/test/fixtures/baidu_ip_202_198_16_3 +19 -0
- data/test/fixtures/baidu_ip_invalid_key +1 -0
- data/test/fixtures/baidu_ip_no_results +1 -0
- data/test/fixtures/baidu_no_results +1 -0
- data/test/fixtures/baidu_reverse +1 -0
- data/test/fixtures/baidu_shanghai_pearl_tower +12 -0
- data/test/fixtures/bing_invalid_key +1 -0
- data/test/fixtures/bing_madison_square_garden +40 -0
- data/test/fixtures/bing_no_results +16 -0
- data/test/fixtures/bing_reverse +42 -0
- data/test/fixtures/cloudmade_invalid_key +1 -0
- data/test/fixtures/cloudmade_madison_square_garden +1 -0
- data/test/fixtures/cloudmade_no_results +1 -0
- data/test/fixtures/esri_madison_square_garden +59 -0
- data/test/fixtures/esri_no_results +8 -0
- data/test/fixtures/esri_reverse +21 -0
- data/test/fixtures/freegeoip_74_200_247_59 +12 -0
- data/test/fixtures/freegeoip_no_results +1 -0
- data/test/fixtures/geocoder_ca_madison_square_garden +1 -0
- data/test/fixtures/geocoder_ca_no_results +1 -0
- data/test/fixtures/geocoder_ca_reverse +34 -0
- data/test/fixtures/geocoder_us_madison_square_garden +1 -0
- data/test/fixtures/geocoder_us_no_results +1 -0
- data/test/fixtures/geocodio_1101_pennsylvania_ave +1 -0
- data/test/fixtures/geocodio_bad_api_key +3 -0
- data/test/fixtures/geocodio_invalid +4 -0
- data/test/fixtures/geocodio_no_results +1 -0
- data/test/fixtures/geocodio_over_query_limit +4 -0
- data/test/fixtures/google_garbage +456 -0
- data/test/fixtures/google_madison_square_garden +57 -0
- data/test/fixtures/google_no_city_data +44 -0
- data/test/fixtures/google_no_locality +51 -0
- data/test/fixtures/google_no_results +4 -0
- data/test/fixtures/google_over_limit +4 -0
- data/test/fixtures/google_places_details_invalid_request +4 -0
- data/test/fixtures/google_places_details_madison_square_garden +120 -0
- data/test/fixtures/google_places_details_no_results +4 -0
- data/test/fixtures/google_places_details_no_reviews +60 -0
- data/test/fixtures/google_places_details_no_types +66 -0
- data/test/fixtures/here_madison_square_garden +72 -0
- data/test/fixtures/here_no_results +8 -0
- data/test/fixtures/mapquest_error +16 -0
- data/test/fixtures/mapquest_invalid_api_key +16 -0
- data/test/fixtures/mapquest_invalid_request +16 -0
- data/test/fixtures/mapquest_madison_square_garden +52 -0
- data/test/fixtures/mapquest_no_results +16 -0
- data/test/fixtures/maxmind_24_24_24_21 +1 -0
- data/test/fixtures/maxmind_24_24_24_22 +1 -0
- data/test/fixtures/maxmind_24_24_24_23 +1 -0
- data/test/fixtures/maxmind_24_24_24_24 +1 -0
- data/test/fixtures/maxmind_74_200_247_59 +1 -0
- data/test/fixtures/maxmind_invalid_key +1 -0
- data/test/fixtures/maxmind_no_results +1 -0
- data/test/fixtures/nominatim_madison_square_garden +150 -0
- data/test/fixtures/nominatim_no_results +1 -0
- data/test/fixtures/nominatim_over_limit +1 -0
- data/test/fixtures/okf_kirstinmaki +67 -0
- data/test/fixtures/okf_no_results +4 -0
- data/test/fixtures/opencagedata_invalid_api_key +25 -0
- data/test/fixtures/opencagedata_invalid_request +26 -0
- data/test/fixtures/opencagedata_madison_square_garden +73 -0
- data/test/fixtures/opencagedata_no_results +29 -0
- data/test/fixtures/opencagedata_over_limit +31 -0
- data/test/fixtures/ovi_madison_square_garden +72 -0
- data/test/fixtures/ovi_no_results +8 -0
- data/test/fixtures/pointpin_10_10_10_10 +1 -0
- data/test/fixtures/pointpin_555_555_555_555 +1 -0
- data/test/fixtures/pointpin_80_111_555_555 +1 -0
- data/test/fixtures/pointpin_no_results +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_WR26NJ +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_generic_error +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_hampshire +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_key_limit_exceeded +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_no_results +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_romsey +1 -0
- data/test/fixtures/postcode_anywhere_uk_geocode_v2_00_unknown_key +1 -0
- data/test/fixtures/smarty_streets_11211 +1 -0
- data/test/fixtures/smarty_streets_madison_square_garden +47 -0
- data/test/fixtures/smarty_streets_no_results +1 -0
- data/test/fixtures/telize_10_10_10_10 +1 -0
- data/test/fixtures/telize_555_555_555_555 +4 -0
- data/test/fixtures/telize_74_200_247_59 +1 -0
- data/test/fixtures/telize_no_results +1 -0
- data/test/fixtures/yahoo_error +1 -0
- data/test/fixtures/yahoo_invalid_key +2 -0
- data/test/fixtures/yahoo_madison_square_garden +52 -0
- data/test/fixtures/yahoo_no_results +10 -0
- data/test/fixtures/yahoo_over_limit +2 -0
- data/test/fixtures/yandex_canada_rue_dupuis_14 +446 -0
- data/test/fixtures/yandex_invalid_key +1 -0
- data/test/fixtures/yandex_kremlin +48 -0
- data/test/fixtures/yandex_new_york +1 -0
- data/test/fixtures/yandex_no_city_and_town +112 -0
- data/test/fixtures/yandex_no_results +16 -0
- data/test/integration/http_client_test.rb +31 -0
- data/test/mongoid_test_helper.rb +43 -0
- data/test/test_helper.rb +416 -0
- data/test/unit/active_record_test.rb +16 -0
- data/test/unit/cache_test.rb +37 -0
- data/test/unit/calculations_test.rb +220 -0
- data/test/unit/configuration_test.rb +55 -0
- data/test/unit/error_handling_test.rb +56 -0
- data/test/unit/geocoder_test.rb +78 -0
- data/test/unit/https_test.rb +17 -0
- data/test/unit/ip_address_test.rb +27 -0
- data/test/unit/lookup_test.rb +153 -0
- data/test/unit/lookups/bing_test.rb +68 -0
- data/test/unit/lookups/dstk_test.rb +26 -0
- data/test/unit/lookups/esri_test.rb +48 -0
- data/test/unit/lookups/freegeoip_test.rb +27 -0
- data/test/unit/lookups/geocoder_ca_test.rb +17 -0
- data/test/unit/lookups/geocodio_test.rb +55 -0
- data/test/unit/lookups/geoip2_test.rb +27 -0
- data/test/unit/lookups/google_places_details_test.rb +122 -0
- data/test/unit/lookups/google_premier_test.rb +22 -0
- data/test/unit/lookups/google_test.rb +84 -0
- data/test/unit/lookups/mapquest_test.rb +60 -0
- data/test/unit/lookups/maxmind_local_test.rb +28 -0
- data/test/unit/lookups/maxmind_test.rb +63 -0
- data/test/unit/lookups/nominatim_test.rb +31 -0
- data/test/unit/lookups/okf_test.rb +38 -0
- data/test/unit/lookups/opencagedata_test.rb +64 -0
- data/test/unit/lookups/pointpin_test.rb +30 -0
- data/test/unit/lookups/postcode_anywhere_uk_test.rb +70 -0
- data/test/unit/lookups/smarty_streets_test.rb +71 -0
- data/test/unit/lookups/telize_test.rb +36 -0
- data/test/unit/lookups/yahoo_test.rb +35 -0
- data/test/unit/method_aliases_test.rb +26 -0
- data/test/unit/model_test.rb +38 -0
- data/test/unit/mongoid_test.rb +47 -0
- data/test/unit/near_test.rb +87 -0
- data/test/unit/oauth_util_test.rb +31 -0
- data/test/unit/proxy_test.rb +37 -0
- data/test/unit/query_test.rb +52 -0
- data/test/unit/rake_task_test.rb +21 -0
- data/test/unit/request_test.rb +35 -0
- data/test/unit/result_test.rb +72 -0
- data/test/unit/test_mode_test.rb +70 -0
- metadata +294 -0
@@ -0,0 +1,90 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require 'geocoder/results/maxmind'
|
3
|
+
require 'csv'
|
4
|
+
|
5
|
+
module Geocoder::Lookup
|
6
|
+
class Maxmind < Base
|
7
|
+
|
8
|
+
def name
|
9
|
+
"MaxMind"
|
10
|
+
end
|
11
|
+
|
12
|
+
def query_url(query)
|
13
|
+
"#{protocol}://geoip.maxmind.com/#{service_code}?" + url_query_string(query)
|
14
|
+
end
|
15
|
+
|
16
|
+
private # ---------------------------------------------------------------
|
17
|
+
|
18
|
+
##
|
19
|
+
# Return the name of the configured service, or raise an exception.
|
20
|
+
#
|
21
|
+
def configured_service!
|
22
|
+
if s = configuration[:service] and services.keys.include?(s)
|
23
|
+
return s
|
24
|
+
else
|
25
|
+
raise(
|
26
|
+
Geocoder::ConfigurationError,
|
27
|
+
"When using MaxMind you MUST specify a service name: " +
|
28
|
+
"Geocoder.configure(:maxmind => {:service => ...}), " +
|
29
|
+
"where '...' is one of: #{services.keys.inspect}"
|
30
|
+
)
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
def service_code
|
35
|
+
services[configured_service!]
|
36
|
+
end
|
37
|
+
|
38
|
+
def service_response_fields_count
|
39
|
+
Geocoder::Result::Maxmind.field_names[configured_service!].size
|
40
|
+
end
|
41
|
+
|
42
|
+
def data_contains_error?(parsed_data)
|
43
|
+
# if all fields given then there is an error
|
44
|
+
parsed_data.size == service_response_fields_count and !parsed_data.last.nil?
|
45
|
+
end
|
46
|
+
|
47
|
+
##
|
48
|
+
# Service names mapped to code used in URL.
|
49
|
+
#
|
50
|
+
def services
|
51
|
+
{
|
52
|
+
:country => "a",
|
53
|
+
:city => "b",
|
54
|
+
:city_isp_org => "f",
|
55
|
+
:omni => "e"
|
56
|
+
}
|
57
|
+
end
|
58
|
+
|
59
|
+
def results(query)
|
60
|
+
# don't look up a loopback address, just return the stored result
|
61
|
+
return [reserved_result] if query.loopback_ip_address?
|
62
|
+
doc = fetch_data(query)
|
63
|
+
if doc and doc.is_a?(Array)
|
64
|
+
if !data_contains_error?(doc)
|
65
|
+
return [doc]
|
66
|
+
elsif doc.last == "INVALID_LICENSE_KEY"
|
67
|
+
raise_error(Geocoder::InvalidApiKey) || warn("Invalid MaxMind API key.")
|
68
|
+
end
|
69
|
+
end
|
70
|
+
return []
|
71
|
+
end
|
72
|
+
|
73
|
+
def parse_raw_data(raw_data)
|
74
|
+
# Maxmind just returns text/plain as csv format but according to documentation,
|
75
|
+
# we get ISO-8859-1 encoded string. We need to convert it.
|
76
|
+
CSV.parse_line raw_data.force_encoding("ISO-8859-1").encode("UTF-8")
|
77
|
+
end
|
78
|
+
|
79
|
+
def reserved_result
|
80
|
+
",,,,0,0,0,0,,,".split(",")
|
81
|
+
end
|
82
|
+
|
83
|
+
def query_url_params(query)
|
84
|
+
{
|
85
|
+
:l => configuration.api_key,
|
86
|
+
:i => query.sanitized_text
|
87
|
+
}.merge(super)
|
88
|
+
end
|
89
|
+
end
|
90
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
require 'ipaddr'
|
2
|
+
require 'geocoder/lookups/base'
|
3
|
+
require 'geocoder/results/maxmind_local'
|
4
|
+
|
5
|
+
module Geocoder::Lookup
|
6
|
+
class MaxmindLocal < Base
|
7
|
+
|
8
|
+
def initialize
|
9
|
+
if !configuration[:file].nil?
|
10
|
+
begin
|
11
|
+
gem = RUBY_PLATFORM == 'java' ? 'jgeoip' : 'geoip'
|
12
|
+
require gem
|
13
|
+
rescue LoadError
|
14
|
+
raise "Could not load geoip dependency. To use MaxMind Local lookup you must add the #{gem} gem to your Gemfile or have it installed in your system."
|
15
|
+
end
|
16
|
+
end
|
17
|
+
super
|
18
|
+
end
|
19
|
+
|
20
|
+
def name
|
21
|
+
"MaxMind Local"
|
22
|
+
end
|
23
|
+
|
24
|
+
def required_api_key_parts
|
25
|
+
[]
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
def results(query)
|
31
|
+
if configuration[:file]
|
32
|
+
geoip_class = RUBY_PLATFORM == "java" ? JGeoIP : GeoIP
|
33
|
+
result = geoip_class.new(configuration[:file]).city(query.to_s)
|
34
|
+
result.nil? ? [] : [result.to_hash]
|
35
|
+
elsif configuration[:package] == :city
|
36
|
+
addr = IPAddr.new(query.text).to_i
|
37
|
+
q = "SELECT l.country, l.region, l.city, l.latitude, l.longitude
|
38
|
+
FROM maxmind_geolite_city_location l WHERE l.loc_id = (SELECT b.loc_id FROM maxmind_geolite_city_blocks b
|
39
|
+
WHERE b.start_ip_num <= #{addr} AND #{addr} <= b.end_ip_num LIMIT 1)"
|
40
|
+
format_result(q, [:country_name, :region_name, :city_name, :latitude, :longitude])
|
41
|
+
elsif configuration[:package] == :country
|
42
|
+
addr = IPAddr.new(query.text).to_i
|
43
|
+
q = "SELECT country, country_code FROM maxmind_geolite_country
|
44
|
+
WHERE start_ip_num <= #{addr} AND #{addr} <= end_ip_num"
|
45
|
+
format_result(q, [:country_name, :country_code2])
|
46
|
+
end
|
47
|
+
end
|
48
|
+
|
49
|
+
def format_result(query, attr_names)
|
50
|
+
if r = ActiveRecord::Base.connection.execute(query).first
|
51
|
+
r = r.values if r.is_a?(Hash) # some db adapters return Hash, some Array
|
52
|
+
[Hash[*attr_names.zip(r).flatten]]
|
53
|
+
else
|
54
|
+
[]
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,52 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require "geocoder/results/nominatim"
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class Nominatim < Base
|
6
|
+
|
7
|
+
def name
|
8
|
+
"Nominatim"
|
9
|
+
end
|
10
|
+
|
11
|
+
def map_link_url(coordinates)
|
12
|
+
"http://www.openstreetmap.org/?lat=#{coordinates[0]}&lon=#{coordinates[1]}&zoom=15&layers=M"
|
13
|
+
end
|
14
|
+
|
15
|
+
def query_url(query)
|
16
|
+
method = query.reverse_geocode? ? "reverse" : "search"
|
17
|
+
host = configuration[:host] || "nominatim.openstreetmap.org"
|
18
|
+
"#{protocol}://#{host}/#{method}?" + url_query_string(query)
|
19
|
+
end
|
20
|
+
|
21
|
+
private # ---------------------------------------------------------------
|
22
|
+
|
23
|
+
def results(query)
|
24
|
+
return [] unless doc = fetch_data(query)
|
25
|
+
doc.is_a?(Array) ? doc : [doc]
|
26
|
+
end
|
27
|
+
|
28
|
+
def parse_raw_data(raw_data)
|
29
|
+
if raw_data.include?("Bandwidth limit exceeded")
|
30
|
+
raise_error(Geocoder::OverQueryLimitError) || warn("Over API query limit.")
|
31
|
+
else
|
32
|
+
super(raw_data)
|
33
|
+
end
|
34
|
+
end
|
35
|
+
|
36
|
+
def query_url_params(query)
|
37
|
+
params = {
|
38
|
+
:format => "json",
|
39
|
+
:addressdetails => "1",
|
40
|
+
:"accept-language" => (query.language || configuration.language)
|
41
|
+
}.merge(super)
|
42
|
+
if query.reverse_geocode?
|
43
|
+
lat,lon = query.coordinates
|
44
|
+
params[:lat] = lat
|
45
|
+
params[:lon] = lon
|
46
|
+
else
|
47
|
+
params[:q] = query.sanitized_text
|
48
|
+
end
|
49
|
+
params
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
@@ -0,0 +1,43 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require "geocoder/results/okf"
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class Okf < Base
|
6
|
+
|
7
|
+
def name
|
8
|
+
"Okf"
|
9
|
+
end
|
10
|
+
|
11
|
+
def query_url(query)
|
12
|
+
"#{protocol}://data.okf.fi/gis/1/geocode/json?" + url_query_string(query)
|
13
|
+
end
|
14
|
+
|
15
|
+
private # ---------------------------------------------------------------
|
16
|
+
|
17
|
+
def valid_response?(response)
|
18
|
+
status = parse_json(response.body)["status"]
|
19
|
+
super(response) and ['OK', 'ZERO_RESULTS'].include?(status)
|
20
|
+
end
|
21
|
+
|
22
|
+
def results(query)
|
23
|
+
return [] unless doc = fetch_data(query)
|
24
|
+
case doc['status']; when "OK" # OK status implies >0 results
|
25
|
+
return doc['results']
|
26
|
+
end
|
27
|
+
return []
|
28
|
+
end
|
29
|
+
|
30
|
+
def query_url_okf_params(query)
|
31
|
+
params = {
|
32
|
+
(query.reverse_geocode? ? :latlng : :address) => query.sanitized_text,
|
33
|
+
:sensor => "false",
|
34
|
+
:language => (query.language || configuration.language)
|
35
|
+
}
|
36
|
+
params
|
37
|
+
end
|
38
|
+
|
39
|
+
def query_url_params(query)
|
40
|
+
query_url_okf_params(query).merge(super)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require 'geocoder/results/opencagedata'
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class Opencagedata < Base
|
6
|
+
|
7
|
+
def name
|
8
|
+
"OpenCageData"
|
9
|
+
end
|
10
|
+
|
11
|
+
def query_url(query)
|
12
|
+
"#{protocol}://api.opencagedata.com/geocode/v1/json?key=#{configuration.api_key}&#{url_query_string(query)}"
|
13
|
+
end
|
14
|
+
|
15
|
+
def required_api_key_parts
|
16
|
+
["key"]
|
17
|
+
end
|
18
|
+
|
19
|
+
private
|
20
|
+
|
21
|
+
def results(query)
|
22
|
+
return [] unless doc = fetch_data(query)
|
23
|
+
# return doc["results"]
|
24
|
+
|
25
|
+
messages = doc['status']['message']
|
26
|
+
case doc['status']['code']
|
27
|
+
when 400 # Error with input
|
28
|
+
raise_error(Geocoder::InvalidRequest, messages) ||
|
29
|
+
warn("Opencagedata Geocoding API error: #{messages}")
|
30
|
+
when 403 # Key related error
|
31
|
+
raise_error(Geocoder::InvalidApiKey, messages) ||
|
32
|
+
warn("Opencagedata Geocoding API error: #{messages}")
|
33
|
+
when 402 # Quata Exceeded
|
34
|
+
raise_error(Geocoder::OverQueryLimitError, messages) ||
|
35
|
+
warn("Opencagedata Geocoding API error: #{messages}")
|
36
|
+
when 500 # Unknown error
|
37
|
+
raise_error(Geocoder::Error, messages) ||
|
38
|
+
warn("Opencagedata Geocoding API error: #{messages}")
|
39
|
+
end
|
40
|
+
|
41
|
+
return doc["results"]
|
42
|
+
end
|
43
|
+
|
44
|
+
def query_url_params(query)
|
45
|
+
params = {
|
46
|
+
:q => query.sanitized_text,
|
47
|
+
:language => (query.language || configuration.language)
|
48
|
+
}.merge(super)
|
49
|
+
|
50
|
+
unless (bounds = query.options[:bounds]).nil?
|
51
|
+
params[:bounds] = bounds.map{ |point| "%f,%f" % point }.join(',')
|
52
|
+
end
|
53
|
+
|
54
|
+
params
|
55
|
+
end
|
56
|
+
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,62 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require 'geocoder/results/ovi'
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class Ovi < Base
|
6
|
+
|
7
|
+
def name
|
8
|
+
"Ovi"
|
9
|
+
end
|
10
|
+
|
11
|
+
def required_api_key_parts
|
12
|
+
[]
|
13
|
+
end
|
14
|
+
|
15
|
+
def query_url(query)
|
16
|
+
"#{protocol}://lbs.ovi.com/search/6.2/#{if query.reverse_geocode? then 'reverse' end}geocode.json?" + url_query_string(query)
|
17
|
+
end
|
18
|
+
|
19
|
+
private # ---------------------------------------------------------------
|
20
|
+
|
21
|
+
def results(query)
|
22
|
+
return [] unless doc = fetch_data(query)
|
23
|
+
return [] unless doc['Response'] && doc['Response']['View']
|
24
|
+
if r=doc['Response']['View']
|
25
|
+
return [] if r.nil? || !r.is_a?(Array) || r.empty?
|
26
|
+
return r.first['Result']
|
27
|
+
end
|
28
|
+
[]
|
29
|
+
end
|
30
|
+
|
31
|
+
def query_url_params(query)
|
32
|
+
options = {
|
33
|
+
:gen=>1,
|
34
|
+
:app_id=>api_key,
|
35
|
+
:app_code=>api_code
|
36
|
+
}
|
37
|
+
|
38
|
+
if query.reverse_geocode?
|
39
|
+
super.merge(options).merge(
|
40
|
+
:prox=>query.sanitized_text,
|
41
|
+
:mode=>:retrieveAddresses
|
42
|
+
)
|
43
|
+
else
|
44
|
+
super.merge(options).merge(
|
45
|
+
:searchtext=>query.sanitized_text
|
46
|
+
)
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
def api_key
|
51
|
+
if a=configuration.api_key
|
52
|
+
return a.first if a.is_a?(Array)
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
def api_code
|
57
|
+
if a=configuration.api_key
|
58
|
+
return a.last if a.is_a?(Array)
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require 'geocoder/results/pointpin'
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class Pointpin < Base
|
6
|
+
|
7
|
+
def name
|
8
|
+
"Pointpin"
|
9
|
+
end
|
10
|
+
|
11
|
+
def required_api_key_parts
|
12
|
+
["key"]
|
13
|
+
end
|
14
|
+
|
15
|
+
def query_url(query)
|
16
|
+
"#{ protocol }://geo.pointp.in/#{ api_key }/json/#{ query.sanitized_text }"
|
17
|
+
end
|
18
|
+
|
19
|
+
private
|
20
|
+
|
21
|
+
def results(query)
|
22
|
+
# don't look up a loopback address, just return the stored result
|
23
|
+
return [] if query.loopback_ip_address?
|
24
|
+
doc = fetch_data(query)
|
25
|
+
if doc and doc.is_a?(Hash)
|
26
|
+
if !data_contains_error?(doc)
|
27
|
+
return [doc]
|
28
|
+
elsif doc['error']
|
29
|
+
case doc['error']
|
30
|
+
when "Invalid IP address"
|
31
|
+
raise_error(Geocoder::InvalidRequest) || warn("Invalid Pointpin request.")
|
32
|
+
when "Invalid API key"
|
33
|
+
raise_error(Geocoder::InvalidApiKey) || warn("Invalid Pointpin API key.")
|
34
|
+
when "Address not found"
|
35
|
+
warn("Address not found.")
|
36
|
+
end
|
37
|
+
else
|
38
|
+
raise_error(Geocoder::Error) || warn("Pointpin server error")
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
return []
|
43
|
+
end
|
44
|
+
|
45
|
+
def data_contains_error?(parsed_data)
|
46
|
+
parsed_data.keys.include?('error')
|
47
|
+
end
|
48
|
+
|
49
|
+
def reserved_result(ip)
|
50
|
+
{
|
51
|
+
"ip" => ip,
|
52
|
+
"city" => "",
|
53
|
+
"region_code" => "",
|
54
|
+
"region_name" => "",
|
55
|
+
"metrocode" => "",
|
56
|
+
"zipcode" => "",
|
57
|
+
"latitude" => "0",
|
58
|
+
"longitude" => "0",
|
59
|
+
"country_name" => "Reserved",
|
60
|
+
"country_code" => "RD"
|
61
|
+
}
|
62
|
+
end
|
63
|
+
|
64
|
+
def api_key
|
65
|
+
configuration.api_key
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
require 'geocoder/lookups/base'
|
2
|
+
require 'geocoder/results/postcode_anywhere_uk'
|
3
|
+
|
4
|
+
module Geocoder::Lookup
|
5
|
+
class PostcodeAnywhereUk < Base
|
6
|
+
# API documentation: http://www.postcodeanywhere.co.uk/Support/WebService/Geocoding/UK/Geocode/2/
|
7
|
+
BASE_URL_GEOCODE_V2_00 = 'services.postcodeanywhere.co.uk/Geocoding/UK/Geocode/v2.00/json.ws'
|
8
|
+
DAILY_LIMIT_EXEEDED_ERROR_CODES = ['8', '17'] # api docs say these two codes are the same error
|
9
|
+
INVALID_API_KEY_ERROR_CODE = '2'
|
10
|
+
|
11
|
+
def name
|
12
|
+
'PostcodeAnywhereUk'
|
13
|
+
end
|
14
|
+
|
15
|
+
def required_api_key_parts
|
16
|
+
%w(key)
|
17
|
+
end
|
18
|
+
|
19
|
+
def query_url(query)
|
20
|
+
format('%s://%s?%s', protocol, BASE_URL_GEOCODE_V2_00, url_query_string(query))
|
21
|
+
end
|
22
|
+
|
23
|
+
private
|
24
|
+
|
25
|
+
def results(query)
|
26
|
+
response = fetch_data(query)
|
27
|
+
return [] if response.nil? || !response.is_a?(Array) || response.empty?
|
28
|
+
|
29
|
+
raise_exception_for_response(response[0]) if response[0]['Error']
|
30
|
+
response
|
31
|
+
end
|
32
|
+
|
33
|
+
def raise_exception_for_response(response)
|
34
|
+
case response['Error']
|
35
|
+
when *DAILY_LIMIT_EXEEDED_ERROR_CODES
|
36
|
+
raise_error(Geocoder::OverQueryLimitError, response['Cause']) || warn(response['Cause'])
|
37
|
+
when INVALID_API_KEY_ERROR_CODE
|
38
|
+
raise_error(Geocoder::InvalidApiKey, response['Cause']) || warn(response['Cause'])
|
39
|
+
else # anything else just raise general error with the api cause
|
40
|
+
raise_error(Geocoder::Error, response['Cause']) || warn(response['Cause'])
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def query_url_params(query)
|
45
|
+
{
|
46
|
+
:location => query.sanitized_text,
|
47
|
+
:key => configuration.api_key
|
48
|
+
}.merge(super)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|