2019-03-07 01:19:38 +08:00
|
|
|
-- Table to store the transaction id from DDL events to avoid multiple executions
|
2019-03-08 15:28:06 +08:00
|
|
|
CREATE TABLE IF NOT EXISTS cartodb.cdb_ddl_execution(txid integer PRIMARY KEY, tag text);
|
2019-03-07 01:19:38 +08:00
|
|
|
|
|
|
|
-- Enqueues a job to run Ghost tables linking process for the provided user_id
|
2019-03-08 15:28:06 +08:00
|
|
|
CREATE OR REPLACE FUNCTION _CDB_LinkGhostTables(username text, db_name text, ddl_tag text)
|
2019-03-07 01:19:38 +08:00
|
|
|
RETURNS void
|
|
|
|
AS $$
|
2019-03-08 15:28:06 +08:00
|
|
|
if not username:
|
2019-03-07 01:19:38 +08:00
|
|
|
return
|
|
|
|
|
2019-03-08 19:03:55 +08:00
|
|
|
if 'json' not in GD:
|
|
|
|
import json
|
|
|
|
GD['json'] = json
|
|
|
|
else:
|
|
|
|
json = GD['json']
|
2019-03-07 01:19:38 +08:00
|
|
|
|
|
|
|
error = ''
|
2019-03-08 19:03:55 +08:00
|
|
|
tis_config = plpy.execute("select cartodb.CDB_Conf_GetConf('invalidation_service');")[0]['cdb_conf_getconf']
|
|
|
|
tis_config_dict = json.loads(tis_config) if tis_config else {}
|
|
|
|
tis_host = tis_config_dict.get('host', '127.0.0.1')
|
|
|
|
tis_port = tis_config_dict.get('port', 3142)
|
|
|
|
tis_timeout = tis_config_dict.get('timeout', 5)
|
|
|
|
tis_retry = tis_config_dict.get('retry', 5)
|
|
|
|
|
|
|
|
client = GD.get('invalidation', None)
|
2019-03-07 01:19:38 +08:00
|
|
|
|
|
|
|
while True:
|
|
|
|
|
|
|
|
if not client:
|
|
|
|
try:
|
|
|
|
import redis
|
2019-03-08 19:03:55 +08:00
|
|
|
client = GD['invalidation'] = redis.Redis(host=tis_host, port=tis_port, socket_timeout=tis_timeout)
|
2019-03-07 01:19:38 +08:00
|
|
|
except Exception as err:
|
|
|
|
error = "client_error - %s" % str(err)
|
|
|
|
# NOTE: no retries on connection error
|
2019-03-08 19:03:55 +08:00
|
|
|
plpy.warning('Invalidation Service connection error: ' + str(err))
|
2019-03-07 01:19:38 +08:00
|
|
|
break
|
|
|
|
|
|
|
|
try:
|
2019-03-08 19:03:55 +08:00
|
|
|
client.execute_command('DBSCH', db_name, username, ddl_tag)
|
2019-03-07 01:19:38 +08:00
|
|
|
break
|
|
|
|
except Exception as err:
|
|
|
|
error = "request_error - %s" % str(err)
|
2019-03-08 19:03:55 +08:00
|
|
|
client = GD['invalidation'] = None # force reconnect
|
|
|
|
if not tis_retry:
|
|
|
|
plpy.warning('Invalidation Service error: ' + str(err))
|
2019-03-07 01:19:38 +08:00
|
|
|
break
|
2019-03-08 19:03:55 +08:00
|
|
|
tis_retry -= 1 # try reconnecting
|
2019-03-07 01:19:38 +08:00
|
|
|
$$ LANGUAGE 'plpythonu' VOLATILE PARALLEL UNSAFE;
|
|
|
|
|
|
|
|
-- Enqueues a job to run Ghost tables linking process for the current user
|
|
|
|
CREATE OR REPLACE FUNCTION CDB_LinkGhostTables()
|
|
|
|
RETURNS void
|
|
|
|
AS $$
|
|
|
|
DECLARE
|
2019-03-08 15:28:06 +08:00
|
|
|
username TEXT;
|
|
|
|
db_name TEXT;
|
|
|
|
ddl_tag TEXT;
|
2019-03-07 01:19:38 +08:00
|
|
|
BEGIN
|
2019-03-08 15:28:06 +08:00
|
|
|
EXECUTE 'SELECT CDB_Username();' INTO username;
|
|
|
|
EXECUTE 'SELECT current_database();' INTO db_name;
|
|
|
|
EXECUTE 'SELECT tag FROM cartodb.cdb_ddl_execution WHERE txid = txid_current();' INTO ddl_tag;
|
|
|
|
PERFORM _CDB_LinkGhostTables(username, db_name, ddl_tag);
|
2019-03-07 01:19:38 +08:00
|
|
|
DELETE FROM cartodb.cdb_ddl_execution WHERE txid = txid_current();
|
2019-03-08 15:28:06 +08:00
|
|
|
RAISE NOTICE '_CDB_LinkGhostTables() called with username=%, ddl_tag=%', username, ddl_tag;
|
2019-03-07 01:19:38 +08:00
|
|
|
END;
|
|
|
|
$$ LANGUAGE plpgsql VOLATILE PARALLEL UNSAFE SECURITY DEFINER;
|
|
|
|
|
|
|
|
-- Trigger function to call CDB_LinkGhostTables()
|
|
|
|
CREATE OR REPLACE FUNCTION _CDB_LinkGhostTablesTrigger()
|
|
|
|
RETURNS trigger
|
|
|
|
AS $$
|
|
|
|
BEGIN
|
|
|
|
PERFORM CDB_LinkGhostTables();
|
|
|
|
RETURN NULL;
|
|
|
|
END;
|
|
|
|
$$ LANGUAGE plpgsql VOLATILE PARALLEL UNSAFE SECURITY DEFINER;
|
|
|
|
|
|
|
|
-- Trigger to call CDB_LinkGhostTables() when adding a row in cartodb.cdb_ddl_execution
|
|
|
|
DROP TRIGGER IF EXISTS check_ddl_update ON cartodb.cdb_ddl_execution;
|
|
|
|
CREATE CONSTRAINT TRIGGER check_ddl_update
|
|
|
|
AFTER INSERT ON cartodb.cdb_ddl_execution
|
|
|
|
INITIALLY DEFERRED
|
|
|
|
FOR EACH ROW
|
|
|
|
EXECUTE PROCEDURE _CDB_LinkGhostTablesTrigger();
|
|
|
|
|
|
|
|
-- Event trigger to save the current transaction in cartodb.cdb_ddl_execution
|
|
|
|
CREATE OR REPLACE FUNCTION CDB_SaveDDLTransaction()
|
|
|
|
RETURNS event_trigger
|
|
|
|
AS $$
|
|
|
|
BEGIN
|
2019-03-08 15:28:06 +08:00
|
|
|
INSERT INTO cartodb.cdb_ddl_execution VALUES (txid_current(), tg_tag) ON CONFLICT (txid) DO NOTHING;
|
2019-03-07 01:19:38 +08:00
|
|
|
END;
|
|
|
|
$$ LANGUAGE plpgsql VOLATILE PARALLEL UNSAFE SECURITY DEFINER;
|
|
|
|
|
|
|
|
-- Creates the trigger on DDL events to link ghost tables
|
|
|
|
CREATE OR REPLACE FUNCTION CDB_EnableGhostTablesTrigger()
|
|
|
|
RETURNS void
|
|
|
|
AS $$
|
|
|
|
BEGIN
|
2019-03-08 15:43:22 +08:00
|
|
|
DROP EVENT TRIGGER IF EXISTS link_ghost_tables;
|
2019-03-07 01:19:38 +08:00
|
|
|
CREATE EVENT TRIGGER link_ghost_tables
|
|
|
|
ON ddl_command_end
|
2019-03-08 15:43:22 +08:00
|
|
|
WHEN TAG IN ('CREATE TABLE', 'SELECT INTO', 'DROP TABLE', 'ALTER TABLE', 'CREATE TRIGGER', 'DROP TRIGGER')
|
2019-03-07 01:19:38 +08:00
|
|
|
EXECUTE PROCEDURE CDB_SaveDDLTransaction();
|
|
|
|
END;
|
|
|
|
$$ LANGUAGE plpgsql VOLATILE PARALLEL UNSAFE;
|
|
|
|
|
|
|
|
-- Drops the trigger on DDL events to link ghost tables
|
|
|
|
CREATE OR REPLACE FUNCTION CDB_DisableGhostTablesTrigger()
|
|
|
|
RETURNS void
|
|
|
|
AS $$
|
|
|
|
BEGIN
|
2019-03-08 15:43:22 +08:00
|
|
|
DROP EVENT TRIGGER IF EXISTS link_ghost_tables;
|
2019-03-07 01:19:38 +08:00
|
|
|
END;
|
|
|
|
$$ LANGUAGE plpgsql VOLATILE PARALLEL UNSAFE;
|