Compare commits
34 Commits
python-0.7
...
python-0.8
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
85d6c2a54e | ||
|
|
cad2051efe | ||
|
|
96a93e3c56 | ||
|
|
facda9e8be | ||
|
|
64fc18b9e0 | ||
|
|
9381d5644b | ||
|
|
9f55f2ee3b | ||
|
|
1087c1266b | ||
|
|
d5a296a30c | ||
|
|
f8caf4314d | ||
|
|
d7910fbbf1 | ||
|
|
d47049c813 | ||
|
|
cc8f93c535 | ||
|
|
3f9441de7e | ||
|
|
fe41359a1f | ||
|
|
46a934b178 | ||
|
|
184358bdec | ||
|
|
a6d546f2ee | ||
|
|
fc99f7aba9 | ||
|
|
e959873b32 | ||
|
|
a98093540d | ||
|
|
78add220cd | ||
|
|
cf2f86136b | ||
|
|
fb183b07ee | ||
|
|
5ab727bcb6 | ||
|
|
1e9b551160 | ||
|
|
699dc9bf0e | ||
|
|
fc291a7c63 | ||
|
|
d73af32c2c | ||
|
|
7ea88fa051 | ||
|
|
d2ca40cf38 | ||
|
|
18ae2525b6 | ||
|
|
06462fdf7a | ||
|
|
71d5ce951a |
23
NEWS.md
23
NEWS.md
@@ -1,3 +1,26 @@
|
|||||||
|
September 28, 2016
|
||||||
|
==========
|
||||||
|
* Released version 0.8.1 of Python package cartodb\_services
|
||||||
|
* Improvements in QPS retry decorator for requests to external services
|
||||||
|
|
||||||
|
September 8, 2016
|
||||||
|
===========
|
||||||
|
* Released version 0.11.1 of the client
|
||||||
|
* Minor change in the name of the function parameter sent to server and Observatory backend for compatibility with the last observatory-extension framework updates
|
||||||
|
|
||||||
|
September 1, 2016
|
||||||
|
===========
|
||||||
|
* Released version 0.11.0 of the client
|
||||||
|
* Include DS table functions to create and populate a table with the GetMeasure function in observatory
|
||||||
|
* Released version 0.15.1 of the server
|
||||||
|
* Rename DS table functions
|
||||||
|
|
||||||
|
August 29, 2016
|
||||||
|
===========
|
||||||
|
* Released version 0.15.0 of the server
|
||||||
|
* Geocode namedplace point functions uses Mapzen search service and in case of error
|
||||||
|
it'll use the internal geocoder
|
||||||
|
|
||||||
August 19, 2016
|
August 19, 2016
|
||||||
===========
|
===========
|
||||||
* Released version 0.7.4.2 of the server python library
|
* Released version 0.7.4.2 of the server python library
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ The CARTO Data Services SQL API
|
|||||||
Steps to deploy a new Data Services API version :
|
Steps to deploy a new Data Services API version :
|
||||||
|
|
||||||
- Deploy new version of dataservices API to all servers
|
- Deploy new version of dataservices API to all servers
|
||||||
- Update the server user using: ALTER EXTENSION cdb_dataservices_server UPDATE TO '<CURRENT_VERSION>';
|
- Update the server user using: ALTER EXTENSION cdb_dataservices_server UPDATE TO '\<CURRENT_VERSION\>';
|
||||||
- Update the python dependencies if needed: **cartodb_geocoder** and **heremaps**
|
- Update the python dependencies if needed: **cartodb_geocoder** and **heremaps**
|
||||||
- Add the needed config in the `cdb_conf` table:
|
- Add the needed config in the `cdb_conf` table:
|
||||||
- `redis_metadata_config` and `redis_metrics_conf`
|
- `redis_metadata_config` and `redis_metrics_conf`
|
||||||
|
|||||||
@@ -13,8 +13,8 @@ OLD_VERSIONS = $(wildcard old_versions/*.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) \
|
||||||
$(OLD_VERSIONS) \
|
$(OLD_VERSIONS) \
|
||||||
cdb_dataservices_client--0.10.1--0.10.2.sql \
|
cdb_dataservices_client--0.11.0--0.11.1.sql \
|
||||||
cdb_dataservices_client--0.10.2--0.10.1.sql
|
cdb_dataservices_client--0.11.1--0.11.0.sql
|
||||||
|
|
||||||
|
|
||||||
REGRESS = $(notdir $(basename $(wildcard test/sql/*test.sql)))
|
REGRESS = $(notdir $(basename $(wildcard test/sql/*test.sql)))
|
||||||
|
|||||||
140
client/cdb_dataservices_client--0.11.0--0.11.1.sql
Normal file
140
client/cdb_dataservices_client--0.11.0--0.11.1.sql
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_client UPDATE TO '0.11.1'" to load this file. \quit
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'OBS_GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
|
.format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute('CREATE TABLE "{schema}".{table_name} ( '
|
||||||
|
'cartodb_id int, the_geom geometry, {columns_with_types} '
|
||||||
|
');'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, columns_with_types=columns_with_types)
|
||||||
|
)
|
||||||
|
|
||||||
|
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'OBS_GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute(
|
||||||
|
"SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)))
|
||||||
|
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [
|
||||||
|
colnames_arr[i] +
|
||||||
|
' ' +
|
||||||
|
coltypes_arr[i] for i in range(
|
||||||
|
0,
|
||||||
|
len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
aliased_colname_list = ','.join(
|
||||||
|
['result.' + name for name in colnames_arr])
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute(
|
||||||
|
"SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, "
|
||||||
|
"{schema}::text, {dbname}::text, {table_name}::text);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_db_role=plpy.quote_literal(user_db_role),
|
||||||
|
schema=plpy.quote_literal(user_schema),
|
||||||
|
dbname=plpy.quote_literal(dbname),
|
||||||
|
table_name=plpy.quote_literal(table_name)))
|
||||||
|
|
||||||
|
if ds_fdw_metadata[0]["schemaname"]:
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error connecting dataset via FDW')
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute(
|
||||||
|
'INSERT INTO "{schema}".{analysis_table_name} '
|
||||||
|
'SELECT ut.cartodb_id, ut.the_geom, {colname_list} '
|
||||||
|
'FROM "{schema}".{table_name} ut '
|
||||||
|
'LEFT JOIN _DST_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, '
|
||||||
|
'{function_name}::text, {params}::json) '
|
||||||
|
'AS result ({columns_with_types}, cartodb_id int) '
|
||||||
|
'ON result.cartodb_id = ut.cartodb_id;' .format(
|
||||||
|
schema=user_schema,
|
||||||
|
analysis_table_name=output_table_name,
|
||||||
|
colname_list=aliased_colname_list,
|
||||||
|
table_name=table_name,
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params),
|
||||||
|
columns_with_types=columns_with_types))
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute(
|
||||||
|
"SELECT cdb_dataservices_client._DST_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, "
|
||||||
|
"{server_table_name}::text, {fdw_server}::text)" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
fdw_server=plpy.quote_literal(server_name)))
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
140
client/cdb_dataservices_client--0.11.1--0.11.0.sql
Normal file
140
client/cdb_dataservices_client--0.11.1--0.11.0.sql
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_client UPDATE TO '0.11.0'" to load this file. \quit
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
|
.format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute('CREATE TABLE "{schema}".{table_name} ( '
|
||||||
|
'cartodb_id int, the_geom geometry, {columns_with_types} '
|
||||||
|
');'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, columns_with_types=columns_with_types)
|
||||||
|
)
|
||||||
|
|
||||||
|
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute(
|
||||||
|
"SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)))
|
||||||
|
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [
|
||||||
|
colnames_arr[i] +
|
||||||
|
' ' +
|
||||||
|
coltypes_arr[i] for i in range(
|
||||||
|
0,
|
||||||
|
len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
aliased_colname_list = ','.join(
|
||||||
|
['result.' + name for name in colnames_arr])
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute(
|
||||||
|
"SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, "
|
||||||
|
"{schema}::text, {dbname}::text, {table_name}::text);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_db_role=plpy.quote_literal(user_db_role),
|
||||||
|
schema=plpy.quote_literal(user_schema),
|
||||||
|
dbname=plpy.quote_literal(dbname),
|
||||||
|
table_name=plpy.quote_literal(table_name)))
|
||||||
|
|
||||||
|
if ds_fdw_metadata[0]["schemaname"]:
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error connecting dataset via FDW')
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute(
|
||||||
|
'INSERT INTO "{schema}".{analysis_table_name} '
|
||||||
|
'SELECT ut.cartodb_id, ut.the_geom, {colname_list} '
|
||||||
|
'FROM "{schema}".{table_name} ut '
|
||||||
|
'LEFT JOIN _DST_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, '
|
||||||
|
'{function_name}::text, {params}::json) '
|
||||||
|
'AS result ({columns_with_types}, cartodb_id int) '
|
||||||
|
'ON result.cartodb_id = ut.cartodb_id;' .format(
|
||||||
|
schema=user_schema,
|
||||||
|
analysis_table_name=output_table_name,
|
||||||
|
colname_list=aliased_colname_list,
|
||||||
|
table_name=table_name,
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params),
|
||||||
|
columns_with_types=columns_with_types))
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute(
|
||||||
|
"SELECT cdb_dataservices_client._DST_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, "
|
||||||
|
"{server_table_name}::text, {fdw_server}::text)" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
fdw_server=plpy.quote_literal(server_name)))
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
1792
client/cdb_dataservices_client--0.11.1.sql
Normal file
1792
client/cdb_dataservices_client--0.11.1.sql
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,5 @@
|
|||||||
comment = 'CartoDB dataservices client API extension'
|
comment = 'CartoDB dataservices client API extension'
|
||||||
default_version = '0.10.2'
|
default_version = '0.11.1'
|
||||||
requires = 'plproxy, cartodb'
|
requires = 'plproxy, cartodb'
|
||||||
superuser = true
|
superuser = true
|
||||||
schema = cdb_dataservices_client
|
schema = cdb_dataservices_client
|
||||||
|
|||||||
289
client/old_versions/cdb_dataservices_client--0.10.2--0.11.0.sql
Normal file
289
client/old_versions/cdb_dataservices_client--0.10.2--0.11.0.sql
Normal file
@@ -0,0 +1,289 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_client UPDATE TO '0.11.0'" to load this file. \quit
|
||||||
|
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_GetTable(text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_AugmentTable(text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client.__OBS_AugmentTable(text, text, text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client.__OBS_GetTable(text, text, text, text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_ConnectUserTable(text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_GetReturnMetadata(text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_FetchJoinFdwTableData(text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._OBS_DisconnectUserTable(text, text, text, text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure(
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
DECLARE
|
||||||
|
username text;
|
||||||
|
user_db_role text;
|
||||||
|
orgname text;
|
||||||
|
user_schema text;
|
||||||
|
result boolean;
|
||||||
|
BEGIN
|
||||||
|
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
||||||
|
RAISE EXCEPTION 'The api_key must be provided';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT session_user INTO user_db_role;
|
||||||
|
|
||||||
|
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
||||||
|
-- JSON value stored "" is taken as literal
|
||||||
|
IF username IS NULL OR username = '' OR username = '""' THEN
|
||||||
|
RAISE EXCEPTION 'Username is a mandatory argument';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
||||||
|
user_schema := 'public';
|
||||||
|
ELSE
|
||||||
|
user_schema := username;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
|
username,
|
||||||
|
orgname,
|
||||||
|
user_db_role,
|
||||||
|
user_schema,
|
||||||
|
output_table_name,
|
||||||
|
params
|
||||||
|
) INTO result;
|
||||||
|
|
||||||
|
RETURN result;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure(
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
DECLARE
|
||||||
|
username text;
|
||||||
|
user_db_role text;
|
||||||
|
orgname text;
|
||||||
|
dbname text;
|
||||||
|
user_schema text;
|
||||||
|
result boolean;
|
||||||
|
BEGIN
|
||||||
|
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
||||||
|
RAISE EXCEPTION 'The api_key must be provided';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT session_user INTO user_db_role;
|
||||||
|
|
||||||
|
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
||||||
|
-- JSON value stored "" is taken as literal
|
||||||
|
IF username IS NULL OR username = '' OR username = '""' THEN
|
||||||
|
RAISE EXCEPTION 'Username is a mandatory argument';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
||||||
|
user_schema := 'public';
|
||||||
|
ELSE
|
||||||
|
user_schema := username;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT current_database() INTO dbname;
|
||||||
|
|
||||||
|
SELECT cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username,
|
||||||
|
orgname,
|
||||||
|
user_db_role,
|
||||||
|
user_schema,
|
||||||
|
dbname,
|
||||||
|
table_name,
|
||||||
|
output_table_name,
|
||||||
|
params
|
||||||
|
) INTO result;
|
||||||
|
|
||||||
|
RETURN result;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
|
.format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute('CREATE TABLE "{schema}".{table_name} ( '
|
||||||
|
'cartodb_id int, the_geom geometry, {columns_with_types} '
|
||||||
|
');'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, columns_with_types=columns_with_types)
|
||||||
|
)
|
||||||
|
|
||||||
|
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute(
|
||||||
|
"SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)))
|
||||||
|
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [
|
||||||
|
colnames_arr[i] +
|
||||||
|
' ' +
|
||||||
|
coltypes_arr[i] for i in range(
|
||||||
|
0,
|
||||||
|
len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
aliased_colname_list = ','.join(
|
||||||
|
['result.' + name for name in colnames_arr])
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute(
|
||||||
|
"SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, "
|
||||||
|
"{schema}::text, {dbname}::text, {table_name}::text);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_db_role=plpy.quote_literal(user_db_role),
|
||||||
|
schema=plpy.quote_literal(user_schema),
|
||||||
|
dbname=plpy.quote_literal(dbname),
|
||||||
|
table_name=plpy.quote_literal(table_name)))
|
||||||
|
|
||||||
|
if ds_fdw_metadata[0]["schemaname"]:
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error connecting dataset via FDW')
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute(
|
||||||
|
'INSERT INTO "{schema}".{analysis_table_name} '
|
||||||
|
'SELECT ut.cartodb_id, ut.the_geom, {colname_list} '
|
||||||
|
'FROM "{schema}".{table_name} ut '
|
||||||
|
'LEFT JOIN _DST_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, '
|
||||||
|
'{function_name}::text, {params}::json) '
|
||||||
|
'AS result ({columns_with_types}, cartodb_id int) '
|
||||||
|
'ON result.cartodb_id = ut.cartodb_id;' .format(
|
||||||
|
schema=user_schema,
|
||||||
|
analysis_table_name=output_table_name,
|
||||||
|
colname_list=aliased_colname_list,
|
||||||
|
table_name=table_name,
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params),
|
||||||
|
columns_with_types=columns_with_types))
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute(
|
||||||
|
"SELECT cdb_dataservices_client._DST_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, "
|
||||||
|
"{server_table_name}::text, {fdw_server}::text)" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
fdw_server=plpy.quote_literal(server_name)))
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_ConnectUserTable(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text
|
||||||
|
)RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_ConnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_GetReturnMetadata(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
function_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_GetReturnMetadata;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_FetchJoinFdwTableData(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
table_schema text,
|
||||||
|
table_name text,
|
||||||
|
function_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS SETOF record AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_FetchJoinFdwTableData;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_DisconnectUserTable(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
table_schema text,
|
||||||
|
table_name text,
|
||||||
|
server_name text
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_DisconnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure(output_table_name text, params json) TO publicuser;
|
||||||
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure(table_name text, output_table_name text, params json) TO publicuser;
|
||||||
|
|
||||||
281
client/old_versions/cdb_dataservices_client--0.11.0--0.10.2.sql
Normal file
281
client/old_versions/cdb_dataservices_client--0.11.0--0.10.2.sql
Normal file
@@ -0,0 +1,281 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_client UPDATE TO '0.10.2'" to load this file. \quit
|
||||||
|
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure(text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure(text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(text, text, text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_ConnectUserTable(text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_GetReturnMetadata(text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_FetchJoinFdwTableData(text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_client._DST_DisconnectUserTable(text, text, text, text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_GetTable(table_name text, output_table_name text, function_name text, params json)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
DECLARE
|
||||||
|
username text;
|
||||||
|
user_db_role text;
|
||||||
|
orgname text;
|
||||||
|
dbname text;
|
||||||
|
user_schema text;
|
||||||
|
result boolean;
|
||||||
|
BEGIN
|
||||||
|
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
||||||
|
RAISE EXCEPTION 'The api_key must be provided';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT session_user INTO user_db_role;
|
||||||
|
|
||||||
|
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
||||||
|
-- JSON value stored "" is taken as literal
|
||||||
|
IF username IS NULL OR username = '' OR username = '""' THEN
|
||||||
|
RAISE EXCEPTION 'Username is a mandatory argument';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
||||||
|
user_schema := 'public';
|
||||||
|
ELSE
|
||||||
|
user_schema := username;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT current_database() INTO dbname;
|
||||||
|
|
||||||
|
SELECT cdb_dataservices_client.__OBS_GetTable(username, orgname, user_db_role, user_schema, dbname, table_name, output_table_name, function_name, params) INTO result;
|
||||||
|
|
||||||
|
RETURN result;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_AugmentTable(table_name text, function_name text, params json)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
DECLARE
|
||||||
|
username text;
|
||||||
|
user_db_role text;
|
||||||
|
orgname text;
|
||||||
|
dbname text;
|
||||||
|
user_schema text;
|
||||||
|
result boolean;
|
||||||
|
BEGIN
|
||||||
|
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
||||||
|
RAISE EXCEPTION 'The api_key must be provided';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT session_user INTO user_db_role;
|
||||||
|
|
||||||
|
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
||||||
|
-- JSON value stored "" is taken as literal
|
||||||
|
IF username IS NULL OR username = '' OR username = '""' THEN
|
||||||
|
RAISE EXCEPTION 'Username is a mandatory argument';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
||||||
|
user_schema := 'public';
|
||||||
|
ELSE
|
||||||
|
user_schema := username;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT current_database() INTO dbname;
|
||||||
|
|
||||||
|
SELECT cdb_dataservices_client.__OBS_AugmentTable(username, orgname, user_db_role, user_schema, dbname, table_name, function_name, params) INTO result;
|
||||||
|
|
||||||
|
RETURN result;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__OBS_AugmentTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text, function_name text, params json)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
from time import strftime
|
||||||
|
try:
|
||||||
|
server_table_name = None
|
||||||
|
temporary_table_name = 'ds_tmp_' + str(strftime("%s")) + table_name
|
||||||
|
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._OBS_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params))
|
||||||
|
)
|
||||||
|
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
colnames = ','.join(colnames_arr)
|
||||||
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute("SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {user_schema}::text, {dbname}::text, {table_name}::text);"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), user_schema=plpy.quote_literal(user_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name))
|
||||||
|
)
|
||||||
|
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
|
||||||
|
# Create temporary table with the augmented results
|
||||||
|
plpy.execute('CREATE UNLOGGED TABLE "{user_schema}".{temp_table_name} AS '
|
||||||
|
'(SELECT {columns}, cartodb_id '
|
||||||
|
'FROM cdb_dataservices_client._OBS_FetchJoinFdwTableData('
|
||||||
|
'{username}::text, {orgname}::text, {schema}::text, {table_name}::text, {function_name}::text, {params}::json) '
|
||||||
|
'AS results({columns_with_types}, cartodb_id int) )'
|
||||||
|
.format(columns=colnames, username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_schema=user_schema, schema=plpy.quote_literal(server_schema), table_name=plpy.quote_literal(server_table_name),
|
||||||
|
function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params), columns_with_types=columns_with_types,
|
||||||
|
temp_table_name=temporary_table_name)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
||||||
|
)
|
||||||
|
|
||||||
|
# Add index to cartodb_id
|
||||||
|
plpy.execute('CREATE UNIQUE INDEX {temp_table_name}_pkey ON "{user_schema}".{temp_table_name} (cartodb_id)'
|
||||||
|
.format(user_schema=user_schema, temp_table_name=temporary_table_name)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Prepare table to receive augmented results in new columns
|
||||||
|
for idx, column in enumerate(colnames_arr):
|
||||||
|
if colnames_arr[idx] is not 'the_geom':
|
||||||
|
plpy.execute('ALTER TABLE "{user_schema}".{table_name} ADD COLUMN {column_name} {column_type}'
|
||||||
|
.format(user_schema=user_schema, table_name=table_name, column_name=colnames_arr[idx], column_type=coltypes_arr[idx])
|
||||||
|
)
|
||||||
|
|
||||||
|
# Populate the user table with the augmented results
|
||||||
|
plpy.execute('UPDATE "{user_schema}".{table_name} SET {columns} = '
|
||||||
|
'(SELECT {columns} FROM "{user_schema}".{temporary_table_name} '
|
||||||
|
'WHERE "{user_schema}".{temporary_table_name}.cartodb_id = "{user_schema}".{table_name}.cartodb_id)'
|
||||||
|
.format(columns = colnames, username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_schema = user_schema, table_name=table_name, function_name=function_name, params=params, columns_with_types=columns_with_types,
|
||||||
|
temporary_table_name=temporary_table_name)
|
||||||
|
)
|
||||||
|
|
||||||
|
plpy.execute('DROP TABLE IF EXISTS "{user_schema}".{temporary_table_name}'
|
||||||
|
.format(user_schema=user_schema, table_name=table_name, temporary_table_name=temporary_table_name)
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
except Exception as e:
|
||||||
|
plpy.warning('Error trying to augment table {0}'.format(e))
|
||||||
|
# Wipe user FDW data from the server in case of failure if the table was connected
|
||||||
|
if server_table_name:
|
||||||
|
# Wipe local temporary table
|
||||||
|
plpy.execute('DROP TABLE IF EXISTS "{user_schema}".{temporary_table_name}'
|
||||||
|
.format(user_schema=user_schema, table_name=table_name, temporary_table_name=temporary_table_name)
|
||||||
|
)
|
||||||
|
|
||||||
|
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__OBS_GetTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text, output_table_name text, function_name text, params json)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
try:
|
||||||
|
server_table_name = None
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._OBS_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params))
|
||||||
|
)
|
||||||
|
|
||||||
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
colnames = ','.join(colnames_arr)
|
||||||
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute("SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {table_name}::text);"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(user_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name))
|
||||||
|
)
|
||||||
|
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
|
||||||
|
# Get list of user columns to include in the new table
|
||||||
|
user_table_columns = ','.join(
|
||||||
|
plpy.execute('SELECT array_agg(\'user_table.\' || attname) AS columns '
|
||||||
|
'FROM pg_attribute WHERE attrelid = \'"{user_schema}".{table_name}\'::regclass '
|
||||||
|
'AND attnum > 0 AND NOT attisdropped AND attname NOT LIKE \'the_geom_webmercator\' '
|
||||||
|
'AND NOT attname LIKE ANY(string_to_array(\'{colnames}\',\',\'));'
|
||||||
|
.format(user_schema=user_schema, table_name=table_name, colnames=colnames)
|
||||||
|
)[0]["columns"]
|
||||||
|
)
|
||||||
|
|
||||||
|
# Populate a new table with the augmented results
|
||||||
|
plpy.execute('CREATE TABLE "{user_schema}".{output_table_name} AS '
|
||||||
|
'(SELECT results.{columns}, {user_table_columns} '
|
||||||
|
'FROM {table_name} AS user_table '
|
||||||
|
'LEFT JOIN cdb_dataservices_client._OBS_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {function_name}::text, {params}::json) as results({columns_with_types}, cartodb_id int) '
|
||||||
|
'ON results.cartodb_id = user_table.cartodb_id)'
|
||||||
|
.format(output_table_name=output_table_name, columns=colnames, user_table_columns=user_table_columns, username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname), user_schema=user_schema, server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
table_name=table_name, function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params), columns_with_types=columns_with_types)
|
||||||
|
)
|
||||||
|
|
||||||
|
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
||||||
|
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
except Exception as e:
|
||||||
|
plpy.warning('Error trying to get table {0}'.format(e))
|
||||||
|
# Wipe user FDW data from the server in case of failure if the table was connected
|
||||||
|
if server_table_name:
|
||||||
|
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_ConnectUserTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text)
|
||||||
|
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
||||||
|
CONNECT _server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._OBS_ConnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
|
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
||||||
|
CONNECT _server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._OBS_GetReturnMetadata;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
|
RETURNS SETOF record AS $$
|
||||||
|
CONNECT _server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._OBS_FetchJoinFdwTableData;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, server_name text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
CONNECT _server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._OBS_DisconnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._obs_augmenttable(table_name text, function_name text, params json) TO publicuser;
|
||||||
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._obs_gettable(table_name text, output_table_name text, function_name text, params json) TO publicuser;
|
||||||
1792
client/old_versions/cdb_dataservices_client--0.11.0.sql
Normal file
1792
client/old_versions/cdb_dataservices_client--0.11.0.sql
Normal file
File diff suppressed because it is too large
Load Diff
@@ -33,7 +33,6 @@
|
|||||||
- { name: admin1_name, type: text}
|
- { name: admin1_name, type: text}
|
||||||
- { name: country_name, type: text}
|
- { name: country_name, type: text}
|
||||||
|
|
||||||
|
|
||||||
- name: cdb_geocode_postalcode_polygon
|
- name: cdb_geocode_postalcode_polygon
|
||||||
return_type: Geometry
|
return_type: Geometry
|
||||||
params:
|
params:
|
||||||
|
|||||||
@@ -1,8 +1,53 @@
|
|||||||
CREATE TYPE cdb_dataservices_client.ds_fdw_metadata as (schemaname text, tabname text, servername text);
|
CREATE TYPE cdb_dataservices_client.ds_fdw_metadata as (schemaname text, tabname text, servername text);
|
||||||
CREATE TYPE cdb_dataservices_client.ds_return_metadata as (colnames text[], coltypes text[]);
|
CREATE TYPE cdb_dataservices_client.ds_return_metadata as (colnames text[], coltypes text[]);
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_GetTable(table_name text, output_table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure(
|
||||||
RETURNS boolean AS $$
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
DECLARE
|
||||||
|
username text;
|
||||||
|
user_db_role text;
|
||||||
|
orgname text;
|
||||||
|
user_schema text;
|
||||||
|
result boolean;
|
||||||
|
BEGIN
|
||||||
|
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
||||||
|
RAISE EXCEPTION 'The api_key must be provided';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT session_user INTO user_db_role;
|
||||||
|
|
||||||
|
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
||||||
|
-- JSON value stored "" is taken as literal
|
||||||
|
IF username IS NULL OR username = '' OR username = '""' THEN
|
||||||
|
RAISE EXCEPTION 'Username is a mandatory argument';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
||||||
|
user_schema := 'public';
|
||||||
|
ELSE
|
||||||
|
user_schema := username;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
SELECT cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
|
username,
|
||||||
|
orgname,
|
||||||
|
user_db_role,
|
||||||
|
user_schema,
|
||||||
|
output_table_name,
|
||||||
|
params
|
||||||
|
) INTO result;
|
||||||
|
|
||||||
|
RETURN result;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure(
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
DECLARE
|
DECLARE
|
||||||
username text;
|
username text;
|
||||||
user_db_role text;
|
user_db_role text;
|
||||||
@@ -31,238 +76,200 @@ BEGIN
|
|||||||
|
|
||||||
SELECT current_database() INTO dbname;
|
SELECT current_database() INTO dbname;
|
||||||
|
|
||||||
SELECT cdb_dataservices_client.__OBS_GetTable(username, orgname, user_db_role, user_schema, dbname, table_name, output_table_name, function_name, params) INTO result;
|
SELECT cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username,
|
||||||
|
orgname,
|
||||||
|
user_db_role,
|
||||||
|
user_schema,
|
||||||
|
dbname,
|
||||||
|
table_name,
|
||||||
|
output_table_name,
|
||||||
|
params
|
||||||
|
) INTO result;
|
||||||
|
|
||||||
RETURN result;
|
RETURN result;
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
||||||
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_AugmentTable(table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PrepareTableOBS_GetMeasure(
|
||||||
RETURNS boolean AS $$
|
username text,
|
||||||
DECLARE
|
orgname text,
|
||||||
username text;
|
user_db_role text,
|
||||||
user_db_role text;
|
user_schema text,
|
||||||
orgname text;
|
output_table_name text,
|
||||||
dbname text;
|
params json
|
||||||
user_schema text;
|
) RETURNS boolean AS $$
|
||||||
result boolean;
|
function_name = 'OBS_GetMeasure'
|
||||||
BEGIN
|
# Obtain return types for augmentation procedure
|
||||||
IF session_user = 'publicuser' OR session_user ~ 'cartodb_publicuser_*' THEN
|
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
||||||
RAISE EXCEPTION 'The api_key must be provided';
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
||||||
END IF;
|
.format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
SELECT session_user INTO user_db_role;
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
SELECT u, o INTO username, orgname FROM cdb_dataservices_client._cdb_entity_config() AS (u text, o text);
|
params=plpy.quote_literal(params)
|
||||||
-- JSON value stored "" is taken as literal
|
|
||||||
IF username IS NULL OR username = '' OR username = '""' THEN
|
|
||||||
RAISE EXCEPTION 'Username is a mandatory argument';
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF orgname IS NULL OR orgname = '' OR orgname = '""' THEN
|
|
||||||
user_schema := 'public';
|
|
||||||
ELSE
|
|
||||||
user_schema := username;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
SELECT current_database() INTO dbname;
|
|
||||||
|
|
||||||
SELECT cdb_dataservices_client.__OBS_AugmentTable(username, orgname, user_db_role, user_schema, dbname, table_name, function_name, params) INTO result;
|
|
||||||
|
|
||||||
RETURN result;
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE 'plpgsql' SECURITY DEFINER;
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__OBS_AugmentTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text, function_name text, params json)
|
|
||||||
RETURNS boolean AS $$
|
|
||||||
from time import strftime
|
|
||||||
try:
|
|
||||||
server_table_name = None
|
|
||||||
temporary_table_name = 'ds_tmp_' + str(strftime("%s")) + table_name
|
|
||||||
|
|
||||||
# Obtain return types for augmentation procedure
|
|
||||||
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
|
||||||
"FROM cdb_dataservices_client._OBS_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params))
|
|
||||||
)
|
)
|
||||||
|
)
|
||||||
|
if ds_return_metadata[0]["colnames"]:
|
||||||
colnames_arr = ds_return_metadata[0]["colnames"]
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
|
else:
|
||||||
# Prepare column and type strings required in the SQL queries
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
colnames = ','.join(colnames_arr)
|
|
||||||
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
|
||||||
columns_with_types = ','.join(columns_with_types_arr)
|
|
||||||
|
|
||||||
|
|
||||||
# Instruct the OBS server side to establish a FDW
|
# Prepare column and type strings required in the SQL queries
|
||||||
# The metadata is obtained as well in order to:
|
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
||||||
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
# - (b) be able to tell OBS to free resources when done.
|
|
||||||
ds_fdw_metadata = plpy.execute("SELECT schemaname, tabname, servername "
|
|
||||||
"FROM cdb_dataservices_client._OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {user_schema}::text, {dbname}::text, {table_name}::text);"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), user_schema=plpy.quote_literal(user_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name))
|
|
||||||
)
|
|
||||||
|
|
||||||
server_schema = ds_fdw_metadata[0]["schemaname"]
|
# Create a new table with the required columns
|
||||||
server_table_name = ds_fdw_metadata[0]["tabname"]
|
plpy.execute('CREATE TABLE "{schema}".{table_name} ( '
|
||||||
server_name = ds_fdw_metadata[0]["servername"]
|
'cartodb_id int, the_geom geometry, {columns_with_types} '
|
||||||
|
');'
|
||||||
# Create temporary table with the augmented results
|
.format(schema=user_schema, table_name=output_table_name, columns_with_types=columns_with_types)
|
||||||
plpy.execute('CREATE UNLOGGED TABLE "{user_schema}".{temp_table_name} AS '
|
|
||||||
'(SELECT {columns}, cartodb_id '
|
|
||||||
'FROM cdb_dataservices_client._OBS_FetchJoinFdwTableData('
|
|
||||||
'{username}::text, {orgname}::text, {schema}::text, {table_name}::text, {function_name}::text, {params}::json) '
|
|
||||||
'AS results({columns_with_types}, cartodb_id int) )'
|
|
||||||
.format(columns=colnames, username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname),
|
|
||||||
user_schema=user_schema, schema=plpy.quote_literal(server_schema), table_name=plpy.quote_literal(server_table_name),
|
|
||||||
function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params), columns_with_types=columns_with_types,
|
|
||||||
temp_table_name=temporary_table_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
# Wipe user FDW data from the server
|
|
||||||
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
|
||||||
)
|
|
||||||
|
|
||||||
# Add index to cartodb_id
|
|
||||||
plpy.execute('CREATE UNIQUE INDEX {temp_table_name}_pkey ON "{user_schema}".{temp_table_name} (cartodb_id)'
|
|
||||||
.format(user_schema=user_schema, temp_table_name=temporary_table_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
# Prepare table to receive augmented results in new columns
|
|
||||||
for idx, column in enumerate(colnames_arr):
|
|
||||||
if colnames_arr[idx] is not 'the_geom':
|
|
||||||
plpy.execute('ALTER TABLE "{user_schema}".{table_name} ADD COLUMN {column_name} {column_type}'
|
|
||||||
.format(user_schema=user_schema, table_name=table_name, column_name=colnames_arr[idx], column_type=coltypes_arr[idx])
|
|
||||||
)
|
|
||||||
|
|
||||||
# Populate the user table with the augmented results
|
|
||||||
plpy.execute('UPDATE "{user_schema}".{table_name} SET {columns} = '
|
|
||||||
'(SELECT {columns} FROM "{user_schema}".{temporary_table_name} '
|
|
||||||
'WHERE "{user_schema}".{temporary_table_name}.cartodb_id = "{user_schema}".{table_name}.cartodb_id)'
|
|
||||||
.format(columns = colnames, username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname),
|
|
||||||
user_schema = user_schema, table_name=table_name, function_name=function_name, params=params, columns_with_types=columns_with_types,
|
|
||||||
temporary_table_name=temporary_table_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
plpy.execute('DROP TABLE IF EXISTS "{user_schema}".{temporary_table_name}'
|
|
||||||
.format(user_schema=user_schema, table_name=table_name, temporary_table_name=temporary_table_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
return True
|
|
||||||
except Exception as e:
|
|
||||||
plpy.warning('Error trying to augment table {0}'.format(e))
|
|
||||||
# Wipe user FDW data from the server in case of failure if the table was connected
|
|
||||||
if server_table_name:
|
|
||||||
# Wipe local temporary table
|
|
||||||
plpy.execute('DROP TABLE IF EXISTS "{user_schema}".{temporary_table_name}'
|
|
||||||
.format(user_schema=user_schema, table_name=table_name, temporary_table_name=temporary_table_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
|
||||||
)
|
|
||||||
return False
|
|
||||||
$$ LANGUAGE plpythonu;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__OBS_GetTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text, output_table_name text, function_name text, params json)
|
|
||||||
RETURNS boolean AS $$
|
|
||||||
try:
|
|
||||||
server_table_name = None
|
|
||||||
# Obtain return types for augmentation procedure
|
|
||||||
ds_return_metadata = plpy.execute("SELECT colnames, coltypes "
|
|
||||||
"FROM cdb_dataservices_client._OBS_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params))
|
|
||||||
)
|
|
||||||
|
|
||||||
colnames_arr = ds_return_metadata[0]["colnames"]
|
|
||||||
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
|
||||||
|
|
||||||
# Prepare column and type strings required in the SQL queries
|
|
||||||
colnames = ','.join(colnames_arr)
|
|
||||||
columns_with_types_arr = [colnames_arr[i] + ' ' + coltypes_arr[i] for i in range(0,len(colnames_arr))]
|
|
||||||
columns_with_types = ','.join(columns_with_types_arr)
|
|
||||||
|
|
||||||
|
|
||||||
# Instruct the OBS server side to establish a FDW
|
|
||||||
# The metadata is obtained as well in order to:
|
|
||||||
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
|
||||||
# - (b) be able to tell OBS to free resources when done.
|
|
||||||
ds_fdw_metadata = plpy.execute("SELECT schemaname, tabname, servername "
|
|
||||||
"FROM cdb_dataservices_client._OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {table_name}::text);"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(user_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name))
|
|
||||||
)
|
|
||||||
|
|
||||||
server_schema = ds_fdw_metadata[0]["schemaname"]
|
|
||||||
server_table_name = ds_fdw_metadata[0]["tabname"]
|
|
||||||
server_name = ds_fdw_metadata[0]["servername"]
|
|
||||||
|
|
||||||
# Get list of user columns to include in the new table
|
|
||||||
user_table_columns = ','.join(
|
|
||||||
plpy.execute('SELECT array_agg(\'user_table.\' || attname) AS columns '
|
|
||||||
'FROM pg_attribute WHERE attrelid = \'"{user_schema}".{table_name}\'::regclass '
|
|
||||||
'AND attnum > 0 AND NOT attisdropped AND attname NOT LIKE \'the_geom_webmercator\' '
|
|
||||||
'AND NOT attname LIKE ANY(string_to_array(\'{colnames}\',\',\'));'
|
|
||||||
.format(user_schema=user_schema, table_name=table_name, colnames=colnames)
|
|
||||||
)[0]["columns"]
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Populate a new table with the augmented results
|
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
||||||
plpy.execute('CREATE TABLE "{user_schema}".{output_table_name} AS '
|
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
||||||
'(SELECT results.{columns}, {user_table_columns} '
|
)
|
||||||
'FROM {table_name} AS user_table '
|
|
||||||
'LEFT JOIN cdb_dataservices_client._OBS_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {function_name}::text, {params}::json) as results({columns_with_types}, cartodb_id int) '
|
|
||||||
'ON results.cartodb_id = user_table.cartodb_id)'
|
|
||||||
.format(output_table_name=output_table_name, columns=colnames, user_table_columns=user_table_columns, username=plpy.quote_nullable(username),
|
|
||||||
orgname=plpy.quote_nullable(orgname), user_schema=user_schema, server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name),
|
|
||||||
table_name=table_name, function_name=plpy.quote_literal(function_name), params=plpy.quote_literal(params), columns_with_types=columns_with_types)
|
|
||||||
)
|
|
||||||
|
|
||||||
plpy.execute('ALTER TABLE "{schema}".{table_name} OWNER TO "{user}";'
|
return True
|
||||||
.format(schema=user_schema, table_name=output_table_name, user=user_db_role)
|
|
||||||
)
|
|
||||||
|
|
||||||
# Wipe user FDW data from the server
|
|
||||||
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
|
||||||
)
|
|
||||||
|
|
||||||
return True
|
|
||||||
except Exception as e:
|
|
||||||
plpy.warning('Error trying to get table {0}'.format(e))
|
|
||||||
# Wipe user FDW data from the server in case of failure if the table was connected
|
|
||||||
if server_table_name:
|
|
||||||
wiped = plpy.execute("SELECT cdb_dataservices_client._OBS_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, {fdw_server}::text)"
|
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), server_schema=plpy.quote_literal(server_schema), server_table_name=plpy.quote_literal(server_table_name), fdw_server=plpy.quote_literal(server_name))
|
|
||||||
)
|
|
||||||
return False
|
|
||||||
$$ LANGUAGE plpythonu;
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client.__DST_PopulateTableOBS_GetMeasure(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text,
|
||||||
|
output_table_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
function_name = 'OBS_GetMeasure'
|
||||||
|
# Obtain return types for augmentation procedure
|
||||||
|
ds_return_metadata = plpy.execute(
|
||||||
|
"SELECT colnames, coltypes "
|
||||||
|
"FROM cdb_dataservices_client._DST_GetReturnMetadata({username}::text, {orgname}::text, {function_name}::text, {params}::json);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params)))
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_ConnectUserTable(username text, orgname text, user_db_role text, user_schema text, dbname text, table_name text)
|
if ds_return_metadata[0]["colnames"]:
|
||||||
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
colnames_arr = ds_return_metadata[0]["colnames"]
|
||||||
CONNECT _server_conn_str();
|
coltypes_arr = ds_return_metadata[0]["coltypes"]
|
||||||
TARGET cdb_dataservices_server._OBS_ConnectUserTable;
|
else:
|
||||||
|
raise Exception('Error retrieving OBS_GetMeasure metadata')
|
||||||
|
|
||||||
|
# Prepare column and type strings required in the SQL queries
|
||||||
|
columns_with_types_arr = [
|
||||||
|
colnames_arr[i] +
|
||||||
|
' ' +
|
||||||
|
coltypes_arr[i] for i in range(
|
||||||
|
0,
|
||||||
|
len(colnames_arr))]
|
||||||
|
columns_with_types = ','.join(columns_with_types_arr)
|
||||||
|
aliased_colname_list = ','.join(
|
||||||
|
['result.' + name for name in colnames_arr])
|
||||||
|
|
||||||
|
# Instruct the OBS server side to establish a FDW
|
||||||
|
# The metadata is obtained as well in order to:
|
||||||
|
# - (a) be able to write the query to grab the actual data to be executed in the remote server via pl/proxy,
|
||||||
|
# - (b) be able to tell OBS to free resources when done.
|
||||||
|
ds_fdw_metadata = plpy.execute(
|
||||||
|
"SELECT schemaname, tabname, servername "
|
||||||
|
"FROM cdb_dataservices_client._DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, "
|
||||||
|
"{schema}::text, {dbname}::text, {table_name}::text);" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
user_db_role=plpy.quote_literal(user_db_role),
|
||||||
|
schema=plpy.quote_literal(user_schema),
|
||||||
|
dbname=plpy.quote_literal(dbname),
|
||||||
|
table_name=plpy.quote_literal(table_name)))
|
||||||
|
|
||||||
|
if ds_fdw_metadata[0]["schemaname"]:
|
||||||
|
server_schema = ds_fdw_metadata[0]["schemaname"]
|
||||||
|
server_table_name = ds_fdw_metadata[0]["tabname"]
|
||||||
|
server_name = ds_fdw_metadata[0]["servername"]
|
||||||
|
else:
|
||||||
|
raise Exception('Error connecting dataset via FDW')
|
||||||
|
|
||||||
|
# Create a new table with the required columns
|
||||||
|
plpy.execute(
|
||||||
|
'INSERT INTO "{schema}".{analysis_table_name} '
|
||||||
|
'SELECT ut.cartodb_id, ut.the_geom, {colname_list} '
|
||||||
|
'FROM "{schema}".{table_name} ut '
|
||||||
|
'LEFT JOIN _DST_FetchJoinFdwTableData({username}::text, {orgname}::text, {server_schema}::text, {server_table_name}::text, '
|
||||||
|
'{function_name}::text, {params}::json) '
|
||||||
|
'AS result ({columns_with_types}, cartodb_id int) '
|
||||||
|
'ON result.cartodb_id = ut.cartodb_id;' .format(
|
||||||
|
schema=user_schema,
|
||||||
|
analysis_table_name=output_table_name,
|
||||||
|
colname_list=aliased_colname_list,
|
||||||
|
table_name=table_name,
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
function_name=plpy.quote_literal(function_name),
|
||||||
|
params=plpy.quote_literal(params),
|
||||||
|
columns_with_types=columns_with_types))
|
||||||
|
|
||||||
|
# Wipe user FDW data from the server
|
||||||
|
wiped = plpy.execute(
|
||||||
|
"SELECT cdb_dataservices_client._DST_DisconnectUserTable({username}::text, {orgname}::text, {server_schema}::text, "
|
||||||
|
"{server_table_name}::text, {fdw_server}::text)" .format(
|
||||||
|
username=plpy.quote_nullable(username),
|
||||||
|
orgname=plpy.quote_nullable(orgname),
|
||||||
|
server_schema=plpy.quote_literal(server_schema),
|
||||||
|
server_table_name=plpy.quote_literal(server_table_name),
|
||||||
|
fdw_server=plpy.quote_literal(server_name)))
|
||||||
|
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_ConnectUserTable(
|
||||||
|
username text,
|
||||||
|
orgname text,
|
||||||
|
user_db_role text,
|
||||||
|
user_schema text,
|
||||||
|
dbname text,
|
||||||
|
table_name text
|
||||||
|
)RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_ConnectUserTable;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_GetReturnMetadata(
|
||||||
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
username text,
|
||||||
CONNECT _server_conn_str();
|
orgname text,
|
||||||
TARGET cdb_dataservices_server._OBS_GetReturnMetadata;
|
function_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_GetReturnMetadata;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_FetchJoinFdwTableData(
|
||||||
RETURNS SETOF record AS $$
|
username text,
|
||||||
CONNECT _server_conn_str();
|
orgname text,
|
||||||
TARGET cdb_dataservices_server._OBS_FetchJoinFdwTableData;
|
table_schema text,
|
||||||
|
table_name text,
|
||||||
|
function_name text,
|
||||||
|
params json
|
||||||
|
) RETURNS SETOF record AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_FetchJoinFdwTableData;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_client._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, server_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_client._DST_DisconnectUserTable(
|
||||||
RETURNS boolean AS $$
|
username text,
|
||||||
CONNECT _server_conn_str();
|
orgname text,
|
||||||
TARGET cdb_dataservices_server._OBS_DisconnectUserTable;
|
table_schema text,
|
||||||
|
table_name text,
|
||||||
|
server_name text
|
||||||
|
) RETURNS boolean AS $$
|
||||||
|
CONNECT cdb_dataservices_client._server_conn_str();
|
||||||
|
TARGET cdb_dataservices_server._DST_DisconnectUserTable;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._obs_augmenttable(table_name text, function_name text, params json) TO publicuser;
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure(output_table_name text, params json) TO publicuser;
|
||||||
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._obs_gettable(table_name text, output_table_name text, function_name text, params json) TO publicuser;
|
GRANT EXECUTE ON FUNCTION cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure(table_name text, output_table_name text, params json) TO publicuser;
|
||||||
|
|||||||
@@ -3,65 +3,63 @@ SET search_path TO public,cartodb,cdb_dataservices_client;
|
|||||||
CREATE TABLE my_table(cartodb_id int);
|
CREATE TABLE my_table(cartodb_id int);
|
||||||
INSERT INTO my_table (cartodb_id) VALUES (1);
|
INSERT INTO my_table (cartodb_id) VALUES (1);
|
||||||
-- Mock the server functions
|
-- Mock the server functions
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
||||||
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN ('dummy_schema'::text, 'dummy_table'::text, 'dummy_server'::text);
|
RETURN ('dummy_schema'::text, 'dummy_table'::text, 'dummy_server'::text);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN (Array['total_pop'], Array['double precision']);
|
RETURN (Array['total_pop'], Array['double precision']);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
RETURNS RECORD AS $$
|
RETURNS RECORD AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN (23.4::double precision, 1::int);
|
RETURN (23.4::double precision, 1::int);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
||||||
RETURNS boolean AS $$
|
RETURNS boolean AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN true;
|
RETURN true;
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
-- Augment a table with the total_pop column
|
-- Create a sample user table
|
||||||
SELECT cdb_dataservices_client._OBS_AugmentTable('my_table', 'dummy', '{"dummy":"dummy"}'::json);
|
CREATE TABLE user_table (cartodb_id int, the_geom geometry);
|
||||||
_obs_augmenttable
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (1, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
-------------------
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (2, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (3, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
|
-- Prepare a table with the total_pop column
|
||||||
|
SELECT cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure('my_table_dst', '{"dummy":"dummy"}'::json);
|
||||||
|
_dst_preparetableobs_getmeasure
|
||||||
|
---------------------------------
|
||||||
t
|
t
|
||||||
(1 row)
|
(1 row)
|
||||||
|
|
||||||
-- The results of the table should return the mocked value of 23.4 in the total_pop column
|
-- The table should now exist and be empty
|
||||||
SELECT * FROM my_table;
|
SELECT * FROM my_table_dst;
|
||||||
cartodb_id | total_pop
|
cartodb_id | the_geom | total_pop
|
||||||
------------+-----------
|
------------+----------+-----------
|
||||||
1 | 23.4
|
(0 rows)
|
||||||
(1 row)
|
|
||||||
|
|
||||||
-- Mock again the function for it to return a different value now
|
-- Populate the table with measurement data
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
SELECT cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure('user_table', 'my_table_dst', '{"dummy":"dummy"}'::json);
|
||||||
RETURNS RECORD AS $$
|
_dst_populatetableobs_getmeasure
|
||||||
BEGIN
|
----------------------------------
|
||||||
RETURN (577777.4::double precision, 1::int);
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE 'plpgsql';
|
|
||||||
-- Augment a new table with total_pop
|
|
||||||
SELECT cdb_dataservices_client._OBS_GetTable('my_table', 'my_table_new', 'dummy', '{"dummy":"dummy"}'::json);
|
|
||||||
_obs_gettable
|
|
||||||
---------------
|
|
||||||
t
|
t
|
||||||
(1 row)
|
(1 row)
|
||||||
|
|
||||||
-- Check that the table contains the new value for total_pop and not the value already existent in the table
|
-- The table should now show the results
|
||||||
SELECT * FROM my_table_new;
|
SELECT * FROM my_table_dst;
|
||||||
total_pop | cartodb_id
|
cartodb_id | the_geom | total_pop
|
||||||
-----------+------------
|
------------+----------------------------------------------------+-----------
|
||||||
577777.4 | 1
|
1 | 0101000020E6100000F74FC902E07D52C05FE24CC7654B4440 | 23.4
|
||||||
(1 row)
|
2 | 0101000020E6100000F74FC902E07D52C05FE24CC7654B4440 |
|
||||||
|
3 | 0101000020E6100000F74FC902E07D52C05FE24CC7654B4440 |
|
||||||
|
(3 rows)
|
||||||
|
|
||||||
-- Clean tables
|
-- Clean tables
|
||||||
DROP TABLE my_table;
|
DROP TABLE my_table_dst;
|
||||||
DROP TABLE my_table_new;
|
|
||||||
|
|||||||
@@ -6,54 +6,51 @@ CREATE TABLE my_table(cartodb_id int);
|
|||||||
INSERT INTO my_table (cartodb_id) VALUES (1);
|
INSERT INTO my_table (cartodb_id) VALUES (1);
|
||||||
|
|
||||||
-- Mock the server functions
|
-- Mock the server functions
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
||||||
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
RETURNS cdb_dataservices_client.ds_fdw_metadata AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN ('dummy_schema'::text, 'dummy_table'::text, 'dummy_server'::text);
|
RETURN ('dummy_schema'::text, 'dummy_table'::text, 'dummy_server'::text);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
RETURNS cdb_dataservices_client.ds_return_metadata AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN (Array['total_pop'], Array['double precision']);
|
RETURN (Array['total_pop'], Array['double precision']);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
RETURNS RECORD AS $$
|
RETURNS RECORD AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN (23.4::double precision, 1::int);
|
RETURN (23.4::double precision, 1::int);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
||||||
RETURNS boolean AS $$
|
RETURNS boolean AS $$
|
||||||
BEGIN
|
BEGIN
|
||||||
RETURN true;
|
RETURN true;
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE 'plpgsql';
|
$$ LANGUAGE 'plpgsql';
|
||||||
|
|
||||||
-- Augment a table with the total_pop column
|
-- Create a sample user table
|
||||||
SELECT cdb_dataservices_client._OBS_AugmentTable('my_table', 'dummy', '{"dummy":"dummy"}'::json);
|
CREATE TABLE user_table (cartodb_id int, the_geom geometry);
|
||||||
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (1, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (2, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
|
INSERT INTO user_table(cartodb_id, the_geom) VALUES (3, '0101000020E6100000F74FC902E07D52C05FE24CC7654B4440');
|
||||||
|
|
||||||
-- The results of the table should return the mocked value of 23.4 in the total_pop column
|
-- Prepare a table with the total_pop column
|
||||||
SELECT * FROM my_table;
|
SELECT cdb_dataservices_client._DST_PrepareTableOBS_GetMeasure('my_table_dst', '{"dummy":"dummy"}'::json);
|
||||||
|
|
||||||
-- Mock again the function for it to return a different value now
|
-- The table should now exist and be empty
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
SELECT * FROM my_table_dst;
|
||||||
RETURNS RECORD AS $$
|
|
||||||
BEGIN
|
|
||||||
RETURN (577777.4::double precision, 1::int);
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE 'plpgsql';
|
|
||||||
|
|
||||||
-- Augment a new table with total_pop
|
-- Populate the table with measurement data
|
||||||
SELECT cdb_dataservices_client._OBS_GetTable('my_table', 'my_table_new', 'dummy', '{"dummy":"dummy"}'::json);
|
SELECT cdb_dataservices_client._DST_PopulateTableOBS_GetMeasure('user_table', 'my_table_dst', '{"dummy":"dummy"}'::json);
|
||||||
|
|
||||||
-- Check that the table contains the new value for total_pop and not the value already existent in the table
|
-- The table should now show the results
|
||||||
SELECT * FROM my_table_new;
|
SELECT * FROM my_table_dst;
|
||||||
|
|
||||||
-- Clean tables
|
-- Clean tables
|
||||||
DROP TABLE my_table;
|
DROP TABLE my_table_dst;
|
||||||
DROP TABLE my_table_new;
|
|
||||||
44
server/extension/cdb_dataservices_server--0.15.0--0.15.1.sql
Normal file
44
server/extension/cdb_dataservices_server--0.15.0--0.15.1.sql
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_server UPDATE TO '0.15.1'" to load this file. \quit
|
||||||
|
|
||||||
|
-- HERE goes your code to upgrade/downgrade
|
||||||
|
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._OBS_ConnectUserTable(text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server.__OBS_ConnectUserTable(text, text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._OBS_GetReturnMetadata(text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._OBS_FetchJoinFdwTableData(text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._OBS_DisconnectUserTable(text, text, text, text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
||||||
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
|
host_addr = plpy.execute("SELECT split_part(inet_client_addr()::text, '/', 1) as user_host")[0]['user_host']
|
||||||
|
return plpy.execute("SELECT * FROM cdb_dataservices_server.__DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {host_addr}::text, {table_name}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(input_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name), host_addr=plpy.quote_literal(host_addr))
|
||||||
|
)[0]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.__DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, host_addr text, table_name text)
|
||||||
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_ConnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
|
RETURNS cdb_dataservices_server.ds_return_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_GetReturnMetadata;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
|
RETURNS SETOF record AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_FetchJoinFdwTableData;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_DisconnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
44
server/extension/cdb_dataservices_server--0.15.1--0.15.0.sql
Normal file
44
server/extension/cdb_dataservices_server--0.15.1--0.15.0.sql
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_server UPDATE TO '0.15.0'" to load this file. \quit
|
||||||
|
|
||||||
|
-- HERE goes your code to upgrade/downgrade
|
||||||
|
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._DST_ConnectUserTable(text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server.__DST_ConnectUserTable(text, text, text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._DST_GetReturnMetadata(text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._DST_FetchJoinFdwTableData(text, text, text, text, text, json);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._DST_DisconnectUserTable(text, text, text, text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
||||||
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
|
host_addr = plpy.execute("SELECT split_part(inet_client_addr()::text, '/', 1) as user_host")[0]['user_host']
|
||||||
|
return plpy.execute("SELECT * FROM cdb_dataservices_server.__OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {host_addr}::text, {table_name}::text)"
|
||||||
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(input_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name), host_addr=plpy.quote_literal(host_addr))
|
||||||
|
)[0]
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.__OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, host_addr text, table_name text)
|
||||||
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_ConnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
|
RETURNS cdb_dataservices_server.ds_return_metadata AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_GetReturnMetadata;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
|
RETURNS SETOF record AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_FetchJoinFdwTableData;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
|
TARGET cdb_observatory._OBS_DisconnectUserTable;
|
||||||
|
$$ LANGUAGE plproxy;
|
||||||
2389
server/extension/cdb_dataservices_server--0.15.1.sql
Normal file
2389
server/extension/cdb_dataservices_server--0.15.1.sql
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,5 @@
|
|||||||
comment = 'CartoDB dataservices server extension'
|
comment = 'CartoDB dataservices server extension'
|
||||||
default_version = '0.14.2'
|
default_version = '0.15.1'
|
||||||
requires = 'plpythonu, plproxy, postgis, cdb_geocoder'
|
requires = 'plpythonu, plproxy, postgis, cdb_geocoder'
|
||||||
superuser = true
|
superuser = true
|
||||||
schema = cdb_dataservices_server
|
schema = cdb_dataservices_server
|
||||||
|
|||||||
@@ -0,0 +1,181 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_server UPDATE TO '0.15.0'" to load this file. \quit
|
||||||
|
|
||||||
|
-- HERE goes your code to upgrade/downgrade
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._get_geocoder_config(text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._get_geocoder_config(username text, orgname text, provider text DEFAULT NULL)
|
||||||
|
RETURNS boolean AS $$
|
||||||
|
cache_key = "user_geocoder_config_{0}".format(username)
|
||||||
|
if cache_key in GD:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
|
geocoder_config = GeocoderConfig(redis_conn, plpy, username, orgname, provider)
|
||||||
|
GD[cache_key] = geocoder_config
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
try:
|
||||||
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3) as point;", ["text", "text", "text"])
|
||||||
|
return plpy.execute(mapzen_plan, [username, orgname, city_name])[0]['point']
|
||||||
|
except BaseException as e:
|
||||||
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3) as point;", ["text", "text", "text"])
|
||||||
|
return plpy.execute(internal_plan, [username, orgname, city_name])[0]['point']
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
try:
|
||||||
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3, NULL, $4) as point;", ["text", "text", "text", "text"])
|
||||||
|
return plpy.execute(mapzen_plan, [username, orgname, city_name, country_name])[0]['point']
|
||||||
|
except BaseException as e:
|
||||||
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3, NULL, $4) as point;", ["text", "text", "text", "text"])
|
||||||
|
return plpy.execute(internal_plan, [username, orgname, city_name, country_name])[0]['point']
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
try:
|
||||||
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3, $4, $5) as point;", ["text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(mapzen_plan, [username, orgname, city_name, admin1_name, country_name])[0]['point']
|
||||||
|
except BaseException as e:
|
||||||
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3, $4, $5) as point;", ["text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(internal_plan, [username, orgname, city_name, admin1_name, country_name])[0]['point']
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_mapzen_geocode_namedplace(username text, orgname text, city_name text, admin1_name text DEFAULT NULL, country_name text DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.mapzen import MapzenGeocoder
|
||||||
|
from cartodb_services.mapzen.types import country_to_iso3
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_geocoder_config({0}, {1}, {2})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname), plpy.quote_nullable('mapzen')))
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
raise Exception('You have reached the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = MapzenGeocoder(user_geocoder_config.mapzen_api_key, logger)
|
||||||
|
country_iso3 = None
|
||||||
|
if country_name:
|
||||||
|
country_iso3 = country_to_iso3(country_name)
|
||||||
|
coordinates = geocoder.geocode(searchtext=city_name, city=None,
|
||||||
|
state_province=admin1_name,
|
||||||
|
country=country_iso3, search_type='locality')
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_service_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_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode city point using mapzen', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode city point using mapzen')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_internal_geocode_namedplace(username text, orgname text, city_name text, admin1_name text DEFAULT NULL, country_name text DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.metrics import InternalGeocoderConfig
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
||||||
|
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
try:
|
||||||
|
if admin1_name and country_name:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2), trim($3)) AS mypoint", ["text", "text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, admin1_name, country_name], 1)
|
||||||
|
elif country_name:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2)) AS mypoint", ["text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, country_name], 1)
|
||||||
|
else:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1)) AS mypoint", ["text"])
|
||||||
|
rv = plpy.execute(plan, [city_name], 1)
|
||||||
|
result = rv[0]["mypoint"]
|
||||||
|
if result:
|
||||||
|
quota_service.increment_success_service_use()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode namedplace point')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_mapzen_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.mapzen import MapzenGeocoder
|
||||||
|
from cartodb_services.mapzen.types import country_to_iso3
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
raise Exception('You have reached the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = MapzenGeocoder(user_geocoder_config.mapzen_api_key, logger)
|
||||||
|
country_iso3 = None
|
||||||
|
if country:
|
||||||
|
country_iso3 = country_to_iso3(country)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city,
|
||||||
|
state_province=state_province,
|
||||||
|
country=country_iso3, search_type='address')
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_service_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_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode street point using mapzen', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode street point using mapzen')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu
|
||||||
|
|
||||||
@@ -0,0 +1,169 @@
|
|||||||
|
--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 "ALTER EXTENSION cdb_dataservices_server UPDATE TO '0.14.2'" to load this file. \quit
|
||||||
|
|
||||||
|
-- HERE goes your code to upgrade/downgrade
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._get_geocoder_config(text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._cdb_mapzen_geocode_namedplace(text, text, text, text, text);
|
||||||
|
DROP FUNCTION IF EXISTS cdb_dataservices_server._cdb_internal_geocode_namedplace(text, text, text, text, text);
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_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:
|
||||||
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
|
geocoder_config = GeocoderConfig(redis_conn, plpy, username, orgname)
|
||||||
|
GD[cache_key] = geocoder_config
|
||||||
|
return True
|
||||||
|
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.metrics import InternalGeocoderConfig
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
||||||
|
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
try:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1)) AS mypoint", ["text"])
|
||||||
|
rv = plpy.execute(plan, [city_name], 1)
|
||||||
|
result = rv[0]["mypoint"]
|
||||||
|
if result:
|
||||||
|
quota_service.increment_success_service_use()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode namedplace point')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.metrics import InternalGeocoderConfig
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
||||||
|
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
try:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2)) AS mypoint", ["text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, country_name], 1)
|
||||||
|
result = rv[0]["mypoint"]
|
||||||
|
if result:
|
||||||
|
quota_service.increment_success_service_use()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode namedplace point')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.metrics import InternalGeocoderConfig
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
||||||
|
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
try:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2), trim($3)) AS mypoint", ["text", "text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, admin1_name, country_name], 1)
|
||||||
|
result = rv[0]["mypoint"]
|
||||||
|
if result:
|
||||||
|
quota_service.increment_success_service_use()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
quota_service.increment_empty_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode namedplace point')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_mapzen_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.mapzen import MapzenGeocoder
|
||||||
|
from cartodb_services.mapzen.types import country_to_iso3
|
||||||
|
from cartodb_services.metrics import QuotaService
|
||||||
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
|
logger_config = GD["logger_config"]
|
||||||
|
logger = Logger(logger_config)
|
||||||
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
raise Exception('You have reached the limit of your quota')
|
||||||
|
|
||||||
|
try:
|
||||||
|
geocoder = MapzenGeocoder(user_geocoder_config.mapzen_api_key, logger)
|
||||||
|
country_iso3 = None
|
||||||
|
if country:
|
||||||
|
country_iso3 = country_to_iso3(country)
|
||||||
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city,
|
||||||
|
state_province=state_province,
|
||||||
|
country=country_iso3)
|
||||||
|
if coordinates:
|
||||||
|
quota_service.increment_success_service_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_service_use()
|
||||||
|
return None
|
||||||
|
except BaseException as e:
|
||||||
|
import sys
|
||||||
|
quota_service.increment_failed_service_use()
|
||||||
|
logger.error('Error trying to geocode street point using mapzen', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
|
raise Exception('Error trying to geocode street point using mapzen')
|
||||||
|
finally:
|
||||||
|
quota_service.increment_total_service_use()
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
@@ -2,34 +2,34 @@ CREATE TYPE cdb_dataservices_server.ds_fdw_metadata as (schemaname text, tabname
|
|||||||
|
|
||||||
CREATE TYPE cdb_dataservices_server.ds_return_metadata as (colnames text[], coltypes text[]);
|
CREATE TYPE cdb_dataservices_server.ds_return_metadata as (colnames text[], coltypes text[]);
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, table_name text)
|
||||||
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
host_addr = plpy.execute("SELECT split_part(inet_client_addr()::text, '/', 1) as user_host")[0]['user_host']
|
host_addr = plpy.execute("SELECT split_part(inet_client_addr()::text, '/', 1) as user_host")[0]['user_host']
|
||||||
return plpy.execute("SELECT * FROM cdb_dataservices_server.__OBS_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {host_addr}::text, {table_name}::text)"
|
return plpy.execute("SELECT * FROM cdb_dataservices_server.__DST_ConnectUserTable({username}::text, {orgname}::text, {user_db_role}::text, {schema}::text, {dbname}::text, {host_addr}::text, {table_name}::text)"
|
||||||
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(input_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name), host_addr=plpy.quote_literal(host_addr))
|
.format(username=plpy.quote_nullable(username), orgname=plpy.quote_nullable(orgname), user_db_role=plpy.quote_literal(user_db_role), schema=plpy.quote_literal(input_schema), dbname=plpy.quote_literal(dbname), table_name=plpy.quote_literal(table_name), host_addr=plpy.quote_literal(host_addr))
|
||||||
)[0]
|
)[0]
|
||||||
$$ LANGUAGE plpythonu;
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server.__OBS_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, host_addr text, table_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.__DST_ConnectUserTable(username text, orgname text, user_db_role text, input_schema text, dbname text, host_addr text, table_name text)
|
||||||
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
RETURNS cdb_dataservices_server.ds_fdw_metadata AS $$
|
||||||
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
TARGET cdb_observatory._OBS_ConnectUserTable;
|
TARGET cdb_observatory._OBS_ConnectUserTable;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_GetReturnMetadata(username text, orgname text, function_name text, params json)
|
||||||
RETURNS cdb_dataservices_server.ds_return_metadata AS $$
|
RETURNS cdb_dataservices_server.ds_return_metadata AS $$
|
||||||
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
TARGET cdb_observatory._OBS_GetReturnMetadata;
|
TARGET cdb_observatory._OBS_GetReturnMetadata;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_FetchJoinFdwTableData(username text, orgname text, table_schema text, table_name text, function_name text, params json)
|
||||||
RETURNS SETOF record AS $$
|
RETURNS SETOF record AS $$
|
||||||
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
TARGET cdb_observatory._OBS_FetchJoinFdwTableData;
|
TARGET cdb_observatory._OBS_FetchJoinFdwTableData;
|
||||||
$$ LANGUAGE plproxy;
|
$$ LANGUAGE plproxy;
|
||||||
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._OBS_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._DST_DisconnectUserTable(username text, orgname text, table_schema text, table_name text, servername text)
|
||||||
RETURNS boolean AS $$
|
RETURNS boolean AS $$
|
||||||
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
CONNECT cdb_dataservices_server._obs_server_conn_str(username, orgname);
|
||||||
TARGET cdb_observatory._OBS_DisconnectUserTable;
|
TARGET cdb_observatory._OBS_DisconnectUserTable;
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ RETURNS boolean AS $$
|
|||||||
return True
|
return True
|
||||||
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server._get_geocoder_config(username text, orgname text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._get_geocoder_config(username text, orgname text, provider text DEFAULT NULL)
|
||||||
RETURNS boolean AS $$
|
RETURNS boolean AS $$
|
||||||
cache_key = "user_geocoder_config_{0}".format(username)
|
cache_key = "user_geocoder_config_{0}".format(username)
|
||||||
if cache_key in GD:
|
if cache_key in GD:
|
||||||
@@ -19,7 +19,7 @@ RETURNS boolean AS $$
|
|||||||
from cartodb_services.metrics import GeocoderConfig
|
from cartodb_services.metrics import GeocoderConfig
|
||||||
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metadata_connection']
|
||||||
geocoder_config = GeocoderConfig(redis_conn, plpy, username, orgname)
|
geocoder_config = GeocoderConfig(redis_conn, plpy, username, orgname, provider)
|
||||||
GD[cache_key] = geocoder_config
|
GD[cache_key] = geocoder_config
|
||||||
return True
|
return True
|
||||||
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
$$ LANGUAGE plpythonu SECURITY DEFINER;
|
||||||
|
|||||||
@@ -159,7 +159,7 @@ RETURNS Geometry AS $$
|
|||||||
country_iso3 = country_to_iso3(country)
|
country_iso3 = country_to_iso3(country)
|
||||||
coordinates = geocoder.geocode(searchtext=searchtext, city=city,
|
coordinates = geocoder.geocode(searchtext=searchtext, city=city,
|
||||||
state_province=state_province,
|
state_province=state_province,
|
||||||
country=country_iso3)
|
country=country_iso3, search_type='address')
|
||||||
if coordinates:
|
if coordinates:
|
||||||
quota_service.increment_success_service_use()
|
quota_service.increment_success_service_use()
|
||||||
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
plan = plpy.prepare("SELECT ST_SetSRID(ST_MakePoint($1, $2), 4326); ", ["double precision", "double precision"])
|
||||||
|
|||||||
@@ -1,76 +1,81 @@
|
|||||||
---- cdb_geocode_namedplace_point(city_name text)
|
---- cdb_geocode_namedplace_point(city_name text)
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text)
|
||||||
RETURNS Geometry AS $$
|
RETURNS Geometry AS $$
|
||||||
from cartodb_services.metrics import QuotaService
|
|
||||||
from cartodb_services.metrics import InternalGeocoderConfig
|
|
||||||
from cartodb_services.tools import Logger,LoggerConfig
|
|
||||||
|
|
||||||
|
|
||||||
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
|
||||||
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
|
||||||
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
|
||||||
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
|
||||||
|
|
||||||
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
|
||||||
logger_config = GD["logger_config"]
|
|
||||||
logger = Logger(logger_config)
|
|
||||||
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
|
||||||
try:
|
try:
|
||||||
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1)) AS mypoint", ["text"])
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3) as point;", ["text", "text", "text"])
|
||||||
rv = plpy.execute(plan, [city_name], 1)
|
return plpy.execute(mapzen_plan, [username, orgname, city_name])[0]['point']
|
||||||
result = rv[0]["mypoint"]
|
|
||||||
if result:
|
|
||||||
quota_service.increment_success_service_use()
|
|
||||||
return result
|
|
||||||
else:
|
|
||||||
quota_service.increment_empty_service_use()
|
|
||||||
return None
|
|
||||||
except BaseException as e:
|
except BaseException as e:
|
||||||
import sys
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3) as point;", ["text", "text", "text"])
|
||||||
quota_service.increment_failed_service_use()
|
return plpy.execute(internal_plan, [username, orgname, city_name])[0]['point']
|
||||||
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
|
||||||
raise Exception('Error trying to geocode namedplace point')
|
|
||||||
finally:
|
|
||||||
quota_service.increment_total_service_use()
|
|
||||||
$$ LANGUAGE plpythonu;
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
---- cdb_geocode_namedplace_point(city_name text, country_name text)
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, country_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, country_name text)
|
||||||
RETURNS Geometry AS $$
|
RETURNS Geometry AS $$
|
||||||
|
try:
|
||||||
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3, NULL, $4) as point;", ["text", "text", "text", "text"])
|
||||||
|
return plpy.execute(mapzen_plan, [username, orgname, city_name, country_name])[0]['point']
|
||||||
|
except BaseException as e:
|
||||||
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3, NULL, $4) as point;", ["text", "text", "text", "text"])
|
||||||
|
return plpy.execute(internal_plan, [username, orgname, city_name, country_name])[0]['point']
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, admin1_name text, country_name text)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
try:
|
||||||
|
mapzen_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_mapzen_geocode_namedplace($1, $2, $3, $4, $5) as point;", ["text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(mapzen_plan, [username, orgname, city_name, admin1_name, country_name])[0]['point']
|
||||||
|
except BaseException as e:
|
||||||
|
internal_plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_internal_geocode_namedplace($1, $2, $3, $4, $5) as point;", ["text", "text", "text", "text", "text"])
|
||||||
|
return plpy.execute(internal_plan, [username, orgname, city_name, admin1_name, country_name])[0]['point']
|
||||||
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_mapzen_geocode_namedplace(username text, orgname text, city_name text, admin1_name text DEFAULT NULL, country_name text DEFAULT NULL)
|
||||||
|
RETURNS Geometry AS $$
|
||||||
|
from cartodb_services.mapzen import MapzenGeocoder
|
||||||
|
from cartodb_services.mapzen.types import country_to_iso3
|
||||||
from cartodb_services.metrics import QuotaService
|
from cartodb_services.metrics import QuotaService
|
||||||
from cartodb_services.metrics import InternalGeocoderConfig
|
|
||||||
from cartodb_services.tools import Logger,LoggerConfig
|
from cartodb_services.tools import Logger,LoggerConfig
|
||||||
|
|
||||||
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
plpy.execute("SELECT cdb_dataservices_server._connect_to_redis('{0}')".format(username))
|
||||||
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
redis_conn = GD["redis_connection_{0}".format(username)]['redis_metrics_connection']
|
||||||
plpy.execute("SELECT cdb_dataservices_server._get_internal_geocoder_config({0}, {1})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname)))
|
plpy.execute("SELECT cdb_dataservices_server._get_geocoder_config({0}, {1}, {2})".format(plpy.quote_nullable(username), plpy.quote_nullable(orgname), plpy.quote_nullable('mapzen')))
|
||||||
user_geocoder_config = GD["user_internal_geocoder_config_{0}".format(username)]
|
user_geocoder_config = GD["user_geocoder_config_{0}".format(username)]
|
||||||
|
|
||||||
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
plpy.execute("SELECT cdb_dataservices_server._get_logger_config()")
|
||||||
logger_config = GD["logger_config"]
|
logger_config = GD["logger_config"]
|
||||||
logger = Logger(logger_config)
|
logger = Logger(logger_config)
|
||||||
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
|
if not quota_service.check_user_quota():
|
||||||
|
raise Exception('You have reached the limit of your quota')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2)) AS mypoint", ["text", "text"])
|
geocoder = MapzenGeocoder(user_geocoder_config.mapzen_api_key, logger)
|
||||||
rv = plpy.execute(plan, [city_name, country_name], 1)
|
country_iso3 = None
|
||||||
result = rv[0]["mypoint"]
|
if country_name:
|
||||||
if result:
|
country_iso3 = country_to_iso3(country_name)
|
||||||
|
coordinates = geocoder.geocode(searchtext=city_name, city=None,
|
||||||
|
state_province=admin1_name,
|
||||||
|
country=country_iso3, search_type='locality')
|
||||||
|
if coordinates:
|
||||||
quota_service.increment_success_service_use()
|
quota_service.increment_success_service_use()
|
||||||
return result
|
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:
|
else:
|
||||||
quota_service.increment_empty_service_use()
|
quota_service.increment_empty_service_use()
|
||||||
return None
|
return None
|
||||||
except BaseException as e:
|
except BaseException as e:
|
||||||
import sys
|
import sys
|
||||||
quota_service.increment_failed_service_use()
|
quota_service.increment_failed_service_use()
|
||||||
logger.error('Error trying to geocode namedplace point', sys.exc_info(), data={"username": username, "orgname": orgname})
|
logger.error('Error trying to geocode city point using mapzen', sys.exc_info(), data={"username": username, "orgname": orgname})
|
||||||
raise Exception('Error trying to geocode namedplace point')
|
raise Exception('Error trying to geocode city point using mapzen')
|
||||||
finally:
|
finally:
|
||||||
quota_service.increment_total_service_use()
|
quota_service.increment_total_service_use()
|
||||||
$$ LANGUAGE plpythonu;
|
$$ LANGUAGE plpythonu;
|
||||||
|
|
||||||
---- cdb_geocode_namedplace_point(city_name text, admin1_name text, country_name text)
|
CREATE OR REPLACE FUNCTION cdb_dataservices_server._cdb_internal_geocode_namedplace(username text, orgname text, city_name text, admin1_name text DEFAULT NULL, country_name text DEFAULT NULL)
|
||||||
CREATE OR REPLACE FUNCTION cdb_dataservices_server.cdb_geocode_namedplace_point(username text, orgname text, city_name text, admin1_name text, country_name text)
|
|
||||||
RETURNS Geometry AS $$
|
RETURNS Geometry AS $$
|
||||||
from cartodb_services.metrics import QuotaService
|
from cartodb_services.metrics import QuotaService
|
||||||
from cartodb_services.metrics import InternalGeocoderConfig
|
from cartodb_services.metrics import InternalGeocoderConfig
|
||||||
@@ -86,8 +91,15 @@ RETURNS Geometry AS $$
|
|||||||
logger = Logger(logger_config)
|
logger = Logger(logger_config)
|
||||||
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
quota_service = QuotaService(user_geocoder_config, redis_conn)
|
||||||
try:
|
try:
|
||||||
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2), trim($3)) AS mypoint", ["text", "text", "text"])
|
if admin1_name and country_name:
|
||||||
rv = plpy.execute(plan, [city_name, admin1_name, country_name], 1)
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2), trim($3)) AS mypoint", ["text", "text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, admin1_name, country_name], 1)
|
||||||
|
elif country_name:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1), trim($2)) AS mypoint", ["text", "text"])
|
||||||
|
rv = plpy.execute(plan, [city_name, country_name], 1)
|
||||||
|
else:
|
||||||
|
plan = plpy.prepare("SELECT cdb_dataservices_server._cdb_geocode_namedplace_point(trim($1)) AS mypoint", ["text"])
|
||||||
|
rv = plpy.execute(plan, [city_name], 1)
|
||||||
result = rv[0]["mypoint"]
|
result = rv[0]["mypoint"]
|
||||||
if result:
|
if result:
|
||||||
quota_service.increment_success_service_use()
|
quota_service.increment_success_service_use()
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ CREATE EXTENSION plpythonu;
|
|||||||
CREATE EXTENSION plproxy;
|
CREATE EXTENSION plproxy;
|
||||||
CREATE EXTENSION cartodb;
|
CREATE EXTENSION cartodb;
|
||||||
CREATE EXTENSION cdb_geocoder;
|
CREATE EXTENSION cdb_geocoder;
|
||||||
CREATE EXTENSION observatory VERSION 'dev';
|
|
||||||
-- Install the extension
|
-- Install the extension
|
||||||
CREATE EXTENSION cdb_dataservices_server;
|
CREATE EXTENSION cdb_dataservices_server;
|
||||||
-- Mock the redis server connection to point to this very test db
|
-- Mock the redis server connection to point to this very test db
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ SELECT exists(SELECT *
|
|||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_connectusertable'
|
AND proname = '_dst_connectusertable'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
||||||
exists
|
exists
|
||||||
--------
|
--------
|
||||||
@@ -13,7 +13,7 @@ SELECT exists(SELECT *
|
|||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_getreturnmetadata'
|
AND proname = '_dst_getreturnmetadata'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, json');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, json');
|
||||||
exists
|
exists
|
||||||
--------
|
--------
|
||||||
@@ -24,7 +24,7 @@ SELECT exists(SELECT *
|
|||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_fetchjoinfdwtabledata'
|
AND proname = '_dst_fetchjoinfdwtabledata'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, json');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, json');
|
||||||
exists
|
exists
|
||||||
--------
|
--------
|
||||||
@@ -35,7 +35,7 @@ SELECT exists(SELECT *
|
|||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_disconnectusertable'
|
AND proname = '_dst_disconnectusertable'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text');
|
||||||
exists
|
exists
|
||||||
--------
|
--------
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ INSERT INTO global_cities_alternates_limited (geoname_id, name, preferred, lower
|
|||||||
'POINT(0.6983 39.26787)',4326)
|
'POINT(0.6983 39.26787)',4326)
|
||||||
);
|
);
|
||||||
-- Insert dummy data into country decoder table
|
-- Insert dummy data into country decoder table
|
||||||
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain'], 'ES');
|
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain', 'Spain'], 'ES');
|
||||||
-- Insert dummy data into admin1 decoder table
|
-- Insert dummy data into admin1 decoder table
|
||||||
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
||||||
-- This should return the point inserted above
|
-- This should return the point inserted above
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ CREATE EXTENSION plpythonu;
|
|||||||
CREATE EXTENSION plproxy;
|
CREATE EXTENSION plproxy;
|
||||||
CREATE EXTENSION cartodb;
|
CREATE EXTENSION cartodb;
|
||||||
CREATE EXTENSION cdb_geocoder;
|
CREATE EXTENSION cdb_geocoder;
|
||||||
CREATE EXTENSION observatory VERSION 'dev';
|
|
||||||
|
|
||||||
-- Install the extension
|
-- Install the extension
|
||||||
CREATE EXTENSION cdb_dataservices_server;
|
CREATE EXTENSION cdb_dataservices_server;
|
||||||
|
|||||||
@@ -2,27 +2,27 @@ SELECT exists(SELECT *
|
|||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_connectusertable'
|
AND proname = '_dst_connectusertable'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, text');
|
||||||
|
|
||||||
SELECT exists(SELECT *
|
SELECT exists(SELECT *
|
||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_getreturnmetadata'
|
AND proname = '_dst_getreturnmetadata'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, json');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, json');
|
||||||
|
|
||||||
SELECT exists(SELECT *
|
SELECT exists(SELECT *
|
||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_fetchjoinfdwtabledata'
|
AND proname = '_dst_fetchjoinfdwtabledata'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, json');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text, json');
|
||||||
|
|
||||||
SELECT exists(SELECT *
|
SELECT exists(SELECT *
|
||||||
FROM pg_proc p
|
FROM pg_proc p
|
||||||
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
INNER JOIN pg_namespace ns ON (p.pronamespace = ns.oid)
|
||||||
WHERE ns.nspname = 'cdb_dataservices_server'
|
WHERE ns.nspname = 'cdb_dataservices_server'
|
||||||
AND proname = '_obs_disconnectusertable'
|
AND proname = '_dst_disconnectusertable'
|
||||||
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text');
|
AND oidvectortypes(p.proargtypes) = 'text, text, text, text, text');
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ INSERT INTO global_cities_alternates_limited (geoname_id, name, preferred, lower
|
|||||||
);
|
);
|
||||||
|
|
||||||
-- Insert dummy data into country decoder table
|
-- Insert dummy data into country decoder table
|
||||||
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain'], 'ES');
|
INSERT INTO country_decoder (synonyms, iso2) VALUES (Array['spain', 'Spain'], 'ES');
|
||||||
|
|
||||||
-- Insert dummy data into admin1 decoder table
|
-- Insert dummy data into admin1 decoder table
|
||||||
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
INSERT INTO admin1_decoder (admin1, synonyms, iso2) VALUES ('Valencia', Array['valencia', 'Valencia'], 'ES');
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# CartoDB dataservices API python module
|
# CARTO dataservices API python module
|
||||||
|
|
||||||
This directory contains the python library used by the server side of CARTO LDS (Location Data Services).
|
This directory contains the python library used by the server side of CARTO LDS (Location Data Services).
|
||||||
|
|
||||||
|
|||||||
@@ -19,3 +19,15 @@ class MalformedResult(Exception):
|
|||||||
class TimeoutException(Exception):
|
class TimeoutException(Exception):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return repr('Timeout requesting to mapzen server')
|
return repr('Timeout requesting to mapzen server')
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceException(Exception):
|
||||||
|
def __init__(self, message, response):
|
||||||
|
self.message = message
|
||||||
|
self.response = response
|
||||||
|
|
||||||
|
def response(self):
|
||||||
|
return self.response
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.message
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import requests
|
|||||||
import json
|
import json
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from exceptions import WrongParams, MalformedResult
|
from exceptions import WrongParams, MalformedResult, ServiceException
|
||||||
from qps import qps_retry
|
from qps import qps_retry
|
||||||
from cartodb_services.tools import Coordinate, PolyLine
|
from cartodb_services.tools import Coordinate, PolyLine
|
||||||
|
|
||||||
@@ -17,11 +17,12 @@ class MapzenGeocoder:
|
|||||||
self._url = base_url
|
self._url = base_url
|
||||||
self._logger = logger
|
self._logger = logger
|
||||||
|
|
||||||
@qps_retry
|
@qps_retry(qps=20)
|
||||||
def geocode(self, searchtext, city=None, state_province=None, country=None):
|
def geocode(self, searchtext, city=None, state_province=None,
|
||||||
|
country=None, search_type=None):
|
||||||
request_params = self._build_requests_parameters(searchtext, city,
|
request_params = self._build_requests_parameters(searchtext, city,
|
||||||
state_province,
|
state_province,
|
||||||
country)
|
country, search_type)
|
||||||
try:
|
try:
|
||||||
response = requests.get(self._url, params=request_params)
|
response = requests.get(self._url, params=request_params)
|
||||||
if response.status_code == requests.codes.ok:
|
if response.status_code == requests.codes.ok:
|
||||||
@@ -31,30 +32,32 @@ class MapzenGeocoder:
|
|||||||
else:
|
else:
|
||||||
self._logger.error('Error trying to geocode using mapzen',
|
self._logger.error('Error trying to geocode using mapzen',
|
||||||
data={"response_status": response.status_code,
|
data={"response_status": response.status_code,
|
||||||
"response_reason": response.reason,
|
"response_reason": response.reason,
|
||||||
"response_content": response.text,
|
"response_content": response.text,
|
||||||
"reponse_url": response.url,
|
"reponse_url": response.url,
|
||||||
"response_headers": response.headers,
|
"response_headers": response.headers,
|
||||||
"searchtext": searchtext,
|
"searchtext": searchtext,
|
||||||
"city": city, "country": country,
|
"city": city, "country": country,
|
||||||
"state_province": state_province })
|
"state_province": state_province})
|
||||||
raise Exception('Error trying to geocode {0} using mapzen'.format(searchtext))
|
raise ServiceException('Error trying to geocode {0} using mapzen'.format(searchtext),
|
||||||
|
response)
|
||||||
except requests.ConnectionError as e:
|
except requests.ConnectionError as e:
|
||||||
# Don't raise the exception to continue with the geocoding job
|
# Don't raise the exception to continue with the geocoding job
|
||||||
self._logger.error('Error connecting to Mapzen geocoding server',
|
self._logger.error('Error connecting to Mapzen geocoding server',
|
||||||
exception=e)
|
exception=e)
|
||||||
return []
|
return []
|
||||||
|
|
||||||
|
|
||||||
def _build_requests_parameters(self, searchtext, city=None,
|
def _build_requests_parameters(self, searchtext, city=None,
|
||||||
state_province=None, country=None):
|
state_province=None, country=None,
|
||||||
|
search_type=None):
|
||||||
request_params = {}
|
request_params = {}
|
||||||
search_string = self._build_search_text(searchtext.strip(),
|
search_string = self._build_search_text(searchtext.strip(),
|
||||||
city,
|
city,
|
||||||
state_province)
|
state_province)
|
||||||
request_params['text'] = search_string
|
request_params['text'] = search_string
|
||||||
request_params['layers'] = 'address'
|
|
||||||
request_params['api_key'] = self._app_key
|
request_params['api_key'] = self._app_key
|
||||||
|
if search_type:
|
||||||
|
request_params['layers'] = search_type
|
||||||
if country:
|
if country:
|
||||||
request_params['boundary.country'] = country
|
request_params['boundary.country'] = country
|
||||||
return request_params
|
return request_params
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import requests
|
import requests
|
||||||
import json
|
import json
|
||||||
from qps import qps_retry
|
from qps import qps_retry
|
||||||
|
from exceptions import ServiceException
|
||||||
|
|
||||||
|
|
||||||
class MatrixClient:
|
class MatrixClient:
|
||||||
@@ -51,6 +52,6 @@ class MatrixClient:
|
|||||||
"response_headers": response.headers,
|
"response_headers": response.headers,
|
||||||
"locations": locations,
|
"locations": locations,
|
||||||
"costing": costing})
|
"costing": costing})
|
||||||
raise Exception('Error trying to get matrix distance from mapzen')
|
raise ServiceException("Error trying to get matrix distance from mapzen", response)
|
||||||
|
|
||||||
return response.json()
|
return response.json()
|
||||||
|
|||||||
@@ -4,18 +4,38 @@ from datetime import datetime
|
|||||||
from exceptions import TimeoutException
|
from exceptions import TimeoutException
|
||||||
|
|
||||||
DEFAULT_RETRY_TIMEOUT = 60
|
DEFAULT_RETRY_TIMEOUT = 60
|
||||||
|
DEFAULT_QUERIES_PER_SECOND = 10
|
||||||
|
|
||||||
|
def qps_retry(original_function=None,**options):
|
||||||
def qps_retry(f):
|
""" Query Per Second retry decorator
|
||||||
def wrapped_f(*args, **kw):
|
The intention of this decorator is to retry requests against third
|
||||||
return QPSService().call(f, *args, **kw)
|
party services that has QPS restriction.
|
||||||
return wrapped_f
|
Parameters:
|
||||||
|
- timeout: Maximum number of seconds to retry
|
||||||
|
- qps: Allowed queries per second. This parameter is used to
|
||||||
|
calculate the next time to retry the request
|
||||||
|
"""
|
||||||
|
if original_function is not None:
|
||||||
|
def wrapped_function(*args, **kwargs):
|
||||||
|
if 'timeout' in options:
|
||||||
|
timeout = options['timeout']
|
||||||
|
else:
|
||||||
|
timeout = DEFAULT_RETRY_TIMEOUT
|
||||||
|
if 'qps' in options:
|
||||||
|
qps = options['qps']
|
||||||
|
else:
|
||||||
|
qps = DEFAULT_QUERIES_PER_SECOND
|
||||||
|
return QPSService(retry_timeout=timeout, queries_per_second=qps).call(original_function, *args, **kwargs)
|
||||||
|
return wrapped_function
|
||||||
|
else:
|
||||||
|
def partial_wrapper(func):
|
||||||
|
return qps_retry(func, **options)
|
||||||
|
return partial_wrapper
|
||||||
|
|
||||||
|
|
||||||
class QPSService:
|
class QPSService:
|
||||||
|
|
||||||
def __init__(self, queries_per_second=10,
|
def __init__(self, queries_per_second, retry_timeout):
|
||||||
retry_timeout=DEFAULT_RETRY_TIMEOUT):
|
|
||||||
self._queries_per_second = queries_per_second
|
self._queries_per_second = queries_per_second
|
||||||
self._retry_timeout = retry_timeout
|
self._retry_timeout = retry_timeout
|
||||||
|
|
||||||
@@ -27,7 +47,7 @@ class QPSService:
|
|||||||
return fn(*args, **kwargs)
|
return fn(*args, **kwargs)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
response = getattr(e, 'response', None)
|
response = getattr(e, 'response', None)
|
||||||
if response and (response.status_code == 429):
|
if response is not None and (response.status_code == 429):
|
||||||
self.retry(start_time, attempt_number)
|
self.retry(start_time, attempt_number)
|
||||||
else:
|
else:
|
||||||
raise e
|
raise e
|
||||||
@@ -35,7 +55,7 @@ class QPSService:
|
|||||||
|
|
||||||
def retry(self, first_request_time, retry_count):
|
def retry(self, first_request_time, retry_count):
|
||||||
elapsed = datetime.now() - first_request_time
|
elapsed = datetime.now() - first_request_time
|
||||||
if elapsed.seconds > self._retry_timeout:
|
if elapsed.microseconds > (self._retry_timeout * 1000.0):
|
||||||
raise TimeoutException()
|
raise TimeoutException()
|
||||||
|
|
||||||
# inverse qps * (1.5 ^ i) is an increased sleep time of 1.5x per
|
# inverse qps * (1.5 ^ i) is an increased sleep time of 1.5x per
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import requests
|
|||||||
import json
|
import json
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from exceptions import WrongParams, MalformedResult
|
from exceptions import WrongParams, MalformedResult, ServiceException
|
||||||
from qps import qps_retry
|
from qps import qps_retry
|
||||||
from cartodb_services.tools import Coordinate, PolyLine
|
from cartodb_services.tools import Coordinate, PolyLine
|
||||||
|
|
||||||
@@ -57,7 +57,7 @@ class MapzenRouting:
|
|||||||
"response_headers": response.headers,
|
"response_headers": response.headers,
|
||||||
"waypoints": waypoints, "mode": mode,
|
"waypoints": waypoints, "mode": mode,
|
||||||
"options": options})
|
"options": options})
|
||||||
raise Exception('Error trying to calculate route using Mapzen')
|
raise ServiceException('Error trying to calculate route using Mapzen', response)
|
||||||
|
|
||||||
def __parse_options(self, options):
|
def __parse_options(self, options):
|
||||||
return dict(option.split('=') for option in options)
|
return dict(option.split('=') for option in options)
|
||||||
|
|||||||
@@ -286,11 +286,11 @@ class GeocoderConfig(ServiceConfig):
|
|||||||
PERIOD_END_DATE = 'period_end_date'
|
PERIOD_END_DATE = 'period_end_date'
|
||||||
DEFAULT_PROVIDER = 'mapzen'
|
DEFAULT_PROVIDER = 'mapzen'
|
||||||
|
|
||||||
def __init__(self, redis_connection, db_conn, username, orgname=None):
|
def __init__(self, redis_connection, db_conn, username, orgname=None, forced_provider=None):
|
||||||
super(GeocoderConfig, self).__init__(redis_connection, db_conn,
|
super(GeocoderConfig, self).__init__(redis_connection, db_conn,
|
||||||
username, orgname)
|
username, orgname)
|
||||||
filtered_config = {key: self._redis_config[key] for key in self.GEOCODER_CONFIG_KEYS if key in self._redis_config.keys()}
|
filtered_config = {key: self._redis_config[key] for key in self.GEOCODER_CONFIG_KEYS if key in self._redis_config.keys()}
|
||||||
self.__parse_config(filtered_config, self._db_config)
|
self.__parse_config(filtered_config, self._db_config, forced_provider)
|
||||||
self.__check_config(filtered_config)
|
self.__check_config(filtered_config)
|
||||||
|
|
||||||
def __check_config(self, filtered_config):
|
def __check_config(self, filtered_config):
|
||||||
@@ -307,9 +307,12 @@ class GeocoderConfig(ServiceConfig):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def __parse_config(self, filtered_config, db_config):
|
def __parse_config(self, filtered_config, db_config, forced_provider):
|
||||||
self._geocoder_provider = filtered_config[self.GEOCODER_PROVIDER].lower()
|
if forced_provider:
|
||||||
if not self._geocoder_provider:
|
self._geocoder_provider = forced_provider
|
||||||
|
elif filtered_config[self.GEOCODER_PROVIDER].lower():
|
||||||
|
self._geocoder_provider = filtered_config[self.GEOCODER_PROVIDER].lower()
|
||||||
|
else:
|
||||||
self._geocoder_provider = self.DEFAULT_PROVIDER
|
self._geocoder_provider = self.DEFAULT_PROVIDER
|
||||||
self._geocoding_quota = float(filtered_config[self.QUOTA_KEY])
|
self._geocoding_quota = float(filtered_config[self.QUOTA_KEY])
|
||||||
self._period_end_date = date_parse(filtered_config[self.PERIOD_END_DATE])
|
self._period_end_date = date_parse(filtered_config[self.PERIOD_END_DATE])
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import plpy
|
|
||||||
import rollbar
|
import rollbar
|
||||||
import logging
|
import logging
|
||||||
import json
|
import json
|
||||||
@@ -6,7 +5,14 @@ import traceback
|
|||||||
import sys
|
import sys
|
||||||
# Monkey patch because plpython sys module doesn't have argv and rollbar
|
# Monkey patch because plpython sys module doesn't have argv and rollbar
|
||||||
# package use it
|
# package use it
|
||||||
sys.__dict__['argv'] = []
|
if 'argv' not in sys.__dict__:
|
||||||
|
sys.__dict__['argv'] = []
|
||||||
|
|
||||||
|
# Only can be imported when is called from PLPython
|
||||||
|
try:
|
||||||
|
import plpy
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class Logger:
|
class Logger:
|
||||||
@@ -30,30 +36,28 @@ class Logger:
|
|||||||
return
|
return
|
||||||
self._send_to_rollbar('debug', text, exception, data)
|
self._send_to_rollbar('debug', text, exception, data)
|
||||||
self._send_to_log_file('debug', text, exception, data)
|
self._send_to_log_file('debug', text, exception, data)
|
||||||
plpy.debug(text)
|
self._send_to_plpy('debug', text)
|
||||||
|
|
||||||
def info(self, text, exception=None, data={}):
|
def info(self, text, exception=None, data={}):
|
||||||
if not self._check_min_level('info'):
|
if not self._check_min_level('info'):
|
||||||
return
|
return
|
||||||
self._send_to_rollbar('info', text, exception, data)
|
self._send_to_rollbar('info', text, exception, data)
|
||||||
self._send_to_log_file('info', text, exception, data)
|
self._send_to_log_file('info', text, exception, data)
|
||||||
plpy.info(text)
|
self._send_to_plpy('info', text)
|
||||||
|
|
||||||
def warning(self, text, exception=None, data={}):
|
def warning(self, text, exception=None, data={}):
|
||||||
if not self._check_min_level('warning'):
|
if not self._check_min_level('warning'):
|
||||||
return
|
return
|
||||||
self._send_to_rollbar('warning', text, exception, data)
|
self._send_to_rollbar('warning', text, exception, data)
|
||||||
self._send_to_log_file('warning', text, exception, data)
|
self._send_to_log_file('warning', text, exception, data)
|
||||||
plpy.warning(text)
|
self._send_to_plpy('warning', text)
|
||||||
|
|
||||||
def error(self, text, exception=None, data={}):
|
def error(self, text, exception=None, data={}):
|
||||||
if not self._check_min_level('error'):
|
if not self._check_min_level('error'):
|
||||||
return
|
return
|
||||||
self._send_to_rollbar('error', text, exception, data)
|
self._send_to_rollbar('error', text, exception, data)
|
||||||
self._send_to_log_file('error', text, exception, data)
|
self._send_to_log_file('error', text, exception, data)
|
||||||
# Plpy.error and fatal raises exceptions and we only want to log an
|
self._send_to_plpy('error', text)
|
||||||
# error, exceptions should be raise explicitly
|
|
||||||
plpy.warning(text)
|
|
||||||
|
|
||||||
def _check_min_level(self, level):
|
def _check_min_level(self, level):
|
||||||
return True if self.LEVELS[level] >= self._min_level else False
|
return True if self.LEVELS[level] >= self._min_level else False
|
||||||
@@ -82,6 +86,19 @@ class Logger:
|
|||||||
elif level == 'error':
|
elif level == 'error':
|
||||||
self._file_logger.error(text, extra=extra_data)
|
self._file_logger.error(text, extra=extra_data)
|
||||||
|
|
||||||
|
def _send_to_plpy(self, level, text):
|
||||||
|
if self._check_plpy():
|
||||||
|
if level == 'debug':
|
||||||
|
plpy.debug(text)
|
||||||
|
elif level == 'info':
|
||||||
|
plpy.info(text)
|
||||||
|
elif level == 'warning':
|
||||||
|
plpy.warning(text)
|
||||||
|
elif level == 'error':
|
||||||
|
# Plpy.error and fatal raises exceptions and we only want to
|
||||||
|
# log an error, exceptions should be raise explicitly
|
||||||
|
plpy.warning(text)
|
||||||
|
|
||||||
def _parse_log_extra_data(self, exception, data):
|
def _parse_log_extra_data(self, exception, data):
|
||||||
extra_data = {}
|
extra_data = {}
|
||||||
if exception:
|
if exception:
|
||||||
@@ -118,6 +135,13 @@ class Logger:
|
|||||||
def _log_file_activated(self):
|
def _log_file_activated(self):
|
||||||
return True if self._config.log_file_path else False
|
return True if self._config.log_file_path else False
|
||||||
|
|
||||||
|
def _check_plpy(self):
|
||||||
|
try:
|
||||||
|
module = sys.modules['plpy']
|
||||||
|
return True
|
||||||
|
except KeyError:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
class ConfigException(Exception):
|
class ConfigException(Exception):
|
||||||
pass
|
pass
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from setuptools import setup, find_packages
|
|||||||
setup(
|
setup(
|
||||||
name='cartodb_services',
|
name='cartodb_services',
|
||||||
|
|
||||||
version='0.7.4.2',
|
version='0.8.1',
|
||||||
|
|
||||||
description='CartoDB Services API Python Library',
|
description='CartoDB Services API Python Library',
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import requests_mock
|
|||||||
from mock import Mock
|
from mock import Mock
|
||||||
|
|
||||||
from cartodb_services.mapzen import MapzenGeocoder
|
from cartodb_services.mapzen import MapzenGeocoder
|
||||||
from cartodb_services.mapzen.exceptions import MalformedResult
|
from cartodb_services.mapzen.exceptions import MalformedResult, TimeoutException
|
||||||
|
|
||||||
requests_mock.Mocker.TEST_PREFIX = 'test_'
|
requests_mock.Mocker.TEST_PREFIX = 'test_'
|
||||||
|
|
||||||
|
|||||||
33
server/lib/python/cartodb_services/test/test_qps.py
Normal file
33
server/lib/python/cartodb_services/test/test_qps.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
import test_helper
|
||||||
|
import requests
|
||||||
|
from unittest import TestCase
|
||||||
|
from nose.tools import assert_raises
|
||||||
|
from datetime import datetime, date
|
||||||
|
from cartodb_services.mapzen.qps import qps_retry
|
||||||
|
from cartodb_services.mapzen.exceptions import ServiceException, TimeoutException
|
||||||
|
import requests_mock
|
||||||
|
import mock
|
||||||
|
|
||||||
|
requests_mock.Mocker.TEST_PREFIX = 'test_'
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
class TestQPS(TestCase):
|
||||||
|
QPS_ERROR_MESSAGE = "Queries per second exceeded: Queries exceeded (10 allowed)"
|
||||||
|
|
||||||
|
def test_qps_timeout(self, req_mock):
|
||||||
|
class TestClass:
|
||||||
|
@qps_retry(timeout=0.001, qps=100)
|
||||||
|
def test(self):
|
||||||
|
response = requests.get('http://localhost/test_qps')
|
||||||
|
if response.status_code == 429:
|
||||||
|
raise ServiceException('Error 429', response)
|
||||||
|
|
||||||
|
def _text_cb(request, context):
|
||||||
|
context.status_code = 429
|
||||||
|
return self.QPS_ERROR_MESSAGE
|
||||||
|
|
||||||
|
req_mock.register_uri('GET', 'http://localhost/test_qps',
|
||||||
|
text=_text_cb)
|
||||||
|
with self.assertRaises(TimeoutException):
|
||||||
|
c = TestClass()
|
||||||
|
c.test()
|
||||||
Reference in New Issue
Block a user