支持的版本: 当前 (17) / 16 / 15 / 14 / 13
开发版本: devel
不支持的版本: 12 / 11 / 10 / 9.6 / 9.5 / 9.4 / 9.3

38.1. 事件触发器行为概述 #

当与其关联的事件在其定义的数据库中发生时,事件触发器会触发。目前,唯一支持的事件是 loginddl_command_startddl_command_endtable_rewritesql_drop。未来版本可能会增加对其他事件的支持。

login 事件在经过身份验证的用户登录系统时发生。此事件的触发器程序中的任何错误都可能阻止成功登录系统。可以通过在连接字符串或配置文件中将 event_triggers 设置为 false 来解决此类错误。或者,您可以在单用户模式下重启系统(因为在此模式下禁用了事件触发器)。有关使用单用户模式的详细信息,请参阅 postgres 参考页面。login 事件也将在备用服务器上触发。为防止服务器变得不可访问,此类触发器在备用服务器上运行时必须避免向数据库写入任何内容。此外,建议避免在 login 事件触发器中运行长时间查询。请注意,例如,取消 psql 中的连接不会取消正在进行的 login 触发器。

ddl_command_start 事件发生在执行 CREATEALTERDROPSECURITY LABELCOMMENTGRANTREVOKE 命令之前。在事件触发器触发之前,不会检查受影响的对象是否存在。但是,作为例外,此事件不会为针对共享对象(数据库、角色和表空间)的 DDL 命令或针对事件触发器本身的命令发生。事件触发机制不支持这些对象类型。ddl_command_start 也发生在执行 SELECT INTO 命令之前,因为这等同于 CREATE TABLE AS

ddl_command_end 事件发生在此同一组命令执行之后。要获取有关DDL操作的更多详细信息,请使用 ddl_command_end 事件触发器代码中的集合返回函数 pg_event_trigger_ddl_commands()(请参阅 第 9.30 节)。请注意,触发器在操作发生后(但在事务提交之前)触发,因此可以读取系统目录,如同已更改一样。

sql_drop 事件在任何删除数据库对象的操作的 ddl_command_end 事件触发器之前触发。要列出已删除的对象,请使用 sql_drop 事件触发器代码中的集合返回函数 pg_event_trigger_dropped_objects()(请参阅 第 9.30 节)。请注意,触发器在从系统目录中删除对象后执行,因此无法再查找它们。

table_rewrite 事件发生在 ALTER TABLEALTER TYPE 命令的某些操作重写表之前。虽然可以使用其他控制语句来重写表,如 CLUSTERVACUUM,但 table_rewrite 事件不会被它们触发。要查找重写表的 OID,请使用函数 pg_event_trigger_table_rewrite_oid()(请参阅 第 9.30 节)。要发现重写的原因,请使用函数 pg_event_trigger_table_rewrite_reason()

事件触发器(与其他函数一样)无法在中止的事务中执行。因此,如果 DDL 命令失败并出现错误,则不会执行任何关联的 ddl_command_end 触发器。相反,如果 ddl_command_start 触发器失败并出现错误,则不会触发进一步的事件触发器,也不会尝试执行命令本身。同样,如果 ddl_command_end 触发器失败并出现错误,则 DDL 语句的效果将被回滚,就像在包含事务中止的任何其他情况下一样。

有关事件触发机制支持的命令的完整列表,请参阅第 38.2 节

使用命令 CREATE EVENT TRIGGER 创建事件触发器。为了创建事件触发器,您必须首先创建一个具有特殊返回类型 event_trigger 的函数。此函数无需(也可能不)返回值;返回类型仅用作指示该函数将作为事件触发器调用的信号。

如果为特定事件定义了多个事件触发器,它们将按触发器名称的字母顺序触发。

触发器定义还可以指定 WHEN 条件,因此,例如,只有当用户希望拦截的特定命令时,才会触发 ddl_command_start 触发器。此类触发器的常见用法是限制用户可以执行的 DDL 操作范围。

提交更正

如果您在文档中发现任何不正确、与您使用特定功能的经验不符或需要进一步澄清的内容,请使用此表单报告文档问题。