Emmanuel BENOîT
3e109b13bc
* Added user mapping on the "remote" logging database for the administrative user. This allows calls to sys.write_sql_log() to succeed when they are executed by code with administrative privileges. * Added test suites for both the link to the database and the function itself. * Replaced RAISE NOTICE with actual logging in the universe generator
291 lines
9.2 KiB
PL/PgSQL
291 lines
9.2 KiB
PL/PgSQL
-- LegacyWorlds Beta 6
|
|
-- PostgreSQL database scripts
|
|
--
|
|
-- Logging functions and triggers
|
|
--
|
|
-- Copyright(C) 2004-2010, DeepClone Development
|
|
-- --------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- LOG CLEANUP --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
|
|
|
|
--
|
|
-- Remove expired entries from a log table
|
|
--
|
|
-- Parameters:
|
|
-- l_type Log table to clean up
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION sys.clean_log_table( l_type log_type )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
expiry INT;
|
|
expiry_int INTERVAL;
|
|
BEGIN
|
|
expiry := floor( sys.get_constant( 'log.maxAge.' || lower( l_type::TEXT ) ) );
|
|
expiry_int := ( expiry || 's' )::INTERVAL;
|
|
EXECUTE 'DELETE FROM ' || l_type::TEXT || '.logs WHERE t <= $1'
|
|
USING now() - expiry_int;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Clean all log tables
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION sys.clean_logs( )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
lt log_type;
|
|
BEGIN
|
|
FOR lt IN SELECT x FROM unnest( enum_range( NULL::log_type ) ) AS x
|
|
LOOP
|
|
PERFORM sys.clean_log_table( lt );
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION sys.clean_logs( ) TO :dbuser;
|
|
|
|
|
|
|
|
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- LOG APPENDERS --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
|
|
|
|
--
|
|
-- Write an user log entry
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION users.write_log( account_id INT , lv log_level , msg TEXT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
INSERT INTO users.logs (credentials_id , level , message)
|
|
VALUES ( account_id , lv , msg );
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Write an administrative log entry
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION admin.write_log( adm_id INT , lv log_level , msg TEXT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
INSERT INTO admin.logs ( admin_id , level , message )
|
|
VALUES ( adm_id , lv , msg );
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Write a system log entry
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION sys.write_log( cmp TEXT , lv log_level , msg TEXT , OUT entry_id BIGINT )
|
|
STRICT VOLATILE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
BEGIN
|
|
INSERT INTO sys.logs( component , level , message )
|
|
VALUES ( cmp , lv , msg )
|
|
RETURNING id INTO entry_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION sys.write_log( TEXT , log_level , TEXT ) TO :dbuser;
|
|
|
|
|
|
/*
|
|
* Remotely append a system log entry
|
|
*
|
|
* This function is called from within transactions in order to write to the
|
|
* system log. Unlike sys.write_log(), entries added through this function
|
|
* will not be lost if the transaction is rolled back.
|
|
*
|
|
* Since the function is meant to be called from SQL code, it does not return
|
|
* anything as the identifier of the new entry is not required.
|
|
*
|
|
* Parameters:
|
|
* _component The component that is appending to the log
|
|
* _level The log level
|
|
* _message The message to write
|
|
*/
|
|
|
|
CREATE OR REPLACE FUNCTION sys.write_sql_log( _component TEXT , _level log_level , _message TEXT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $write_sql_log$
|
|
BEGIN
|
|
BEGIN
|
|
PERFORM dblink_connect( 'cn_logging' , 'srv_logging' );
|
|
EXCEPTION
|
|
WHEN duplicate_object THEN
|
|
-- Ignore the error, assume we're connected
|
|
END;
|
|
|
|
PERFORM * FROM dblink( 'cn_logging' ,
|
|
'SELECT * FROM sys.write_log( '
|
|
|| quote_literal( _component ) || ' , '''
|
|
|| _level::text || '''::log_level , '
|
|
|| quote_literal( _message ) || ' )'
|
|
) AS ( entry_id bigint );
|
|
END;
|
|
$write_sql_log$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION sys.write_sql_log( TEXT , log_level , TEXT ) TO :dbuser;
|
|
|
|
|
|
|
|
--
|
|
-- Append an exception log entry
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION sys.append_exception( l_id BIGINT , cname TEXT , msg TEXT , OUT entry_id BIGINT )
|
|
CALLED ON NULL INPUT
|
|
VOLATILE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
d INT;
|
|
BEGIN
|
|
SELECT INTO d max( depth ) FROM sys.exceptions
|
|
WHERE log_id = l_id;
|
|
IF d IS NULL THEN
|
|
d := 0;
|
|
END IF;
|
|
d := d + 1;
|
|
|
|
INSERT INTO sys.exceptions ( log_id , depth , exc_class , message )
|
|
VALUES ( l_id , d , cname , msg )
|
|
RETURNING id INTO entry_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION sys.append_exception( BIGINT , TEXT , TEXT ) TO :dbuser;
|
|
|
|
|
|
|
|
--
|
|
-- Append a stack trace entry
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION sys.append_trace( e_id BIGINT , loc TEXT , f_name TEXT , lnb INT )
|
|
RETURNS VOID
|
|
CALLED ON NULL INPUT
|
|
VOLATILE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
d INT;
|
|
BEGIN
|
|
SELECT INTO d max( depth ) FROM sys.stack_traces
|
|
WHERE exception_id = e_id;
|
|
IF d IS NULL THEN
|
|
d := 0;
|
|
END IF;
|
|
d := d + 1;
|
|
|
|
INSERT INTO sys.stack_traces ( exception_id , depth , location , file_name , line_number )
|
|
VALUES ( e_id , d , loc , f_name , lnb );
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION sys.append_trace( BIGINT , TEXT , TEXT , INT ) TO :dbuser;
|
|
|
|
|
|
|
|
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- LOGS VIEW FOR EASY SELECTION --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
|
|
|
|
CREATE VIEW sys.system_logs_view
|
|
AS SELECT l.id , l.t , l.component , l.level , l.message ,
|
|
( CASE WHEN e.id IS NOT NULL THEN l.id ELSE NULL END )::BIGINT AS exception_id
|
|
FROM sys.logs l
|
|
LEFT OUTER JOIN sys.exceptions e ON e.log_id = l.id AND e.depth = 1;
|
|
|
|
GRANT SELECT ON sys.system_logs_view TO :dbuser;
|
|
|
|
|
|
CREATE VIEW sys.admin_logs_view
|
|
AS SELECT NULL::BIGINT AS id , l.t , a.appear_as AS component , l.level ,
|
|
l.message , NULL::BIGINT AS exception_id
|
|
FROM admin.logs l
|
|
INNER JOIN admin.administrators a ON a.id = l.admin_id;
|
|
|
|
GRANT SELECT ON sys.admin_logs_view TO :dbuser;
|
|
|
|
|
|
CREATE VIEW sys.player_logs_view
|
|
AS SELECT NULL::BIGINT AS id , l.t , a.address AS component , l.level ,
|
|
l.message , NULL::BIGINT AS exception_id
|
|
FROM users.logs l
|
|
INNER JOIN users.addresses a ON a.id = l.credentials_id;
|
|
|
|
GRANT SELECT ON sys.player_logs_view TO :dbuser;
|
|
|
|
|
|
|
|
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- ADMIN ERROR REPORT FUNCTION --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
-- --------------------------------------------------------------------------------------------------------------------------------------------------------------- --
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION admin.get_error_entries( )
|
|
RETURNS SETOF BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
last_report TIMESTAMP WITHOUT TIME ZONE;
|
|
new_report TIMESTAMP WITHOUT TIME ZONE;
|
|
e_id BIGINT;
|
|
BEGIN
|
|
new_report := now( );
|
|
SELECT INTO last_report last_error_recap FROM sys.status FOR UPDATE;
|
|
UPDATE sys.status SET last_error_recap = new_report;
|
|
|
|
RETURN QUERY SELECT id FROM sys.system_logs_view
|
|
WHERE t >= last_report AND t < new_report AND exception_id IS NOT NULL
|
|
ORDER BY t DESC;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
GRANT EXECUTE ON FUNCTION admin.get_error_entries( ) TO :dbuser;
|