103 lines
3.4 KiB
PL/PgSQL
103 lines
3.4 KiB
PL/PgSQL
-- Create user and enable Ghost tables trigger
|
|
\set QUIET on
|
|
SET client_min_messages TO error;
|
|
|
|
-- Recreate the function without extra error messages as it changes depending on the python-redis version
|
|
CREATE OR REPLACE FUNCTION cartodb._CDB_LinkGhostTables(username text, db_name text, event_name text)
|
|
RETURNS void
|
|
AS $$
|
|
if not username:
|
|
return
|
|
|
|
if 'json' not in GD:
|
|
import json
|
|
GD['json'] = json
|
|
else:
|
|
json = GD['json']
|
|
|
|
tis_config = plpy.execute("select cartodb.CDB_Conf_GetConf('invalidation_service');")[0]['cdb_conf_getconf']
|
|
if not tis_config:
|
|
plpy.warning('Invalidation service configuration not found. Skipping Ghost Tables linking.')
|
|
return
|
|
|
|
tis_config_dict = json.loads(tis_config)
|
|
tis_host = tis_config_dict.get('host')
|
|
tis_port = tis_config_dict.get('port')
|
|
tis_timeout = tis_config_dict.get('timeout', 5)
|
|
tis_retry = tis_config_dict.get('retry', 5)
|
|
|
|
client = GD.get('invalidation', None)
|
|
|
|
while True:
|
|
|
|
if not client:
|
|
try:
|
|
import redis
|
|
client = redis.Redis(host=tis_host, port=tis_port, socket_timeout=tis_timeout)
|
|
GD['invalidation'] = client
|
|
except Exception as err:
|
|
# NOTE: no retries on connection error
|
|
plpy.warning('Error trying to connect to Invalidation Service to link Ghost Tables')
|
|
break
|
|
|
|
try:
|
|
client.execute_command('DBSCH', db_name, username, event_name)
|
|
break
|
|
except Exception as err:
|
|
client = GD['invalidation'] = None # force reconnect
|
|
if not tis_retry:
|
|
plpy.warning('Error calling Invalidation Service to link Ghost Tables')
|
|
break
|
|
tis_retry -= 1 # try reconnecting
|
|
$$ LANGUAGE '@@plpythonu@@' VOLATILE PARALLEL UNSAFE;
|
|
|
|
SELECT CDB_EnableGhostTablesTrigger();
|
|
CREATE ROLE "fulano" LOGIN;
|
|
GRANT ALL ON SCHEMA cartodb TO "fulano";
|
|
GRANT SELECT ON cartodb.cdb_ddl_execution TO "fulano";
|
|
GRANT EXECUTE ON FUNCTION CDB_Username() TO "fulano";
|
|
GRANT EXECUTE ON FUNCTION CDB_LinkGhostTables(text) TO "fulano";
|
|
SELECT cartodb.CDB_Conf_SetConf('api_keys_fulano', '{"username": "fulanito", "permissions":[]}');
|
|
DELETE FROM cdb_conf WHERE key = 'invalidation_service';
|
|
SET SESSION AUTHORIZATION "fulano";
|
|
SET client_min_messages TO notice;
|
|
\set QUIET off
|
|
|
|
SELECT CDB_LinkGhostTables(); -- _CDB_LinkGhostTables called (configuration not found)
|
|
|
|
-- Add TIS configuration
|
|
\set QUIET on
|
|
SET SESSION AUTHORIZATION postgres;
|
|
SELECT cartodb.CDB_Conf_SetConf('invalidation_service', '{"host": "fake-tis-host", "port": 3142}');
|
|
SET SESSION AUTHORIZATION "fulano";
|
|
\set QUIET off
|
|
|
|
SELECT CDB_LinkGhostTables(); -- _CDB_LinkGhostTables called
|
|
|
|
BEGIN;
|
|
SELECT to_regclass('cartodb.cdb_ddl_execution'); -- exists
|
|
SELECT COUNT(*) FROM cartodb.cdb_ddl_execution; -- 0
|
|
CREATE TABLE tmp(id INT);
|
|
SELECT COUNT(*) FROM cartodb.cdb_ddl_execution; -- 1
|
|
END; -- _CDB_LinkGhostTables called
|
|
|
|
-- Disable Ghost tables trigger
|
|
\set QUIET on
|
|
SET SESSION AUTHORIZATION postgres;
|
|
SELECT CDB_DisableGhostTablesTrigger();
|
|
SET SESSION AUTHORIZATION "fulano";
|
|
\set QUIET off
|
|
|
|
SELECT to_regclass('cartodb.cdb_ddl_execution'); -- not exists
|
|
DROP TABLE tmp; -- _CDB_LinkGhostTables not called
|
|
|
|
-- Cleanup
|
|
\set QUIET on
|
|
SET SESSION AUTHORIZATION postgres;
|
|
REVOKE EXECUTE ON FUNCTION CDB_LinkGhostTables(text) FROM "fulano";
|
|
REVOKE EXECUTE ON FUNCTION CDB_Username() FROM "fulano";
|
|
REVOKE ALL ON SCHEMA cartodb FROM "fulano";
|
|
DROP ROLE "fulano";
|
|
DELETE FROM cdb_conf WHERE key = 'api_keys_fulano' OR key = 'invalidation_service';
|
|
\set QUIET off
|