我有一个“明星”数据库。一个是做一些工作;另外两个具有将元数据从第一个数据库中拉入其引用表的作业。我想阻止任何人更新,删除或插入两个“奴隶”中的参考表上的任何记录;这些表应该由预定的作业更新。
我目前正在使用触发器执行此操作,该触发器检查当前SID是否在USER_SCHEDULER_RUNNING_JOBS中,并且具有我期望运行的作业名称。我想将其更改为使用我正在运行的作业的JOB_ID。
这是我目前的设置;假设一个非常简单的表:
create table a ( b number );
以及以下工作:
begin
dbms_scheduler.create_job(
job_name => 'test_job'
, job_type => 'PLSQL_BLOCK'
, job_action => 'begin
merge into a a
using ( select 1 x from dual@db2 ) b
on (1 = 2)
when not matched then
insert values (b.x);
commit;
end;'
, start_date => sysdate
, repeat_interval => 'FREQ = MINUTELY;'
, enabled => true
);
end;
/
我正在使用此触发器:
create or replace trigger tr_blah
before insert or update on a
declare
l_ct number;
begin
select count(*) into l_ct
from user_scheduler_running_jobs
where session_id = sys_context('USERENV','SID')
and job_name = 'TEST_JOB'
;
if l_ct = 0 then
raise_application_error(-20000, 'FAIL');
end if;
end;
/
这是不优雅的;但是,更糟糕的是,我必须为每个数据库中的每个表创建一个单独的触发器,并且每次都更改作业名称。无法动态创建触发器;这变得令人厌倦,并且存在很多可能出现的错误。
SYS_CONTEXT()
的参数为FG_JOB_ID
和BG_JOB_ID
。他们的描述,特别是FG_JOB_ID
的描述暗示它们可能是当前正在运行的作业的JOB_ID。将触发器更改为以下(我已尝试过两者):
create or replace trigger tr_a
before insert or update or delete on a
declare
l_ct number;
begin
raise_application_error(-20000, sys_context('USER_ENV', 'BG_JOB_ID'));
end;
/
以下结果
ORA-20000:
ORA-06512: at "REF.TR_A", line 4
ORA-04088: error during execution of trigger 'REF.TR_A'
ORA-06512: at line 2
这意味着FG_JOB_ID
和BG_JOB_ID
都为空。有没有一种方法可以确定当前会话中运行的作业的ID,所以我不需要每次都使用JOB_NAME?
答案 0 :(得分:0)
最优雅的解决方案是使用不同的数据库用户。确保作业在具有表上的更新,插入和删除授权的用户下运行(可能是表的架构所有者)。不要将这些拨款提供给其他用户。
无需乱搞触发器等。