commit
d1f61dadd4
@ -1,2 +1,3 @@
|
|||||||
|
--DO NOT MODIFY THIS FILE, IT IS GENERATED AUTOMATICALLY FROM SOURCES
|
||||||
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
||||||
\echo Use "CREATE EXTENSION cdb_geocoder_client" to load this file. \quit
|
\echo Use "CREATE EXTENSION cdb_geocoder_client" to load this file. \quit
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
|
--DO NOT MODIFY THIS FILE, IT IS GENERATED AUTOMATICALLY FROM SOURCES
|
||||||
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
||||||
\echo Use "CREATE EXTENSION cdb_geocoder_client" to load this file. \quit
|
\echo Use "CREATE EXTENSION cdb_geocoder_client" to load this file. \quit
|
||||||
|
1
server/extension/.gitignore
vendored
1
server/extension/.gitignore
vendored
@ -3,3 +3,4 @@ regression.diffs
|
|||||||
regression.out
|
regression.out
|
||||||
cdb_geocoder_server--0.0.1.sql
|
cdb_geocoder_server--0.0.1.sql
|
||||||
cdb_geocoder_server--0.1.0.sql
|
cdb_geocoder_server--0.1.0.sql
|
||||||
|
cdb_geocoder_server--0.2.0.sql
|
||||||
|
@ -12,8 +12,11 @@ NEW_EXTENSION_ARTIFACT = $(EXTENSION)--$(EXTVERSION).sql
|
|||||||
# @see http://www.postgresql.org/docs/current/static/extend-pgxs.html
|
# @see http://www.postgresql.org/docs/current/static/extend-pgxs.html
|
||||||
DATA = $(NEW_EXTENSION_ARTIFACT) \
|
DATA = $(NEW_EXTENSION_ARTIFACT) \
|
||||||
cdb_geocoder_server--0.0.1.sql \
|
cdb_geocoder_server--0.0.1.sql \
|
||||||
|
cdb_geocoder_server--0.1.0.sql \
|
||||||
cdb_geocoder_server--0.1.0--0.0.1.sql \
|
cdb_geocoder_server--0.1.0--0.0.1.sql \
|
||||||
cdb_geocoder_server--0.0.1--0.1.0.sql
|
cdb_geocoder_server--0.0.1--0.1.0.sql \
|
||||||
|
cdb_geocoder_server--0.2.0--0.1.0.sql \
|
||||||
|
cdb_geocoder_server--0.1.0--0.2.0.sql
|
||||||
|
|
||||||
|
|
||||||
REGRESS = $(notdir $(basename $(wildcard test/$(EXTVERSION)/sql/*test.sql)))
|
REGRESS = $(notdir $(basename $(wildcard test/$(EXTVERSION)/sql/*test.sql)))
|
||||||
|
122
server/extension/cdb_geocoder_server--0.1.0--0.2.0.sql
Normal file
122
server/extension/cdb_geocoder_server--0.1.0--0.2.0.sql
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._get_geocoder_config(username text, orgname text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "user_geocoder_config_{0}".format(username)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
import json
|
||||||
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
|
plpy.execute("SELECT cdb_geocoder_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
|
heremaps_conf_json = plpy.execute("SELECT cartodb.CDB_Conf_GetConf('heremaps_conf') as heremaps_conf", 1)[0]['heremaps_conf']
|
||||||
|
if not heremaps_conf_json:
|
||||||
|
heremaps_app_id = None
|
||||||
|
heremaps_app_code = None
|
||||||
|
else:
|
||||||
|
heremaps_conf = json.loads(heremaps_conf_json)
|
||||||
|
heremaps_app_id = heremaps_conf['app_id']
|
||||||
|
heremaps_app_code = heremaps_conf['app_code']
|
||||||
|
geocoder_config = GeocoderConfig(redis_conn, username, orgname, heremaps_app_id, heremaps_app_code)
|
||||||
|
# --Think about the security concerns with this kind of global cache, it should be only available
|
||||||
|
# --for this user session but...
|
||||||
|
GD[cache_key] = geocoder_config
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
-- Get the connection to redis from cache or create a new one
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._connect_to_redis(user_id text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "redis_connection_{0}".format(user_id)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
from cartodb_services.tools import RedisConnection
|
||||||
|
metadata_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metadata_config') c;""")[0]
|
||||||
|
metrics_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metrics_config') c;""")[0]
|
||||||
|
redis_metadata_connection = RedisConnection(metadata_config_params['sentinel_host'],
|
||||||
|
metadata_config_params['sentinel_port'],
|
||||||
|
metadata_config_params['sentinel_master_id'],
|
||||||
|
timeout=metadata_config_params['timeout'],
|
||||||
|
redis_db=metadata_config_params['redis_db']).redis_connection()
|
||||||
|
redis_metrics_connection = RedisConnection(metrics_config_params['sentinel_host'],
|
||||||
|
metrics_config_params['sentinel_port'],
|
||||||
|
metrics_config_params['sentinel_master_id'],
|
||||||
|
timeout=metrics_config_params['timeout'],
|
||||||
|
redis_db=metrics_config_params['redis_db']).redis_connection()
|
||||||
|
GD[cache_key] = {
|
||||||
|
'redis_metadata_connection': redis_metadata_connection,
|
||||||
|
'redis_metrics_connection': redis_metrics_connection,
|
||||||
|
}
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_here_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.here import HereMapsGeocoder
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
# -- Check the quota
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
plpy.error('You have reach the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = HereMapsGeocoder(user_geocoder_config.heremaps_app_id, user_geocoder_config.heremaps_app_code)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city, state=state_province, country=country)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_geocoder_use()
|
||||||
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
point = plpy.execute(plan, [coordinates[0], coordinates[1]], 1)[0]
|
||||||
|
return point['st_setsrid']
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_geocoder_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys, traceback
|
||||||
|
type_, value_, traceback_ = sys.exc_info()
|
||||||
|
quota_service.increment_failed_geocoder_use()
|
||||||
|
error_msg = 'There was an error trying to geocode using here maps geocoder: {0}'.format(e)
|
||||||
|
plpy.notice(traceback.format_tb(traceback_))
|
||||||
|
plpy.error(error_msg)
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_geocoder_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_google_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.google import GoogleMapsGeocoder
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = GoogleMapsGeocoder(user_geocoder_config.google_client_id, user_geocoder_config.google_api_key)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city, state=state_province, country=country)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_geocoder_use()
|
||||||
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
point = plpy.execute(plan, [coordinates[0], coordinates[1]], 1)[0]
|
||||||
|
return point['st_setsrid']
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_geocoder_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys, traceback
|
||||||
|
type_, value_, traceback_ = sys.exc_info()
|
||||||
|
quota_service.increment_failed_geocoder_use()
|
||||||
|
error_msg = 'There was an error trying to geocode using google maps geocoder: {0}'.format(e)
|
||||||
|
plpy.notice(traceback.format_tb(traceback_))
|
||||||
|
plpy.error(error_msg)
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_geocoder_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
95
server/extension/cdb_geocoder_server--0.2.0--0.1.0.sql
Normal file
95
server/extension/cdb_geocoder_server--0.2.0--0.1.0.sql
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
-- Get the Redis configuration from the _conf table --
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._get_geocoder_config(username text, orgname text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "user_geocoder_config_{0}".format(username)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
import json
|
||||||
|
from cartodb_geocoder import config_helper
|
||||||
|
plpy.execute("SELECT cdb_geocoder_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
|
heremaps_conf_json = plpy.execute("SELECT cartodb.CDB_Conf_GetConf('heremaps_conf') as heremaps_conf", 1)[0]['heremaps_conf']
|
||||||
|
if not heremaps_conf_json:
|
||||||
|
heremaps_app_id = None
|
||||||
|
heremaps_app_code = None
|
||||||
|
else:
|
||||||
|
heremaps_conf = json.loads(heremaps_conf_json)
|
||||||
|
heremaps_app_id = heremaps_conf['app_id']
|
||||||
|
heremaps_app_code = heremaps_conf['app_code']
|
||||||
|
geocoder_config = config_helper.GeocoderConfig(redis_conn, username, orgname, heremaps_app_id, heremaps_app_code)
|
||||||
|
# --Think about the security concerns with this kind of global cache, it should be only available
|
||||||
|
# --for this user session but...
|
||||||
|
GD[cache_key] = geocoder_config
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
-- Get the connection to redis from cache or create a new one
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._connect_to_redis(user_id text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "redis_connection_{0}".format(user_id)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
from cartodb_geocoder import redis_helper
|
||||||
|
metadata_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metadata_config') c;""")[0]
|
||||||
|
metrics_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metrics_config') c;""")[0]
|
||||||
|
redis_metadata_connection = redis_helper.RedisHelper(metadata_config_params['sentinel_host'],
|
||||||
|
metadata_config_params['sentinel_port'],
|
||||||
|
metadata_config_params['sentinel_master_id'],
|
||||||
|
timeout=metadata_config_params['timeout'],
|
||||||
|
redis_db=metadata_config_params['redis_db']).redis_connection()
|
||||||
|
redis_metrics_connection = redis_helper.RedisHelper(metrics_config_params['sentinel_host'],
|
||||||
|
metrics_config_params['sentinel_port'],
|
||||||
|
metrics_config_params['sentinel_master_id'],
|
||||||
|
timeout=metrics_config_params['timeout'],
|
||||||
|
redis_db=metrics_config_params['redis_db']).redis_connection()
|
||||||
|
GD[cache_key] = {
|
||||||
|
'redis_metadata_connection': redis_metadata_connection,
|
||||||
|
'redis_metrics_connection': redis_metrics_connection,
|
||||||
|
}
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_here_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from heremaps import heremapsgeocoder
|
||||||
|
from cartodb_geocoder import quota_service
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
# -- Check the quota
|
||||||
|
quota_service = quota_service.QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
plpy.error('You have reach the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = heremapsgeocoder.Geocoder(user_geocoder_config.heremaps_app_id, user_geocoder_config.heremaps_app_code)
|
||||||
|
coordinates = geocoder.geocode_address(searchtext=searchtext, city=city, state=state_province, country=country)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_geocoder_use()
|
||||||
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
point = plpy.execute(plan, [coordinates[0], coordinates[1]], 1)[0]
|
||||||
|
return point['st_setsrid']
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_geocoder_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys, traceback
|
||||||
|
type_, value_, traceback_ = sys.exc_info()
|
||||||
|
quota_service.increment_failed_geocoder_use()
|
||||||
|
error_msg = 'There was an error trying to geocode using here maps geocoder: {0}'.format(e)
|
||||||
|
plpy.notice(traceback.format_tb(traceback_))
|
||||||
|
plpy.error(error_msg)
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_google_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.error('Google geocoder is not available yet')
|
||||||
|
return None
|
||||||
|
$$ LANGUAGE plpythonu;
|
@ -1,6 +1,6 @@
|
|||||||
# cdb geocoder server extension
|
# cdb geocoder server extension
|
||||||
comment = 'CartoDB server geocoder extension'
|
comment = 'CartoDB server geocoder extension'
|
||||||
default_version = '0.1.0'
|
default_version = '0.2.0'
|
||||||
requires = 'plpythonu, postgis, cdb_geocoder'
|
requires = 'plpythonu, postgis, cdb_geocoder'
|
||||||
superuser = true
|
superuser = true
|
||||||
schema = cdb_geocoder_server
|
schema = cdb_geocoder_server
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
|
--DO NOT MODIFY THIS FILE, IT IS GENERATED AUTOMATICALLY FROM SOURCES
|
||||||
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
||||||
\echo Use "CREATE EXTENSION cdb_geocoder_server" to load this file. \quit
|
\echo Use "CREATE EXTENSION cdb_geocoder_server" to load this file. \quit
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
|
--DO NOT MODIFY THIS FILE, IT IS GENERATED AUTOMATICALLY FROM SOURCES
|
||||||
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
||||||
\echo Use "CREATE EXTENSION cdb_geocoder_server" to load this file. \quit
|
\echo Use "CREATE EXTENSION cdb_geocoder_server" to load this file. \quit
|
||||||
|
3
server/extension/sql/0.2.0/00_header.sql
Normal file
3
server/extension/sql/0.2.0/00_header.sql
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
--DO NOT MODIFY THIS FILE, IT IS GENERATED AUTOMATICALLY FROM SOURCES
|
||||||
|
-- Complain if script is sourced in psql, rather than via CREATE EXTENSION
|
||||||
|
\echo Use "CREATE EXTENSION cdb_geocoder_server" to load this file. \quit
|
57
server/extension/sql/0.2.0/10_redis_helper.sql
Normal file
57
server/extension/sql/0.2.0/10_redis_helper.sql
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
CREATE TYPE cdb_geocoder_server._redis_conf_params AS (
|
||||||
|
sentinel_host text,
|
||||||
|
sentinel_port int,
|
||||||
|
sentinel_master_id text,
|
||||||
|
redis_db text,
|
||||||
|
timeout float
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Get the Redis configuration from the _conf table --
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._get_redis_conf_v2(config_key text)
|
||||||
|
RETURNS cdb_geocoder_server._redis_conf_params AS $$
|
||||||
|
conf_query = "SELECT cartodb.CDB_Conf_GetConf('{0}') as conf".format(config_key)
|
||||||
|
conf = plpy.execute(conf_query)[0]['conf']
|
||||||
|
if conf is None:
|
||||||
|
plpy.error("There is no redis configuration defined")
|
||||||
|
else:
|
||||||
|
import json
|
||||||
|
params = json.loads(conf)
|
||||||
|
return {
|
||||||
|
"sentinel_host": params['sentinel_host'],
|
||||||
|
"sentinel_port": params['sentinel_port'],
|
||||||
|
"sentinel_master_id": params['sentinel_master_id'],
|
||||||
|
"timeout": params['timeout'],
|
||||||
|
"redis_db": params['redis_db']
|
||||||
|
}
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
-- Get the connection to redis from cache or create a new one
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._connect_to_redis(user_id text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "redis_connection_{0}".format(user_id)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
from cartodb_services.tools import RedisConnection
|
||||||
|
metadata_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metadata_config') c;""")[0]
|
||||||
|
metrics_config_params = plpy.execute("""select c.sentinel_host, c.sentinel_port,
|
||||||
|
c.sentinel_master_id, c.timeout, c.redis_db
|
||||||
|
from cdb_geocoder_server._get_redis_conf_v2('redis_metrics_config') c;""")[0]
|
||||||
|
redis_metadata_connection = RedisConnection(metadata_config_params['sentinel_host'],
|
||||||
|
metadata_config_params['sentinel_port'],
|
||||||
|
metadata_config_params['sentinel_master_id'],
|
||||||
|
timeout=metadata_config_params['timeout'],
|
||||||
|
redis_db=metadata_config_params['redis_db']).redis_connection()
|
||||||
|
redis_metrics_connection = RedisConnection(metrics_config_params['sentinel_host'],
|
||||||
|
metrics_config_params['sentinel_port'],
|
||||||
|
metrics_config_params['sentinel_master_id'],
|
||||||
|
timeout=metrics_config_params['timeout'],
|
||||||
|
redis_db=metrics_config_params['redis_db']).redis_connection()
|
||||||
|
GD[cache_key] = {
|
||||||
|
'redis_metadata_connection': redis_metadata_connection,
|
||||||
|
'redis_metrics_connection': redis_metrics_connection,
|
||||||
|
}
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
25
server/extension/sql/0.2.0/15_config_helper.sql
Normal file
25
server/extension/sql/0.2.0/15_config_helper.sql
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
-- Get the Redis configuration from the _conf table --
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._get_geocoder_config(username text, orgname text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "user_geocoder_config_{0}".format(username)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
import json
|
||||||
|
from cartodb_services.quota import GeocoderConfig
|
||||||
|
plpy.execute("SELECT cdb_geocoder_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
|
heremaps_conf_json = plpy.execute("SELECT cartodb.CDB_Conf_GetConf('heremaps_conf') as heremaps_conf", 1)[0]['heremaps_conf']
|
||||||
|
if not heremaps_conf_json:
|
||||||
|
heremaps_app_id = None
|
||||||
|
heremaps_app_code = None
|
||||||
|
else:
|
||||||
|
heremaps_conf = json.loads(heremaps_conf_json)
|
||||||
|
heremaps_app_id = heremaps_conf['app_id']
|
||||||
|
heremaps_app_code = heremaps_conf['app_code']
|
||||||
|
geocoder_config = GeocoderConfig(redis_conn, username, orgname, heremaps_app_id, heremaps_app_code)
|
||||||
|
# --Think about the security concerns with this kind of global cache, it should be only available
|
||||||
|
# --for this user session but...
|
||||||
|
GD[cache_key] = geocoder_config
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
84
server/extension/sql/0.2.0/20_geocode_street.sql
Normal file
84
server/extension/sql/0.2.0/20_geocode_street.sql
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
-- Geocodes a street address given a searchtext and a state and/or country
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_street_point_v2(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.execute("SELECT cdb_geocoder_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_geocoder_server._get_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
if user_geocoder_config.heremaps_geocoder:
|
||||||
|
here_plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_here_geocode_street_point($1, $2, $3, $4, $5, $6) as point; ", ["text", "text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(here_plan, [username, orgname, searchtext, city, state_province, country], 1)[0]['point']
|
||||||
|
elif user_geocoder_config.google_geocoder:
|
||||||
|
google_plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_google_geocode_street_point($1, $2, $3, $4, $5, $6) as point; ", ["text", "text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(google_plan, [username, orgname, searchtext, city, state_province, country], 1)[0]['point']
|
||||||
|
else:
|
||||||
|
plpy.error('Requested geocoder is not available')
|
||||||
|
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_here_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.here import HereMapsGeocoder
|
||||||
|
from cartodb_services.quota import QuotaService
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
# -- Check the quota
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
plpy.error('You have reach the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = HereMapsGeocoder(user_geocoder_config.heremaps_app_id, user_geocoder_config.heremaps_app_code)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city, state=state_province, country=country)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_geocoder_use()
|
||||||
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
point = plpy.execute(plan, [coordinates[0], coordinates[1]], 1)[0]
|
||||||
|
return point['st_setsrid']
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_geocoder_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys, traceback
|
||||||
|
type_, value_, traceback_ = sys.exc_info()
|
||||||
|
quota_service.increment_failed_geocoder_use()
|
||||||
|
error_msg = 'There was an error trying to geocode using here maps geocoder: {0}'.format(e)
|
||||||
|
plpy.notice(traceback.format_tb(traceback_))
|
||||||
|
plpy.error(error_msg)
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_geocoder_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_google_geocode_street_point(username TEXT, orgname TEXT, searchtext TEXT, city TEXT DEFAULT NULL, state_province TEXT DEFAULT NULL, country TEXT DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.google import GoogleMapsGeocoder
|
||||||
|
from cartodb_services.quota import QuotaService
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = GoogleMapsGeocoder(user_geocoder_config.google_client_id, user_geocoder_config.google_api_key)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city, state=state_province, country=country)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_geocoder_use()
|
||||||
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
point = plpy.execute(plan, [coordinates[0], coordinates[1]], 1)[0]
|
||||||
|
return point['st_setsrid']
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_geocoder_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys, traceback
|
||||||
|
type_, value_, traceback_ = sys.exc_info()
|
||||||
|
quota_service.increment_failed_geocoder_use()
|
||||||
|
error_msg = 'There was an error trying to geocode using google maps geocoder: {0}'.format(e)
|
||||||
|
plpy.notice(traceback.format_tb(traceback_))
|
||||||
|
plpy.error(error_msg)
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_geocoder_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
37
server/extension/sql/0.2.0/30_admin0.sql
Normal file
37
server/extension/sql/0.2.0/30_admin0.sql
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
-- Interface of the server extension
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_admin0_polygon(username text, orgname text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_admin0_polygons')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_admin0_polygon($1) AS mypolygon", ["text"])
|
||||||
|
rv = plpy.execute(plan, [country_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from cdb_geocode_admin0_polygons')
|
||||||
|
return rv[0]["mypolygon"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- Implementation of the server extension
|
||||||
|
-- Note: these functions depend on the cdb_geocoder extension
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_admin0_polygon(country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT n.the_geom as geom INTO ret
|
||||||
|
FROM (SELECT q, lower(regexp_replace(q, '[^a-zA-Z\u00C0-\u00ff]+', '', 'g'))::text x
|
||||||
|
FROM (SELECT country_name q) g) d
|
||||||
|
LEFT OUTER JOIN admin0_synonyms s ON name_ = d.x
|
||||||
|
LEFT OUTER JOIN ne_admin0_v3 n ON s.adm0_a3 = n.adm0_a3 GROUP BY d.q, n.the_geom, s.adm0_a3;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
89
server/extension/sql/0.2.0/40_admin1.sql
Normal file
89
server/extension/sql/0.2.0/40_admin1.sql
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
-- Interfacess of the server extension
|
||||||
|
|
||||||
|
---- cdb_geocode_admin1_polygon(admin1_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_admin1_polygon(username text, orgname text, admin1_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_admin1_polygon(admin1_name text)')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_admin1_polygon($1) AS mypolygon", ["text"])
|
||||||
|
rv = plpy.execute(plan, [admin1_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from cdb_geocode_admin1_polygons')
|
||||||
|
return rv[0]["mypolygon"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_admin1_polygon(admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_admin1_polygon(username text, orgname text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_admin1_polygon(admin1_name text, country_name text)')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_admin1_polygon($1, $2) AS mypolygon", ["text", "text"])
|
||||||
|
rv = plpy.execute(plan, [admin1_name, country_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from cdb_geocode_admin1_polygon(admin1_name text, country_name text)')
|
||||||
|
return rv[0]["mypolygon"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- Implementation of the server extension
|
||||||
|
-- Note: these functions depend on the cdb_geocoder extension
|
||||||
|
|
||||||
|
---- cdb_geocode_admin1_polygon(admin1_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_admin1_polygon(admin1_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_province_polygons
|
||||||
|
WHERE d.c = ANY (synonyms)
|
||||||
|
ORDER BY frequency DESC LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
trim(replace(lower(admin1_name),'.',' ')) c, admin1_name q
|
||||||
|
) d
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
---- cdb_geocode_admin1_polygon(admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_admin1_polygon(admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
WITH p AS (SELECT r.c, r.q, (SELECT iso3 FROM country_decoder WHERE lower(country_name) = ANY (synonyms)) i FROM (SELECT trim(replace(lower(admin1_name),'.',' ')) c, country_name q) r)
|
||||||
|
SELECT
|
||||||
|
geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_province_polygons
|
||||||
|
WHERE p.c = ANY (synonyms)
|
||||||
|
AND iso3 = p.i
|
||||||
|
ORDER BY frequency DESC LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM p) n;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
121
server/extension/sql/0.2.0/50_namedplaces.sql
Normal file
121
server/extension/sql/0.2.0/50_namedplaces.sql
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
-- Interfacess of the server extension
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_namedplace_point(city_name text)')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_namedplace_point($1) AS mypoint", ["text"])
|
||||||
|
rv = plpy.execute(plan, [city_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from geocode_namedplace')
|
||||||
|
return rv[0]["mypoint"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_namedplace_point(city_name text, country_name text)')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_namedplace_point($1, $2) AS mypoint", ["text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, country_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from geocode_namedplace')
|
||||||
|
return rv[0]["mypoint"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_namedplace_point($1, $2, $3) AS mypoint", ["text", "text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, admin1_name, country_name], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from Returning from geocode_namedplace')
|
||||||
|
return rv[0]["mypoint"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- Implementation of the server extension
|
||||||
|
-- Note: these functions depend on the cdb_geocoder extension
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_namedplace_point(city_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
WITH best AS (SELECT s AS q, (SELECT the_geom FROM global_cities_points_limited gp WHERE gp.lowername = lower(p.s) ORDER BY population DESC LIMIT 1) AS geom FROM (SELECT city_name as s) p),
|
||||||
|
next AS (SELECT p.s AS q, (SELECT gp.the_geom FROM global_cities_points_limited gp, global_cities_alternates_limited ga WHERE lower(p.s) = ga.lowername AND ga.geoname_id = gp.geoname_id ORDER BY preferred DESC LIMIT 1) geom FROM (SELECT city_name as s) p WHERE p.s NOT IN (SELECT q FROM best WHERE geom IS NOT NULL))
|
||||||
|
SELECT q, geom, TRUE AS success FROM best WHERE geom IS NOT NULL
|
||||||
|
UNION ALL
|
||||||
|
SELECT q, geom, CASE WHEN geom IS NULL THEN FALSE ELSE TRUE END AS success FROM next
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
WITH p AS (SELECT r.s, r.c, (SELECT iso2 FROM country_decoder WHERE lower(r.c) = ANY (synonyms)) i FROM (SELECT city_name AS s, country_name::text AS c) r),
|
||||||
|
best AS (SELECT p.s AS q, p.c AS c, (SELECT gp.the_geom AS geom FROM global_cities_points_limited gp WHERE gp.lowername = lower(p.s) AND gp.iso2 = p.i ORDER BY population DESC LIMIT 1) AS geom FROM p),
|
||||||
|
next AS (SELECT p.s AS q, p.c AS c, (SELECT gp.the_geom FROM global_cities_points_limited gp, global_cities_alternates_limited ga WHERE lower(p.s) = ga.lowername AND gp.iso2 = p.i AND ga.geoname_id = gp.geoname_id ORDER BY preferred DESC LIMIT 1) geom FROM p WHERE p.s NOT IN (SELECT q FROM best WHERE c = p.c AND geom IS NOT NULL))
|
||||||
|
SELECT geom FROM best WHERE geom IS NOT NULL
|
||||||
|
UNION ALL
|
||||||
|
SELECT geom FROM next
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
WITH inputcountry AS (
|
||||||
|
SELECT iso2 as isoTwo FROM country_decoder WHERE lower(country_name) = ANY (synonyms) LIMIT 1
|
||||||
|
),
|
||||||
|
p AS (
|
||||||
|
SELECT r.s, r.a1, (SELECT admin1 FROM admin1_decoder, inputcountry WHERE lower(r.a1) = ANY (synonyms) AND admin1_decoder.iso2 = inputcountry.isoTwo LIMIT 1) i FROM (SELECT city_name AS s, admin1_name::text AS a1) r),
|
||||||
|
best AS (SELECT p.s AS q, p.a1 as a1, (SELECT gp.the_geom AS geom FROM global_cities_points_limited gp WHERE gp.lowername = lower(p.s) AND gp.admin1 = p.i ORDER BY population DESC LIMIT 1) AS geom FROM p),
|
||||||
|
next AS (SELECT p.s AS q, p.a1 AS a1, (SELECT gp.the_geom FROM global_cities_points_limited gp, global_cities_alternates_limited ga WHERE lower(p.s) = ga.lowername AND ga.admin1 = p.i AND ga.geoname_id = gp.geoname_id ORDER BY preferred DESC LIMIT 1) geom FROM p WHERE p.s NOT IN (SELECT q FROM best WHERE geom IS NOT NULL))
|
||||||
|
SELECT geom FROM best WHERE geom IS NOT NULL
|
||||||
|
UNION ALL
|
||||||
|
SELECT geom FROM next
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
162
server/extension/sql/0.2.0/60_postalcodes.sql
Normal file
162
server/extension/sql/0.2.0/60_postalcodes.sql
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
-- Interface of the server extension
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_postalcode_point(username text, orgname text, code text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering _cdb_geocode_postalcode_point')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_postalcode_point($1) AS point", ["text"])
|
||||||
|
rv = plpy.execute(plan, [code], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from _cdb_geocode_postalcode_point')
|
||||||
|
return rv[0]["point"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_postalcode_point(username text, orgname text, code text, country text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering _cdb_geocode_postalcode_point')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_postalcode_point($1, $2) AS point", ["TEXT", "TEXT"])
|
||||||
|
rv = plpy.execute(plan, [code, country], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from _cdb_geocode_postalcode_point')
|
||||||
|
return rv[0]["point"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_postalcode_polygon(username text, orgname text, code text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering _cdb_geocode_postalcode_polygon')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_postalcode_polygon($1) AS polygon", ["text"])
|
||||||
|
rv = plpy.execute(plan, [code], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from _cdb_geocode_postalcode_polygon')
|
||||||
|
return rv[0]["polygon"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_postalcode_polygon(username text, orgname text, code text, country text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering _cdb_geocode_postalcode_point')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_postalcode_polygon($1, $2) AS polygon", ["TEXT", "TEXT"])
|
||||||
|
rv = plpy.execute(plan, [code, country], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from _cdb_geocode_postalcode_point')
|
||||||
|
return rv[0]["polygon"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- Implementation of the server extension
|
||||||
|
-- Note: these functions depend on the cdb_geocoder extension
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_postalcode_point(code text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_postal_code_points
|
||||||
|
WHERE postal_code = upper(d.q)
|
||||||
|
LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM (SELECT code q) d
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_postalcode_point(code text, country text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_postal_code_points
|
||||||
|
WHERE postal_code = upper(d.q)
|
||||||
|
AND iso3 = (
|
||||||
|
SELECT iso3 FROM country_decoder WHERE
|
||||||
|
lower(country) = ANY (synonyms) LIMIT 1
|
||||||
|
)
|
||||||
|
LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM (SELECT code q) d
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_postalcode_polygon(code text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_postal_code_polygons
|
||||||
|
WHERE postal_code = upper(d.q)
|
||||||
|
LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM (SELECT code q) d
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_postalcode_polygon(code text, country text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
BEGIN
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM (
|
||||||
|
SELECT
|
||||||
|
q, (
|
||||||
|
SELECT the_geom
|
||||||
|
FROM global_postal_code_polygons
|
||||||
|
WHERE postal_code = upper(d.q)
|
||||||
|
AND iso3 = (
|
||||||
|
SELECT iso3 FROM country_decoder WHERE
|
||||||
|
lower(country) = ANY (synonyms) LIMIT 1
|
||||||
|
)
|
||||||
|
LIMIT 1
|
||||||
|
) geom
|
||||||
|
FROM (SELECT code q) d
|
||||||
|
) v;
|
||||||
|
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
49
server/extension/sql/0.2.0/70_ips.sql
Normal file
49
server/extension/sql/0.2.0/70_ips.sql
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
-- Interface of the server extension
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server.cdb_geocode_ipaddress_point(username text, orgname text, ip text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
plpy.debug('Entering _cdb_geocode_ipaddress_point')
|
||||||
|
plpy.debug('user = %s' % username)
|
||||||
|
|
||||||
|
#--TODO: rate limiting check
|
||||||
|
#--TODO: quota check
|
||||||
|
|
||||||
|
#-- Copied from the doc, see http://www.postgresql.org/docs/9.4/static/plpython-database.html
|
||||||
|
plan = plpy.prepare("SELECT cdb_geocoder_server._cdb_geocode_ipaddress_point($1) AS point", ["TEXT"])
|
||||||
|
rv = plpy.execute(plan, [ip], 1)
|
||||||
|
|
||||||
|
plpy.debug('Returning from _cdb_geocode_ipaddress_point')
|
||||||
|
return rv[0]["point"]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- Implementation of the server extension
|
||||||
|
-- Note: these functions depend on the cdb_geocoder extension
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_geocoder_server._cdb_geocode_ipaddress_point(ip text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
DECLARE
|
||||||
|
ret Geometry;
|
||||||
|
|
||||||
|
new_ip INET;
|
||||||
|
BEGIN
|
||||||
|
BEGIN
|
||||||
|
IF family(ip::inet) = 6 THEN
|
||||||
|
new_ip := ip::inet;
|
||||||
|
ELSE
|
||||||
|
new_ip := ('::ffff:' || ip)::inet;
|
||||||
|
END IF;
|
||||||
|
EXCEPTION WHEN OTHERS THEN
|
||||||
|
SELECT NULL as geom INTO ret;
|
||||||
|
RETURN ret;
|
||||||
|
END;
|
||||||
|
|
||||||
|
WITH
|
||||||
|
ips AS (SELECT ip s, new_ip net),
|
||||||
|
matches AS (SELECT s, (SELECT the_geom FROM ip_address_locations WHERE network_start_ip <= ips.net ORDER BY network_start_ip DESC LIMIT 1) geom FROM ips)
|
||||||
|
SELECT geom INTO ret
|
||||||
|
FROM matches;
|
||||||
|
RETURN ret;
|
||||||
|
END
|
||||||
|
$$ LANGUAGE plpgsql;
|
15
server/extension/sql/0.2.0/90_geocoder_server_user.sql
Normal file
15
server/extension/sql/0.2.0/90_geocoder_server_user.sql
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
DO $$
|
||||||
|
BEGIN
|
||||||
|
IF NOT EXISTS (
|
||||||
|
SELECT *
|
||||||
|
FROM pg_catalog.pg_user
|
||||||
|
WHERE usename = 'geocoder_api') THEN
|
||||||
|
|
||||||
|
CREATE USER geocoder_api;
|
||||||
|
END IF;
|
||||||
|
GRANT EXECUTE ON ALL FUNCTIONS IN SCHEMA cdb_geocoder_server TO geocoder_api;
|
||||||
|
GRANT EXECUTE ON ALL FUNCTIONS IN SCHEMA public TO geocoder_api;
|
||||||
|
GRANT USAGE ON SCHEMA cdb_geocoder_server TO geocoder_api;
|
||||||
|
GRANT USAGE ON SCHEMA public TO geocoder_api;
|
||||||
|
GRANT SELECT ON ALL TABLES IN SCHEMA public TO geocoder_api;
|
||||||
|
END$$;
|
30
server/extension/test/0.2.0/expected/00_install_test.out
Normal file
30
server/extension/test/0.2.0/expected/00_install_test.out
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
-- Install dependencies
|
||||||
|
CREATE EXTENSION postgis;
|
||||||
|
CREATE EXTENSION schema_triggers;
|
||||||
|
CREATE EXTENSION plpythonu;
|
||||||
|
CREATE EXTENSION cartodb;
|
||||||
|
CREATE EXTENSION cdb_geocoder;
|
||||||
|
-- Install the extension
|
||||||
|
CREATE EXTENSION cdb_geocoder_server;
|
||||||
|
-- Mock the redis server connection to point to this very test db
|
||||||
|
SELECT cartodb.cdb_conf_setconf('redis_conf', '{"sentinel_host": "localhost", "sentinel_port": 26739, "sentinel_master_id": "mymaster", "timeout": 0.1, "redis_db": 5}');
|
||||||
|
cdb_conf_setconf
|
||||||
|
------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Mock the varnish invalidation function
|
||||||
|
-- (used by cdb_geocoder tests)
|
||||||
|
CREATE OR REPLACE FUNCTION public.cdb_invalidate_varnish(table_name text) RETURNS void AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN;
|
||||||
|
END
|
||||||
|
$$
|
||||||
|
LANGUAGE plpgsql;
|
||||||
|
-- Set user quota
|
||||||
|
SELECT cartodb.CDB_SetUserQuotaInBytes(0);
|
||||||
|
cdb_setuserquotainbytes
|
||||||
|
-------------------------
|
||||||
|
0
|
||||||
|
(1 row)
|
||||||
|
|
12
server/extension/test/0.2.0/expected/20_street_test.out
Normal file
12
server/extension/test/0.2.0/expected/20_street_test.out
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
-- Check for namedplaces signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_street_point_v2'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
47
server/extension/test/0.2.0/expected/30_admin0_test.out
Normal file
47
server/extension/test/0.2.0/expected/30_admin0_test.out
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin0_polygon('test_user', 'test_orgname', 'Spain');
|
||||||
|
cdb_geocode_admin0_polygon
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Insert some dummy synonym
|
||||||
|
INSERT INTO admin0_synonyms (name, adm0_a3) VALUES ('Spain', 'ESP');
|
||||||
|
-- Insert some dummy geometry to return
|
||||||
|
INSERT INTO ne_admin0_v3 (adm0_a3, the_geom) VALUES('ESP', ST_GeomFromText(
|
||||||
|
'POLYGON((-71.1031880899493 42.3152774590236,
|
||||||
|
-71.1031627617667 42.3152960829043,
|
||||||
|
-71.102923838298 42.3149156848307,
|
||||||
|
-71.1031880899493 42.3152774590236))',4326)
|
||||||
|
);
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin0_polygon('test_user', 'test_orgname', 'Spain');
|
||||||
|
cdb_geocode_admin0_polygon
|
||||||
|
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
0103000020E61000000100000004000000D0EA37A29AC651C00FD603035B284540FEFCFB379AC651C0C0503E9F5B284540FFDDDD4D96C651C033AC3B284F284540D0EA37A29AC651C00FD603035B284540
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Check for admin0 signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin0_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin0_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
81
server/extension/test/0.2.0/expected/40_admin1_test.out
Normal file
81
server/extension/test/0.2.0/expected/40_admin1_test.out
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California');
|
||||||
|
cdb_geocode_admin1_polygon
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California', 'United States');
|
||||||
|
cdb_geocode_admin1_polygon
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Insert dummy data into country decoder table
|
||||||
|
INSERT INTO country_decoder (synonyms, iso3) VALUES (Array['united states'], 'USA');
|
||||||
|
-- Insert some dummy data and geometry to return
|
||||||
|
INSERT INTO global_province_polygons (synonyms, iso3, the_geom) VALUES (Array['california'], 'USA', ST_GeomFromText(
|
||||||
|
'POLYGON((-71.1031880899493 42.3152774590236,
|
||||||
|
-71.1031627617667 42.3152960829043,
|
||||||
|
-71.102923838298 42.3149156848307,
|
||||||
|
-71.1031880899493 42.3152774590236))',4326)
|
||||||
|
);
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California');
|
||||||
|
cdb_geocode_admin1_polygon
|
||||||
|
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
0103000020E61000000100000004000000D0EA37A29AC651C00FD603035B284540FEFCFB379AC651C0C0503E9F5B284540FFDDDD4D96C651C033AC3B284F284540D0EA37A29AC651C00FD603035B284540
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California', 'United States');
|
||||||
|
cdb_geocode_admin1_polygon
|
||||||
|
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
0103000020E61000000100000004000000D0EA37A29AC651C00FD603035B284540FEFCFB379AC651C0C0503E9F5B284540FFDDDD4D96C651C033AC3B284F284540D0EA37A29AC651C00FD603035B284540
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Check for admin1 signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
136
server/extension/test/0.2.0/expected/50_namedplaces_test.out
Normal file
136
server/extension/test/0.2.0/expected/50_namedplaces_test.out
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Valencia', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Insert dummy data into points table
|
||||||
|
INSERT INTO global_cities_points_limited (geoname_id, name, iso2, admin1, admin2, population, lowername, the_geom) VALUES (3128760, 'Elche', 'ES', 'Valencia', 'AL', 34534, 'elche', ST_GeomFromText(
|
||||||
|
'POINT(0.6983 39.26787)',4326)
|
||||||
|
);
|
||||||
|
-- Insert dummy data into alternates table
|
||||||
|
INSERT INTO global_cities_alternates_limited (geoname_id, name, preferred, lowername, admin1_geonameid, iso2, admin1, the_geom) VALUES (3128760, 'Elx', true, 'elx', '000000', 'ES', 'Valencia', ST_GeomFromText(
|
||||||
|
'POINT(0.6983 39.26787)',4326)
|
||||||
|
);
|
||||||
|
-- Insert dummy data into country decoder table
|
||||||
|
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain'], 'ES');
|
||||||
|
-- Insert dummy data into admin1 decoder table
|
||||||
|
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
||||||
|
-- This should return the point inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Valencia', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche', 'valencia', 'Spain');
|
||||||
|
cdb_geocode_namedplace_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E6100000637FD93D7958E63F2ECA6C9049A24340
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Check for namedplaces signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
163
server/extension/test/0.2.0/expected/60_postalcodes_test.out
Normal file
163
server/extension/test/0.2.0/expected/60_postalcodes_test.out
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
-- Make sure dbs are clean
|
||||||
|
DELETE FROM global_postal_code_points;
|
||||||
|
DELETE FROM global_postal_code_polygons;
|
||||||
|
DELETE FROM country_decoder;
|
||||||
|
DELETE FROM available_services;
|
||||||
|
DELETE FROM admin0_synonyms;
|
||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204');
|
||||||
|
cdb_geocode_postalcode_point
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Insert dummy data into ip_address_locations
|
||||||
|
INSERT INTO global_postal_code_points (the_geom, iso3, postal_code, postal_code_num) VALUES (
|
||||||
|
'0101000020E61000000000000000E040408036B47414764840',
|
||||||
|
'ESP',
|
||||||
|
'03204',
|
||||||
|
3204
|
||||||
|
);
|
||||||
|
INSERT INTO global_postal_code_polygons (the_geom, iso3, postal_code, postal_code_num) VALUES (
|
||||||
|
'0106000020E610000001000000010300000001000000040000000000000000E000C01F383D7839B740400000000000E000C0AA3C0EDE220F3B4000000000004812404FB7FCCD04893D400000000000E000C01F383D7839B74040',
|
||||||
|
'ESP',
|
||||||
|
'03204',
|
||||||
|
3204
|
||||||
|
);
|
||||||
|
INSERT INTO country_decoder (iso3, synonyms) VALUES (
|
||||||
|
'ESP',
|
||||||
|
Array['spain', 'Spain', 'ESP']
|
||||||
|
);
|
||||||
|
INSERT INTO available_services (adm0_a3, admin0, postal_code_points, postal_code_polygons) VALUES (
|
||||||
|
'ESP',
|
||||||
|
't',
|
||||||
|
't',
|
||||||
|
't'
|
||||||
|
);
|
||||||
|
INSERT INTO admin0_synonyms (adm0_a3, name, name_, rank) VALUES (
|
||||||
|
'ESP',
|
||||||
|
'Spain',
|
||||||
|
'spain',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204');
|
||||||
|
cdb_geocode_postalcode_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E61000000000000000E040408036B47414764840
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204', 'spain');
|
||||||
|
cdb_geocode_postalcode_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E61000000000000000E040408036B47414764840
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_polygon('test_user', 'test_org', '03204');
|
||||||
|
cdb_geocode_postalcode_polygon
|
||||||
|
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
0106000020E610000001000000010300000001000000040000000000000000E000C01F383D7839B740400000000000E000C0AA3C0EDE220F3B4000000000004812404FB7FCCD04893D400000000000E000C01F383D7839B74040
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_polygon('test_user', 'test_org', '03204', 'spain');
|
||||||
|
cdb_geocode_postalcode_polygon
|
||||||
|
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
0106000020E610000001000000010300000001000000040000000000000000E000C01F383D7839B740400000000000E000C0AA3C0EDE220F3B4000000000004812404FB7FCCD04893D400000000000E000C01F383D7839B74040
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Clean dbs
|
||||||
|
DELETE FROM global_postal_code_points;
|
||||||
|
DELETE FROM global_postal_code_polygons;
|
||||||
|
DELETE FROM country_decoder;
|
||||||
|
DELETE FROM available_services;
|
||||||
|
DELETE FROM admin0_synonyms;
|
||||||
|
-- Check for namedplaces signatures (point and polygon)
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
40
server/extension/test/0.2.0/expected/70_ips_test.out
Normal file
40
server/extension/test/0.2.0/expected/70_ips_test.out
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_ipaddress_point('test_user', 'test_orgname', '0.0.0.0');
|
||||||
|
cdb_geocode_ipaddress_point
|
||||||
|
-----------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Insert dummy data into ip_address_locations
|
||||||
|
INSERT INTO ip_address_locations VALUES ('::ffff:0.0.0.0'::inet, (ST_SetSRID(ST_MakePoint('40.40', '3.71'), 4326)));
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_ipaddress_point('test_user', 'test_orgname', '0.0.0.0');
|
||||||
|
cdb_geocode_ipaddress_point
|
||||||
|
----------------------------------------------------
|
||||||
|
0101000020E61000003333333333334440AE47E17A14AE0D40
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
-- Check for namedplaces signatures (point and polygon)
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_ipaddress_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_ipaddress_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
exists
|
||||||
|
--------
|
||||||
|
t
|
||||||
|
(1 row)
|
||||||
|
|
@ -0,0 +1,5 @@
|
|||||||
|
REVOKE EXECUTE ON ALL FUNCTIONS IN SCHEMA cdb_geocoder_server FROM geocoder_api;
|
||||||
|
REVOKE EXECUTE ON ALL FUNCTIONS IN SCHEMA public FROM geocoder_api;
|
||||||
|
REVOKE USAGE ON SCHEMA cdb_geocoder_server FROM geocoder_api;
|
||||||
|
REVOKE USAGE ON SCHEMA public FROM geocoder_api;
|
||||||
|
REVOKE SELECT ON ALL TABLES IN SCHEMA public FROM geocoder_api;
|
24
server/extension/test/0.2.0/sql/00_install_test.sql
Normal file
24
server/extension/test/0.2.0/sql/00_install_test.sql
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
-- Install dependencies
|
||||||
|
CREATE EXTENSION postgis;
|
||||||
|
CREATE EXTENSION schema_triggers;
|
||||||
|
CREATE EXTENSION plpythonu;
|
||||||
|
CREATE EXTENSION cartodb;
|
||||||
|
CREATE EXTENSION cdb_geocoder;
|
||||||
|
|
||||||
|
-- Install the extension
|
||||||
|
CREATE EXTENSION cdb_geocoder_server;
|
||||||
|
|
||||||
|
-- Mock the redis server connection to point to this very test db
|
||||||
|
SELECT cartodb.cdb_conf_setconf('redis_conf', '{"sentinel_host": "localhost", "sentinel_port": 26739, "sentinel_master_id": "mymaster", "timeout": 0.1, "redis_db": 5}');
|
||||||
|
|
||||||
|
-- Mock the varnish invalidation function
|
||||||
|
-- (used by cdb_geocoder tests)
|
||||||
|
CREATE OR REPLACE FUNCTION public.cdb_invalidate_varnish(table_name text) RETURNS void AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN;
|
||||||
|
END
|
||||||
|
$$
|
||||||
|
LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
-- Set user quota
|
||||||
|
SELECT cartodb.CDB_SetUserQuotaInBytes(0);
|
7
server/extension/test/0.2.0/sql/20_street_test.sql
Normal file
7
server/extension/test/0.2.0/sql/20_street_test.sql
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
-- Check for namedplaces signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_street_point_v2'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
32
server/extension/test/0.2.0/sql/30_admin0_test.sql
Normal file
32
server/extension/test/0.2.0/sql/30_admin0_test.sql
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin0_polygon('test_user', 'test_orgname', 'Spain');
|
||||||
|
|
||||||
|
-- Insert some dummy synonym
|
||||||
|
INSERT INTO admin0_synonyms (name, adm0_a3) VALUES ('Spain', 'ESP');
|
||||||
|
|
||||||
|
-- Insert some dummy geometry to return
|
||||||
|
INSERT INTO ne_admin0_v3 (adm0_a3, the_geom) VALUES('ESP', ST_GeomFromText(
|
||||||
|
'POLYGON((-71.1031880899493 42.3152774590236,
|
||||||
|
-71.1031627617667 42.3152960829043,
|
||||||
|
-71.102923838298 42.3149156848307,
|
||||||
|
-71.1031880899493 42.3152774590236))',4326)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin0_polygon('test_user', 'test_orgname', 'Spain');
|
||||||
|
|
||||||
|
-- Check for admin0 signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin0_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin0_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
48
server/extension/test/0.2.0/sql/40_admin1_test.sql
Normal file
48
server/extension/test/0.2.0/sql/40_admin1_test.sql
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California', 'United States');
|
||||||
|
|
||||||
|
-- Insert dummy data into country decoder table
|
||||||
|
INSERT INTO country_decoder (synonyms, iso3) VALUES (Array['united states'], 'USA');
|
||||||
|
|
||||||
|
-- Insert some dummy data and geometry to return
|
||||||
|
INSERT INTO global_province_polygons (synonyms, iso3, the_geom) VALUES (Array['california'], 'USA', ST_GeomFromText(
|
||||||
|
'POLYGON((-71.1031880899493 42.3152774590236,
|
||||||
|
-71.1031627617667 42.3152960829043,
|
||||||
|
-71.102923838298 42.3149156848307,
|
||||||
|
-71.1031880899493 42.3152774590236))',4326)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_admin1_polygon('test_user', 'test_orgname', 'California', 'United States');
|
||||||
|
|
||||||
|
-- Check for admin1 signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_admin1_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
72
server/extension/test/0.2.0/sql/50_namedplaces_test.sql
Normal file
72
server/extension/test/0.2.0/sql/50_namedplaces_test.sql
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Spain');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Valencia', 'Spain');
|
||||||
|
|
||||||
|
-- Insert dummy data into points table
|
||||||
|
INSERT INTO global_cities_points_limited (geoname_id, name, iso2, admin1, admin2, population, lowername, the_geom) VALUES (3128760, 'Elche', 'ES', 'Valencia', 'AL', 34534, 'elche', ST_GeomFromText(
|
||||||
|
'POINT(0.6983 39.26787)',4326)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Insert dummy data into alternates table
|
||||||
|
INSERT INTO global_cities_alternates_limited (geoname_id, name, preferred, lowername, admin1_geonameid, iso2, admin1, the_geom) VALUES (3128760, 'Elx', true, 'elx', '000000', 'ES', 'Valencia', ST_GeomFromText(
|
||||||
|
'POINT(0.6983 39.26787)',4326)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Insert dummy data into country decoder table
|
||||||
|
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain'], 'ES');
|
||||||
|
|
||||||
|
-- Insert dummy data into admin1 decoder table
|
||||||
|
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
||||||
|
|
||||||
|
-- This should return the point inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Spain');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche', 'Spain');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elx', 'Valencia', 'Spain');
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_namedplace_point('test_user', 'test_orgname', 'Elche', 'valencia', 'Spain');
|
||||||
|
|
||||||
|
-- Check for namedplaces signatures
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_namedplace_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
117
server/extension/test/0.2.0/sql/60_postalcodes_test.sql
Normal file
117
server/extension/test/0.2.0/sql/60_postalcodes_test.sql
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
-- Make sure dbs are clean
|
||||||
|
DELETE FROM global_postal_code_points;
|
||||||
|
DELETE FROM global_postal_code_polygons;
|
||||||
|
DELETE FROM country_decoder;
|
||||||
|
DELETE FROM available_services;
|
||||||
|
DELETE FROM admin0_synonyms;
|
||||||
|
|
||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204');
|
||||||
|
|
||||||
|
-- Insert dummy data into ip_address_locations
|
||||||
|
INSERT INTO global_postal_code_points (the_geom, iso3, postal_code, postal_code_num) VALUES (
|
||||||
|
'0101000020E61000000000000000E040408036B47414764840',
|
||||||
|
'ESP',
|
||||||
|
'03204',
|
||||||
|
3204
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO global_postal_code_polygons (the_geom, iso3, postal_code, postal_code_num) VALUES (
|
||||||
|
'0106000020E610000001000000010300000001000000040000000000000000E000C01F383D7839B740400000000000E000C0AA3C0EDE220F3B4000000000004812404FB7FCCD04893D400000000000E000C01F383D7839B74040',
|
||||||
|
'ESP',
|
||||||
|
'03204',
|
||||||
|
3204
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO country_decoder (iso3, synonyms) VALUES (
|
||||||
|
'ESP',
|
||||||
|
Array['spain', 'Spain', 'ESP']
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO available_services (adm0_a3, admin0, postal_code_points, postal_code_polygons) VALUES (
|
||||||
|
'ESP',
|
||||||
|
't',
|
||||||
|
't',
|
||||||
|
't'
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO admin0_synonyms (adm0_a3, name, name_, rank) VALUES (
|
||||||
|
'ESP',
|
||||||
|
'Spain',
|
||||||
|
'spain',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204');
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_point('test_user', 'test_org', '03204', 'spain');
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_polygon('test_user', 'test_org', '03204');
|
||||||
|
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_postalcode_polygon('test_user', 'test_org', '03204', 'spain');
|
||||||
|
|
||||||
|
-- Clean dbs
|
||||||
|
DELETE FROM global_postal_code_points;
|
||||||
|
DELETE FROM global_postal_code_polygons;
|
||||||
|
DELETE FROM country_decoder;
|
||||||
|
DELETE FROM available_services;
|
||||||
|
DELETE FROM admin0_synonyms;
|
||||||
|
|
||||||
|
-- Check for namedplaces signatures (point and polygon)
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_postalcode_polygon'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text');
|
24
server/extension/test/0.2.0/sql/70_ips_test.sql
Normal file
24
server/extension/test/0.2.0/sql/70_ips_test.sql
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
-- Check that the public function is callable, even with no data
|
||||||
|
-- It should return NULL
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_ipaddress_point('test_user', 'test_orgname', '0.0.0.0');
|
||||||
|
|
||||||
|
-- Insert dummy data into ip_address_locations
|
||||||
|
INSERT INTO ip_address_locations VALUES ('::ffff:0.0.0.0'::inet, (ST_SetSRID(ST_MakePoint('40.40', '3.71'), 4326)));
|
||||||
|
|
||||||
|
-- This should return the polygon inserted above
|
||||||
|
SELECT cdb_geocoder_server.cdb_geocode_ipaddress_point('test_user', 'test_orgname', '0.0.0.0');
|
||||||
|
|
||||||
|
-- Check for namedplaces signatures (point and polygon)
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = 'cdb_geocode_ipaddress_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text, text, text');
|
||||||
|
|
||||||
|
SELECT exists(SELECT *
|
||||||
|
FROM pg_proc p
|
||||||
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
|
WHERE ns.nspname = 'cdb_geocoder_server'
|
||||||
|
AND proname = '_cdb_geocode_ipaddress_point'
|
||||||
|
AND oidvectortypes(p.proargtypes) = 'text');
|
@ -0,0 +1,5 @@
|
|||||||
|
REVOKE EXECUTE ON ALL FUNCTIONS IN SCHEMA cdb_geocoder_server FROM geocoder_api;
|
||||||
|
REVOKE EXECUTE ON ALL FUNCTIONS IN SCHEMA public FROM geocoder_api;
|
||||||
|
REVOKE USAGE ON SCHEMA cdb_geocoder_server FROM geocoder_api;
|
||||||
|
REVOKE USAGE ON SCHEMA public FROM geocoder_api;
|
||||||
|
REVOKE SELECT ON ALL TABLES IN SCHEMA public FROM geocoder_api;
|
@ -0,0 +1 @@
|
|||||||
|
from geocoder import GoogleMapsGeocoder
|
@ -16,11 +16,6 @@ class NoGeocodingParams(Exception):
|
|||||||
return repr('No params for geocoding specified')
|
return repr('No params for geocoding specified')
|
||||||
|
|
||||||
|
|
||||||
class EmptyGeocoderResponse(Exception):
|
|
||||||
def __str__(self):
|
|
||||||
return repr('The request could not be geocoded')
|
|
||||||
|
|
||||||
|
|
||||||
class MalformedResult(Exception):
|
class MalformedResult(Exception):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return repr('Result structure is malformed')
|
return repr('Result structure is malformed')
|
@ -0,0 +1,50 @@
|
|||||||
|
#!/usr/local/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
import googlemaps
|
||||||
|
|
||||||
|
from exceptions import MalformedResult
|
||||||
|
|
||||||
|
|
||||||
|
class GoogleMapsGeocoder:
|
||||||
|
"""A Google Maps Geocoder wrapper for python"""
|
||||||
|
|
||||||
|
def __init__(self, client_id, client_secret):
|
||||||
|
self.client_id = self._clean_client_id(client_id)
|
||||||
|
self.client_secret = client_secret
|
||||||
|
self.geocoder = googlemaps.Client(
|
||||||
|
client_id=self.client_id, client_secret=self.client_secret)
|
||||||
|
|
||||||
|
def geocode(self, searchtext, city=None, state=None,
|
||||||
|
country=None):
|
||||||
|
try:
|
||||||
|
opt_params = self._build_optional_parameters(city, state, country)
|
||||||
|
results = self.geocoder.geocode(address=searchtext,
|
||||||
|
components=opt_params)
|
||||||
|
if results:
|
||||||
|
return self._extract_lng_lat_from_result(results[0])
|
||||||
|
else:
|
||||||
|
return []
|
||||||
|
except KeyError:
|
||||||
|
raise MalformedResult()
|
||||||
|
|
||||||
|
def _extract_lng_lat_from_result(self, result):
|
||||||
|
location = result['geometry']['location']
|
||||||
|
longitude = location['lng']
|
||||||
|
latitude = location['lat']
|
||||||
|
return [longitude, latitude]
|
||||||
|
|
||||||
|
def _build_optional_parameters(self, city=None, state=None,
|
||||||
|
country=None):
|
||||||
|
optional_params = {}
|
||||||
|
if city:
|
||||||
|
optional_params['locality'] = city
|
||||||
|
if state:
|
||||||
|
optional_params['administrative_area'] = state
|
||||||
|
if country:
|
||||||
|
optional_params['country'] = country
|
||||||
|
return optional_params
|
||||||
|
|
||||||
|
def _clean_client_id(self, client_id):
|
||||||
|
# Consistency with how the client_id is saved in metadata
|
||||||
|
return client_id.replace('client=', '')
|
@ -0,0 +1 @@
|
|||||||
|
from geocoder import HereMapsGeocoder
|
@ -0,0 +1,21 @@
|
|||||||
|
#!/usr/local/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import json
|
||||||
|
|
||||||
|
|
||||||
|
class BadGeocodingParams(Exception):
|
||||||
|
def __init__(self, value):
|
||||||
|
self.value = value
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return repr('Bad geocoding params: ' + json.dumps(self.value))
|
||||||
|
|
||||||
|
|
||||||
|
class NoGeocodingParams(Exception):
|
||||||
|
def __str__(self):
|
||||||
|
return repr('No params for geocoding specified')
|
||||||
|
|
||||||
|
|
||||||
|
class MalformedResult(Exception):
|
||||||
|
def __str__(self):
|
||||||
|
return repr('Result structure is malformed')
|
@ -4,13 +4,10 @@
|
|||||||
import json
|
import json
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
from heremaps.heremapsexceptions import BadGeocodingParams
|
from exceptions import *
|
||||||
from heremaps.heremapsexceptions import EmptyGeocoderResponse
|
|
||||||
from heremaps.heremapsexceptions import NoGeocodingParams
|
|
||||||
from heremaps.heremapsexceptions import MalformedResult
|
|
||||||
|
|
||||||
|
|
||||||
class Geocoder:
|
class HereMapsGeocoder:
|
||||||
'A Here Maps Geocoder wrapper for python'
|
'A Here Maps Geocoder wrapper for python'
|
||||||
|
|
||||||
PRODUCTION_GEOCODE_JSON_URL = 'https://geocoder.api.here.com/6.2/geocode.json'
|
PRODUCTION_GEOCODE_JSON_URL = 'https://geocoder.api.here.com/6.2/geocode.json'
|
||||||
@ -50,10 +47,6 @@ class Geocoder:
|
|||||||
'strictlanguagemode'
|
'strictlanguagemode'
|
||||||
] + ADDRESS_PARAMS
|
] + ADDRESS_PARAMS
|
||||||
|
|
||||||
app_id = ''
|
|
||||||
app_code = ''
|
|
||||||
maxresults = ''
|
|
||||||
|
|
||||||
def __init__(self, app_id, app_code, maxresults=DEFAULT_MAXRESULTS,
|
def __init__(self, app_id, app_code, maxresults=DEFAULT_MAXRESULTS,
|
||||||
gen=DEFAULT_GEN, host=PRODUCTION_GEOCODE_JSON_URL):
|
gen=DEFAULT_GEN, host=PRODUCTION_GEOCODE_JSON_URL):
|
||||||
self.app_id = app_id
|
self.app_id = app_id
|
||||||
@ -62,18 +55,28 @@ class Geocoder:
|
|||||||
self.gen = gen
|
self.gen = gen
|
||||||
self.host = host
|
self.host = host
|
||||||
|
|
||||||
def geocode(self, params):
|
def geocode(self, **kwargs):
|
||||||
|
params = {}
|
||||||
|
for key, value in kwargs.iteritems():
|
||||||
|
if value:
|
||||||
|
params[key] = value
|
||||||
|
if not params:
|
||||||
|
raise NoGeocodingParams()
|
||||||
|
return self._execute_geocode(params)
|
||||||
|
|
||||||
|
def _execute_geocode(self, params):
|
||||||
if not set(params.keys()).issubset(set(self.ADDRESS_PARAMS)):
|
if not set(params.keys()).issubset(set(self.ADDRESS_PARAMS)):
|
||||||
raise BadGeocodingParams(params)
|
raise BadGeocodingParams(params)
|
||||||
response = self.perform_request(params)
|
|
||||||
try:
|
try:
|
||||||
results = response['Response']['View'][0]['Result']
|
response = self._perform_request(params)
|
||||||
|
results = response['Response']['View'][0]['Result'][0]
|
||||||
|
return self._extract_lng_lat_from_result(results)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
raise EmptyGeocoderResponse()
|
return []
|
||||||
|
except KeyError:
|
||||||
|
raise MalformedResult()
|
||||||
|
|
||||||
return results
|
def _perform_request(self, params):
|
||||||
|
|
||||||
def perform_request(self, params):
|
|
||||||
request_params = {
|
request_params = {
|
||||||
'app_id': self.app_id,
|
'app_id': self.app_id,
|
||||||
'app_code': self.app_code,
|
'app_code': self.app_code,
|
||||||
@ -87,21 +90,8 @@ class Geocoder:
|
|||||||
else:
|
else:
|
||||||
response.raise_for_status()
|
response.raise_for_status()
|
||||||
|
|
||||||
def geocode_address(self, **kwargs):
|
def _extract_lng_lat_from_result(self, result):
|
||||||
params = {}
|
location = result['Location']
|
||||||
for key, value in kwargs.iteritems():
|
|
||||||
if value:
|
|
||||||
params[key] = value
|
|
||||||
if not params:
|
|
||||||
raise NoGeocodingParams()
|
|
||||||
return self.geocode(params)
|
|
||||||
|
|
||||||
def extract_lng_lat_from_result(self, result):
|
|
||||||
try:
|
|
||||||
location = result['Location']
|
|
||||||
except KeyError:
|
|
||||||
raise MalformedResult()
|
|
||||||
|
|
||||||
longitude = location['DisplayPosition']['Longitude']
|
longitude = location['DisplayPosition']['Longitude']
|
||||||
latitude = location['DisplayPosition']['Latitude']
|
latitude = location['DisplayPosition']['Latitude']
|
||||||
|
|
@ -0,0 +1,3 @@
|
|||||||
|
from config import GeocoderConfig, ConfigException
|
||||||
|
from quota import QuotaService
|
||||||
|
from user import UserMetricsService
|
@ -1,4 +1,4 @@
|
|||||||
import user_service
|
from user import UserMetricsService
|
||||||
from datetime import date
|
from datetime import date
|
||||||
|
|
||||||
|
|
||||||
@ -8,7 +8,7 @@ class QuotaService:
|
|||||||
|
|
||||||
def __init__(self, user_geocoder_config, redis_connection):
|
def __init__(self, user_geocoder_config, redis_connection):
|
||||||
self._user_geocoder_config = user_geocoder_config
|
self._user_geocoder_config = user_geocoder_config
|
||||||
self._user_service = user_service.UserService(
|
self._user_service = UserMetricsService(
|
||||||
self._user_geocoder_config, redis_connection)
|
self._user_geocoder_config, redis_connection)
|
||||||
|
|
||||||
def check_user_quota(self):
|
def check_user_quota(self):
|
||||||
@ -32,19 +32,16 @@ class QuotaService:
|
|||||||
self._user_service.increment_service_use(
|
self._user_service.increment_service_use(
|
||||||
self._user_geocoder_config.service_type, "success_responses",
|
self._user_geocoder_config.service_type, "success_responses",
|
||||||
amount=amount)
|
amount=amount)
|
||||||
self.increment_total_geocoder_use(amount)
|
|
||||||
|
|
||||||
def increment_empty_geocoder_use(self, amount=1):
|
def increment_empty_geocoder_use(self, amount=1):
|
||||||
self._user_service.increment_service_use(
|
self._user_service.increment_service_use(
|
||||||
self._user_geocoder_config.service_type, "empty_responses",
|
self._user_geocoder_config.service_type, "empty_responses",
|
||||||
amount=amount)
|
amount=amount)
|
||||||
self.increment_total_geocoder_use(amount)
|
|
||||||
|
|
||||||
def increment_failed_geocoder_use(self, amount=1):
|
def increment_failed_geocoder_use(self, amount=1):
|
||||||
self._user_service.increment_service_use(
|
self._user_service.increment_service_use(
|
||||||
self._user_geocoder_config.service_type, "fail_responses",
|
self._user_geocoder_config.service_type, "fail_responses",
|
||||||
amount=amount)
|
amount=amount)
|
||||||
self.increment_total_geocoder_use(amount)
|
|
||||||
|
|
||||||
def increment_total_geocoder_use(self, amount=1):
|
def increment_total_geocoder_use(self, amount=1):
|
||||||
self._user_service.increment_service_use(
|
self._user_service.increment_service_use(
|
@ -2,7 +2,7 @@ from datetime import date, timedelta
|
|||||||
from dateutil.relativedelta import relativedelta
|
from dateutil.relativedelta import relativedelta
|
||||||
|
|
||||||
|
|
||||||
class UserService:
|
class UserMetricsService:
|
||||||
""" Class to manage all the user info """
|
""" Class to manage all the user info """
|
||||||
|
|
||||||
SERVICE_GEOCODER_NOKIA = 'geocoder_here'
|
SERVICE_GEOCODER_NOKIA = 'geocoder_here'
|
@ -0,0 +1 @@
|
|||||||
|
from redis_tools import RedisConnection
|
@ -1,7 +1,7 @@
|
|||||||
from redis.sentinel import Sentinel
|
from redis.sentinel import Sentinel
|
||||||
|
|
||||||
|
|
||||||
class RedisHelper:
|
class RedisConnection:
|
||||||
|
|
||||||
REDIS_DEFAULT_USER_DB = 5
|
REDIS_DEFAULT_USER_DB = 5
|
||||||
REDIS_DEFAULT_TIMEOUT = 2 #seconds
|
REDIS_DEFAULT_TIMEOUT = 2 #seconds
|
@ -1,8 +1,13 @@
|
|||||||
redis==2.10.5
|
redis==2.10.5
|
||||||
|
hiredis==0.1.5
|
||||||
# Dependency with incsv in the import
|
# Dependency with incsv in the import
|
||||||
python-dateutil==2.2
|
python-dateutil==2.2
|
||||||
|
googlemaps==2.4.2
|
||||||
|
# Dependency for googlemaps package
|
||||||
|
requests<=2.9.1
|
||||||
|
|
||||||
# Test
|
# Test
|
||||||
mock==1.3.0
|
mock==1.3.0
|
||||||
mockredispy==2.9.0.11
|
mockredispy==2.9.0.11
|
||||||
nose==1.3.7
|
nose==1.3.7
|
||||||
|
requests-mock==0.7.0
|
@ -1,5 +1,5 @@
|
|||||||
"""
|
"""
|
||||||
CartoDB Geocoder Python Library
|
CartoDB Services Python Library
|
||||||
|
|
||||||
See:
|
See:
|
||||||
https://github.com/CartoDB/geocoder-api
|
https://github.com/CartoDB/geocoder-api
|
||||||
@ -8,11 +8,11 @@ https://github.com/CartoDB/geocoder-api
|
|||||||
from setuptools import setup, find_packages
|
from setuptools import setup, find_packages
|
||||||
|
|
||||||
setup(
|
setup(
|
||||||
name='cartodb_geocoder',
|
name='cartodb_services',
|
||||||
|
|
||||||
version='0.0.1',
|
version='0.1.0',
|
||||||
|
|
||||||
description='CartoDB Geocoder Python Library',
|
description='CartoDB Services API Python Library',
|
||||||
|
|
||||||
url='https://github.com/CartoDB/geocoder-api',
|
url='https://github.com/CartoDB/geocoder-api',
|
||||||
|
|
||||||
@ -29,7 +29,7 @@ setup(
|
|||||||
'Programming Language :: Python :: 2.7',
|
'Programming Language :: Python :: 2.7',
|
||||||
],
|
],
|
||||||
|
|
||||||
keywords='maps api mapping tools geocoder',
|
keywords='maps api mapping tools geocoder routing',
|
||||||
|
|
||||||
packages=find_packages(exclude=['contrib', 'docs', 'tests']),
|
packages=find_packages(exclude=['contrib', 'docs', 'tests']),
|
||||||
|
|
@ -1,41 +1,41 @@
|
|||||||
import test_helper
|
import test_helper
|
||||||
from cartodb_geocoder import config_helper
|
from cartodb_services.metrics import GeocoderConfig, ConfigException
|
||||||
from unittest import TestCase
|
from unittest import TestCase
|
||||||
from nose.tools import assert_raises
|
from nose.tools import assert_raises
|
||||||
from mockredis import MockRedis
|
from mockredis import MockRedis
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
|
||||||
class TestConfigHelper(TestCase):
|
class TestConfig(TestCase):
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.redis_conn = MockRedis()
|
self.redis_conn = MockRedis()
|
||||||
|
|
||||||
def test_should_return_list_of_nokia_geocoder_config_if_its_ok(self):
|
def test_should_return_list_of_nokia_geocoder_config_if_its_ok(self):
|
||||||
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
||||||
geocoder_config = config_helper.GeocoderConfig(self.redis_conn,
|
geocoder_config = GeocoderConfig(self.redis_conn,
|
||||||
'test_user', None,
|
'test_user', None,
|
||||||
'nokia_id', 'nokia_cod')
|
'nokia_id', 'nokia_cod')
|
||||||
assert geocoder_config.heremaps_geocoder == True
|
assert geocoder_config.heremaps_geocoder is True
|
||||||
assert geocoder_config.geocoding_quota == 100
|
assert geocoder_config.geocoding_quota == 100
|
||||||
assert geocoder_config.soft_geocoding_limit == False
|
assert geocoder_config.soft_geocoding_limit is False
|
||||||
|
|
||||||
def test_should_return_list_of_nokia_geocoder_config_ok_for_org(self):
|
def test_should_return_list_of_nokia_geocoder_config_ok_for_org(self):
|
||||||
yesterday = datetime.today() - timedelta(days=1)
|
yesterday = datetime.today() - timedelta(days=1)
|
||||||
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
||||||
test_helper.build_redis_org_config(self.redis_conn, 'test_org',
|
test_helper.build_redis_org_config(self.redis_conn, 'test_org',
|
||||||
quota=200, end_date=yesterday)
|
quota=200, end_date=yesterday)
|
||||||
geocoder_config = config_helper.GeocoderConfig(self.redis_conn,
|
geocoder_config = GeocoderConfig(self.redis_conn,
|
||||||
'test_user', 'test_org',
|
'test_user', 'test_org',
|
||||||
'nokia_id', 'nokia_cod')
|
'nokia_id', 'nokia_cod')
|
||||||
assert geocoder_config.heremaps_geocoder == True
|
assert geocoder_config.heremaps_geocoder is True
|
||||||
assert geocoder_config.geocoding_quota == 200
|
assert geocoder_config.geocoding_quota == 200
|
||||||
assert geocoder_config.soft_geocoding_limit == False
|
assert geocoder_config.soft_geocoding_limit is False
|
||||||
assert geocoder_config.period_end_date.date() == yesterday.date()
|
assert geocoder_config.period_end_date.date() == yesterday.date()
|
||||||
|
|
||||||
def test_should_raise_configuration_exception_when_missing_nokia_geocoder_parameters(self):
|
def test_should_raise_configuration_exception_when_missing_nokia_geocoder_parameters(self):
|
||||||
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
test_helper.build_redis_user_config(self.redis_conn, 'test_user')
|
||||||
assert_raises(config_helper.ConfigException,
|
assert_raises(ConfigException,
|
||||||
config_helper.GeocoderConfig,
|
GeocoderConfig,
|
||||||
self.redis_conn, 'test_user',
|
self.redis_conn, 'test_user',
|
||||||
None, None, None)
|
None, None, None)
|
119
server/lib/python/cartodb_services/test/test_googlegeocoder.py
Normal file
119
server/lib/python/cartodb_services/test/test_googlegeocoder.py
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
#!/usr/local/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
import requests_mock
|
||||||
|
|
||||||
|
from cartodb_services.google import GoogleMapsGeocoder
|
||||||
|
from cartodb_services.google.exceptions import BadGeocodingParams
|
||||||
|
from cartodb_services.google.exceptions import NoGeocodingParams
|
||||||
|
from cartodb_services.google.exceptions import MalformedResult
|
||||||
|
|
||||||
|
requests_mock.Mocker.TEST_PREFIX = 'test_'
|
||||||
|
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
class GoogleGeocoderTestCase(unittest.TestCase):
|
||||||
|
GOOGLE_MAPS_GEOCODER_URL = 'https://maps.googleapis.com/maps/api/geocode/json'
|
||||||
|
|
||||||
|
EMPTY_RESPONSE = """{
|
||||||
|
"results" : [],
|
||||||
|
"status" : "ZERO_RESULTS"
|
||||||
|
}"""
|
||||||
|
|
||||||
|
GOOD_RESPONSE = """{
|
||||||
|
"results" : [
|
||||||
|
{
|
||||||
|
"address_components" : [
|
||||||
|
{
|
||||||
|
"long_name" : "1600",
|
||||||
|
"short_name" : "1600",
|
||||||
|
"types" : [ "street_number" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "Amphitheatre Pkwy",
|
||||||
|
"short_name" : "Amphitheatre Pkwy",
|
||||||
|
"types" : [ "route" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "Mountain View",
|
||||||
|
"short_name" : "Mountain View",
|
||||||
|
"types" : [ "locality", "political" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "Santa Clara County",
|
||||||
|
"short_name" : "Santa Clara County",
|
||||||
|
"types" : [ "administrative_area_level_2", "political" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "California",
|
||||||
|
"short_name" : "CA",
|
||||||
|
"types" : [ "administrative_area_level_1", "political" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "United States",
|
||||||
|
"short_name" : "US",
|
||||||
|
"types" : [ "country", "political" ]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"long_name" : "94043",
|
||||||
|
"short_name" : "94043",
|
||||||
|
"types" : [ "postal_code" ]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"formatted_address" : "1600 Amphitheatre Parkway, Mountain View, CA 94043, USA",
|
||||||
|
"geometry" : {
|
||||||
|
"location" : {
|
||||||
|
"lat" : 37.4224764,
|
||||||
|
"lng" : -122.0842499
|
||||||
|
},
|
||||||
|
"location_type" : "ROOFTOP",
|
||||||
|
"viewport" : {
|
||||||
|
"northeast" : {
|
||||||
|
"lat" : 37.4238253802915,
|
||||||
|
"lng" : -122.0829009197085
|
||||||
|
},
|
||||||
|
"southwest" : {
|
||||||
|
"lat" : 37.4211274197085,
|
||||||
|
"lng" : -122.0855988802915
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"place_id" : "ChIJ2eUgeAK6j4ARbn5u_wAGqWA",
|
||||||
|
"types" : [ "street_address" ]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"status" : "OK"
|
||||||
|
}"""
|
||||||
|
|
||||||
|
MALFORMED_RESPONSE = """{"manolo": "escobar"}"""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.geocoder = GoogleMapsGeocoder('dummy_client_id',
|
||||||
|
'MgxyOFxjZXIyOGO52jJlMzEzY1Oqy4hsO49E')
|
||||||
|
|
||||||
|
def test_geocode_address_with_valid_params(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', self.GOOGLE_MAPS_GEOCODER_URL,
|
||||||
|
text=self.GOOD_RESPONSE)
|
||||||
|
response = self.geocoder.geocode(
|
||||||
|
searchtext='Calle Eloy Gonzalo 27',
|
||||||
|
city='Madrid',
|
||||||
|
country='España')
|
||||||
|
|
||||||
|
self.assertEqual(response[0], -122.0842499)
|
||||||
|
self.assertEqual(response[1], 37.4224764)
|
||||||
|
|
||||||
|
def test_geocode_address_empty_response(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', self.GOOGLE_MAPS_GEOCODER_URL,
|
||||||
|
text=self.EMPTY_RESPONSE)
|
||||||
|
result = self.geocoder.geocode(searchtext='lkajfñlasjfñ')
|
||||||
|
self.assertEqual(result, [])
|
||||||
|
|
||||||
|
def test_geocode_with_malformed_result(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', self.GOOGLE_MAPS_GEOCODER_URL,
|
||||||
|
text=self.MALFORMED_RESPONSE)
|
||||||
|
with self.assertRaises(MalformedResult):
|
||||||
|
self.geocoder.geocode(
|
||||||
|
searchtext='Calle Eloy Gonzalo 27',
|
||||||
|
city='Madrid',
|
||||||
|
country='España')
|
@ -2,25 +2,28 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
import unittest
|
import unittest
|
||||||
|
import requests_mock
|
||||||
|
|
||||||
from heremaps import heremapsgeocoder
|
from cartodb_services.here import HereMapsGeocoder
|
||||||
from heremaps.heremapsexceptions import BadGeocodingParams
|
from cartodb_services.here.exceptions import BadGeocodingParams
|
||||||
from heremaps.heremapsexceptions import EmptyGeocoderResponse
|
from cartodb_services.here.exceptions import NoGeocodingParams
|
||||||
from heremaps.heremapsexceptions import NoGeocodingParams
|
from cartodb_services.here.exceptions import MalformedResult
|
||||||
from heremaps.heremapsexceptions import MalformedResult
|
|
||||||
|
requests_mock.Mocker.TEST_PREFIX = 'test_'
|
||||||
|
|
||||||
|
|
||||||
class GeocoderTestCase(unittest.TestCase):
|
@requests_mock.Mocker()
|
||||||
EMPTY_RESPONSE = {
|
class HereMapsGeocoderTestCase(unittest.TestCase):
|
||||||
|
EMPTY_RESPONSE = """{
|
||||||
"Response": {
|
"Response": {
|
||||||
"MetaInfo": {
|
"MetaInfo": {
|
||||||
"Timestamp": "2015-11-04T16:31:57.273+0000"
|
"Timestamp": "2015-11-04T16:31:57.273+0000"
|
||||||
},
|
},
|
||||||
"View": []
|
"View": []
|
||||||
}
|
}
|
||||||
}
|
}"""
|
||||||
|
|
||||||
GOOD_RESPONSE = {
|
GOOD_RESPONSE = """{
|
||||||
"Response": {
|
"Response": {
|
||||||
"MetaInfo": {
|
"MetaInfo": {
|
||||||
"Timestamp": "2015-11-04T16:30:32.187+0000"
|
"Timestamp": "2015-11-04T16:30:32.187+0000"
|
||||||
@ -28,7 +31,7 @@ class GeocoderTestCase(unittest.TestCase):
|
|||||||
"View": [{
|
"View": [{
|
||||||
"_type": "SearchResultsViewType",
|
"_type": "SearchResultsViewType",
|
||||||
"ViewId": 0,
|
"ViewId": 0,
|
||||||
"Result": [{
|
"Result": {
|
||||||
"Relevance": 0.89,
|
"Relevance": 0.89,
|
||||||
"MatchLevel": "street",
|
"MatchLevel": "street",
|
||||||
"MatchQuality": {
|
"MatchQuality": {
|
||||||
@ -57,7 +60,7 @@ class GeocoderTestCase(unittest.TestCase):
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Address": {
|
"Address": {
|
||||||
"Label": "Calle de Eloy Gonzalo, Madrid, España",
|
"Label": "Calle de Eloy Gonzalo, Madrid, Espana",
|
||||||
"Country": "ESP",
|
"Country": "ESP",
|
||||||
"State": "Comunidad de Madrid",
|
"State": "Comunidad de Madrid",
|
||||||
"County": "Madrid",
|
"County": "Madrid",
|
||||||
@ -65,7 +68,7 @@ class GeocoderTestCase(unittest.TestCase):
|
|||||||
"District": "Trafalgar",
|
"District": "Trafalgar",
|
||||||
"Street": "Calle de Eloy Gonzalo",
|
"Street": "Calle de Eloy Gonzalo",
|
||||||
"AdditionalData": [{
|
"AdditionalData": [{
|
||||||
"value": "España",
|
"value": "Espana",
|
||||||
"key": "CountryName"
|
"key": "CountryName"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -78,45 +81,52 @@ class GeocoderTestCase(unittest.TestCase):
|
|||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}]
|
}
|
||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
}
|
}"""
|
||||||
|
|
||||||
|
MALFORMED_RESPONSE = """{"manolo": "escobar"}"""
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.geocoder = heremapsgeocoder.Geocoder(None, None)
|
self.geocoder = HereMapsGeocoder(None, None)
|
||||||
|
|
||||||
def test_geocode_address_with_valid_params(self):
|
def test_geocode_address_with_valid_params(self, req_mock):
|
||||||
self.geocoder.perform_request = lambda x: self.GOOD_RESPONSE
|
req_mock.register_uri('GET', HereMapsGeocoder.PRODUCTION_GEOCODE_JSON_URL,
|
||||||
response = self.geocoder.geocode_address(
|
text=self.GOOD_RESPONSE)
|
||||||
|
response = self.geocoder.geocode(
|
||||||
searchtext='Calle Eloy Gonzalo 27',
|
searchtext='Calle Eloy Gonzalo 27',
|
||||||
city='Madrid',
|
city='Madrid',
|
||||||
country='España')
|
country='España')
|
||||||
|
|
||||||
def test_geocode_address_with_invalid_params(self):
|
self.assertEqual(response[0], -3.70126)
|
||||||
|
self.assertEqual(response[1], 40.43433)
|
||||||
|
|
||||||
|
def test_geocode_address_with_invalid_params(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', HereMapsGeocoder.PRODUCTION_GEOCODE_JSON_URL,
|
||||||
|
text=self.GOOD_RESPONSE)
|
||||||
with self.assertRaises(BadGeocodingParams):
|
with self.assertRaises(BadGeocodingParams):
|
||||||
self.geocoder.geocode_address(
|
self.geocoder.geocode(
|
||||||
searchtext='Calle Eloy Gonzalo 27',
|
searchtext='Calle Eloy Gonzalo 27',
|
||||||
manolo='escobar')
|
manolo='escobar')
|
||||||
|
|
||||||
def test_geocode_address_with_no_params(self):
|
def test_geocode_address_with_no_params(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', HereMapsGeocoder.PRODUCTION_GEOCODE_JSON_URL,
|
||||||
|
text=self.GOOD_RESPONSE)
|
||||||
with self.assertRaises(NoGeocodingParams):
|
with self.assertRaises(NoGeocodingParams):
|
||||||
self.geocoder.geocode_address()
|
self.geocoder.geocode()
|
||||||
|
|
||||||
def test_geocode_address_empty_response(self):
|
def test_geocode_address_empty_response(self, req_mock):
|
||||||
self.geocoder.perform_request = lambda x: self.EMPTY_RESPONSE
|
req_mock.register_uri('GET', HereMapsGeocoder.PRODUCTION_GEOCODE_JSON_URL,
|
||||||
with self.assertRaises(EmptyGeocoderResponse):
|
text=self.EMPTY_RESPONSE)
|
||||||
self.geocoder.geocode_address(searchtext='lkajfñlasjfñ')
|
result = self.geocoder.geocode(searchtext='lkajfñlasjfñ')
|
||||||
|
self.assertEqual(result, [])
|
||||||
def test_extract_lng_lat_from_result(self):
|
|
||||||
result = self.GOOD_RESPONSE['Response']['View'][0]['Result'][0]
|
|
||||||
coordinates = self.geocoder.extract_lng_lat_from_result(result)
|
|
||||||
|
|
||||||
self.assertEqual(coordinates[0], -3.70126)
|
|
||||||
self.assertEqual(coordinates[1], 40.43433)
|
|
||||||
|
|
||||||
def test_extract_lng_lat_from_result_with_malformed_result(self):
|
|
||||||
result = {'manolo': 'escobar'}
|
|
||||||
|
|
||||||
|
def test_geocode_with_malformed_result(self, req_mock):
|
||||||
|
req_mock.register_uri('GET', HereMapsGeocoder.PRODUCTION_GEOCODE_JSON_URL,
|
||||||
|
text=self.MALFORMED_RESPONSE)
|
||||||
with self.assertRaises(MalformedResult):
|
with self.assertRaises(MalformedResult):
|
||||||
self.geocoder.extract_lng_lat_from_result(result)
|
self.geocoder.geocode(
|
||||||
|
searchtext='Calle Eloy Gonzalo 27',
|
||||||
|
city='Madrid',
|
||||||
|
country='España')
|
@ -1,7 +1,7 @@
|
|||||||
import test_helper
|
import test_helper
|
||||||
from mockredis import MockRedis
|
from mockredis import MockRedis
|
||||||
from cartodb_geocoder import quota_service
|
from cartodb_services.metrics import QuotaService
|
||||||
from cartodb_geocoder import config_helper
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
from unittest import TestCase
|
from unittest import TestCase
|
||||||
from nose.tools import assert_raises
|
from nose.tools import assert_raises
|
||||||
from datetime import datetime, date
|
from datetime import datetime, date
|
||||||
@ -14,10 +14,6 @@ class TestQuotaService(TestCase):
|
|||||||
# organization user
|
# organization user
|
||||||
# org:<orgname>:<service>:<metric>:YYYYMM:DD
|
# org:<orgname>:<service>:<metric>:YYYYMM:DD
|
||||||
|
|
||||||
# def increment_geocoder_uses(self, username, orgname=None,
|
|
||||||
# date=date.today(), service='geocoder_here',
|
|
||||||
# metric='success_responses', amount=20):
|
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.redis_conn = MockRedis()
|
self.redis_conn = MockRedis()
|
||||||
|
|
||||||
@ -68,28 +64,19 @@ class TestQuotaService(TestCase):
|
|||||||
def test_should_check_user_increment_and_quota_check_correctly(self):
|
def test_should_check_user_increment_and_quota_check_correctly(self):
|
||||||
qs = self.__build_quota_service('test_user', quota=2)
|
qs = self.__build_quota_service('test_user', quota=2)
|
||||||
qs.increment_success_geocoder_use()
|
qs.increment_success_geocoder_use()
|
||||||
assert qs.check_user_quota() == True
|
assert qs.check_user_quota() is True
|
||||||
qs.increment_success_geocoder_use(amount=2)
|
qs.increment_success_geocoder_use(amount=2)
|
||||||
assert qs.check_user_quota() == False
|
assert qs.check_user_quota() is False
|
||||||
month = date.today().strftime('%Y%m')
|
month = date.today().strftime('%Y%m')
|
||||||
name = 'user:test_user:geocoder_here:total_requests:{0}'.format(month)
|
|
||||||
total_requests = self.redis_conn.zscore(name, date.today().day)
|
|
||||||
assert total_requests == 3
|
|
||||||
|
|
||||||
def test_should_check_org_increment_and_quota_check_correctly(self):
|
def test_should_check_org_increment_and_quota_check_correctly(self):
|
||||||
qs = self.__build_quota_service('test_user', orgname='test_org',
|
qs = self.__build_quota_service('test_user', orgname='test_org',
|
||||||
quota=2)
|
quota=2)
|
||||||
qs.increment_success_geocoder_use()
|
qs.increment_success_geocoder_use()
|
||||||
assert qs.check_user_quota() == True
|
assert qs.check_user_quota() is True
|
||||||
qs.increment_success_geocoder_use(amount=2)
|
qs.increment_success_geocoder_use(amount=2)
|
||||||
assert qs.check_user_quota() == False
|
assert qs.check_user_quota() is False
|
||||||
month = date.today().strftime('%Y%m')
|
month = date.today().strftime('%Y%m')
|
||||||
org_name = 'org:test_org:geocoder_here:total_requests:{0}'.format(month)
|
|
||||||
org_total_requests = self.redis_conn.zscore(org_name, date.today().day)
|
|
||||||
assert org_total_requests == 3
|
|
||||||
user_name = 'user:test_user:geocoder_here:total_requests:{0}'.format(month)
|
|
||||||
user_total_requests = self.redis_conn.zscore(user_name, date.today().day)
|
|
||||||
assert user_total_requests == 3
|
|
||||||
|
|
||||||
def __build_quota_service(self, username, quota=100, service='heremaps',
|
def __build_quota_service(self, username, quota=100, service='heremaps',
|
||||||
orgname=None, soft_limit=False,
|
orgname=None, soft_limit=False,
|
||||||
@ -101,9 +88,9 @@ class TestQuotaService(TestCase):
|
|||||||
if orgname:
|
if orgname:
|
||||||
test_helper.build_redis_org_config(self.redis_conn, orgname,
|
test_helper.build_redis_org_config(self.redis_conn, orgname,
|
||||||
quota=quota, end_date=end_date)
|
quota=quota, end_date=end_date)
|
||||||
geocoder_config = config_helper.GeocoderConfig(self.redis_conn,
|
geocoder_config = GeocoderConfig(self.redis_conn,
|
||||||
username, orgname,
|
username, orgname,
|
||||||
'nokia_id', 'nokia_cod')
|
'nokia_id', 'nokia_cod')
|
||||||
return quota_service.QuotaService(geocoder_config,
|
return QuotaService(geocoder_config,
|
||||||
redis_connection = self.redis_conn)
|
redis_connection = self.redis_conn)
|
||||||
|
|
@ -1,7 +1,7 @@
|
|||||||
import test_helper
|
import test_helper
|
||||||
from mockredis import MockRedis
|
from mockredis import MockRedis
|
||||||
from cartodb_geocoder import user_service
|
from cartodb_services.metrics import UserMetricsService
|
||||||
from cartodb_geocoder import config_helper
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
from datetime import datetime, date
|
from datetime import datetime, date
|
||||||
from unittest import TestCase
|
from unittest import TestCase
|
||||||
from nose.tools import assert_raises
|
from nose.tools import assert_raises
|
||||||
@ -84,7 +84,7 @@ class TestUserService(TestCase):
|
|||||||
if orgname:
|
if orgname:
|
||||||
test_helper.build_redis_org_config(self.redis_conn, orgname,
|
test_helper.build_redis_org_config(self.redis_conn, orgname,
|
||||||
quota=quota, end_date=end_date)
|
quota=quota, end_date=end_date)
|
||||||
geocoder_config = config_helper.GeocoderConfig(self.redis_conn,
|
geocoder_config = GeocoderConfig(self.redis_conn,
|
||||||
username, orgname,
|
username, orgname,
|
||||||
'nokia_id', 'nokia_cod')
|
'nokia_id', 'nokia_cod')
|
||||||
return user_service.UserService(geocoder_config, self.redis_conn)
|
return UserMetricsService(geocoder_config, self.redis_conn)
|
@ -1,4 +0,0 @@
|
|||||||
requests==2.9.1
|
|
||||||
|
|
||||||
# Test
|
|
||||||
nose==1.3.7
|
|
@ -1,41 +0,0 @@
|
|||||||
"""
|
|
||||||
A Here Maps API Python wrapper
|
|
||||||
|
|
||||||
See:
|
|
||||||
https://developer.here.com
|
|
||||||
https://github.com/CartoDB/geocoder-api
|
|
||||||
"""
|
|
||||||
|
|
||||||
from setuptools import setup, find_packages
|
|
||||||
|
|
||||||
setup(
|
|
||||||
name='heremaps',
|
|
||||||
|
|
||||||
version='0.0.1',
|
|
||||||
|
|
||||||
description='A Here Maps API Python wrapper',
|
|
||||||
|
|
||||||
url='https://github.com/CartoDB/geocoder-api',
|
|
||||||
|
|
||||||
author='Data Services Team - CartoDB',
|
|
||||||
author_email='dataservices@cartodb.com',
|
|
||||||
|
|
||||||
license='MIT',
|
|
||||||
|
|
||||||
classifiers=[
|
|
||||||
'Development Status :: 5 - Production',
|
|
||||||
'Intended Audience :: Mapping comunity',
|
|
||||||
'Topic :: Maps :: Mapping Tools',
|
|
||||||
'License :: OSI Approved :: MIT License',
|
|
||||||
'Programming Language :: Python :: 2.7',
|
|
||||||
],
|
|
||||||
|
|
||||||
keywords='maps api mapping tools',
|
|
||||||
|
|
||||||
packages=find_packages(exclude=['contrib', 'docs', 'tests']),
|
|
||||||
|
|
||||||
extras_require={
|
|
||||||
'dev': ['unittest'],
|
|
||||||
'test': ['unittest'],
|
|
||||||
}
|
|
||||||
)
|
|
Loading…
Reference in New Issue
Block a user