
Looks like someone forgot to fill their data model properly... :-)
This is about Oracle SQL development and may sometimes even cover DBA topics.
spool C:\Temp\pkg-test.txt
set echo on pagesize 100 linesize 150 serverout on
create package call_test
as
function f_pub(x in number) return number;
procedure p_pub_direct;
procedure p_pub_direct_qualified;
procedure p_pub_immediate;
procedure p_pub_immediate_qualified;
procedure p_priv_direct;
procedure p_priv_immediate;
procedure p_priv_immediate_qualified;
procedure p_priv_var_direct;
procedure p_priv_var_immediate;
end call_test;
/
show errors package call_test
create package body call_test
as
function f_pub(x in number) return number
as
begin
return x * 10;
end f_pub;
function f_priv(x in number) return number
as
begin
return x * 20;
end f_priv;
procedure p_pub_direct
as
begin
dbms_output.put_line('direct call of public function');
for c in ( select f_pub(2) as num from dual ) loop
dbms_output.put_line('value ' || c.num);
end loop;
end p_pub_direct;
procedure p_pub_direct_qualified
as
begin
dbms_output.put_line('direct call of public function using qualified name');
for c in ( select call_test.f_pub(2) as num from dual ) loop
dbms_output.put_line('value ' || c.num);
end loop;
end p_pub_direct_qualified;
procedure p_pub_immediate
as
tmp number;
begin
dbms_output.put_line('execute immediate with public function');
execute immediate 'select f_pub(2) as num from dual' into tmp;
dbms_output.put_line('value ' || tmp);
end p_pub_immediate;
procedure p_pub_immediate_qualified
as
tmp number;
begin
dbms_output.put_line('execute immediate with public function using qualified name');
execute immediate 'select call_test.f_pub(2) as num from dual' into tmp;
dbms_output.put_line('value ' || tmp);
end p_pub_immediate_qualified;
procedure p_priv_direct
as
begin
dbms_output.put_line('direct call of private function');
dbms_output.put_line('this will not compile');
/*
for c in ( select f_priv(2) as num from dual ) loop
dbms_output.put_line('value ' || c.num);
end loop;
for c in ( select call_test.f_priv(2) as num from dual ) loop
dbms_output.put_line('value ' || c.num);
end loop;
*/
end p_priv_direct;
procedure p_priv_immediate
as
tmp number;
begin
dbms_output.put_line('execute immediate with private function');
execute immediate 'select f_priv(2) as num from dual' into tmp;
dbms_output.put_line('value ' || tmp);
end p_priv_immediate;
procedure p_priv_immediate_qualified
as
tmp number;
begin
dbms_output.put_line('execute immediate with private function using qualified name');
execute immediate 'select call_test.f_priv(2) as num from dual' into tmp;
dbms_output.put_line('value ' || tmp);
end p_priv_immediate_qualified;
procedure p_priv_var_direct
as
foo number := 123;
begin
dbms_output.put_line('direct call with private variable');
for c in ( select foo as num from dual ) loop
dbms_output.put_line('value ' || c.num);
end loop;
end p_priv_var_direct;
procedure p_priv_var_immediate
as
tmp number;
foo number := 123;
begin
dbms_output.put_line('execute immediate with private variable');
execute immediate 'select foo as num from dual' into tmp;
dbms_output.put_line('value ' || tmp);
end p_priv_var_immediate;
end call_test;
/
show errors package body call_test
set echo off
show release
exec call_test.p_pub_direct;
exec call_test.p_pub_direct_qualified;
exec call_test.p_pub_immediate;
exec call_test.p_pub_immediate_qualified;
exec call_test.p_priv_direct;
exec call_test.p_priv_immediate;
exec call_test.p_priv_immediate_qualified;
exec call_test.p_priv_var_direct;
exec call_test.p_priv_var_immediate;
drop package call_test
/
spool off
release 1002000400
direct call of public function
value 20
PL/SQL procedure successfully completed.
direct call of public function using qualified name
value 20
PL/SQL procedure successfully completed.
execute immediate with public function
BEGIN call_test.p_pub_immediate; END;
*
ERROR at line 1:
ORA-00904: "F_PUB": invalid identifier
ORA-06512: at "RK.CALL_TEST", line 39
ORA-06512: at line 1
execute immediate with public function using qualified name
value 20
PL/SQL procedure successfully completed.
direct call of private function
this will not compile
PL/SQL procedure successfully completed.
execute immediate with private function
BEGIN call_test.p_priv_immediate; END;
*
ERROR at line 1:
ORA-00904: "F_PRIV": invalid identifier
ORA-06512: at "RK.CALL_TEST", line 72
ORA-06512: at line 1
execute immediate with private function using qualified name
BEGIN call_test.p_priv_immediate_qualified; END;
*
ERROR at line 1:
ORA-00904: "CALL_TEST"."F_PRIV": invalid identifier
ORA-06512: at "RK.CALL_TEST", line 81
ORA-06512: at line 1
direct call with private variable
value 123
PL/SQL procedure successfully completed.
execute immediate with private variable
BEGIN call_test.p_priv_var_immediate; END;
*
ERROR at line 1:
ORA-00904: "FOO": invalid identifier
ORA-06512: at "RK.CALL_TEST", line 101
ORA-06512: at line 1
alter session set nls_timestamp_format = 'YYYY-MM-DD HH24:MI.SS.FF3';
prompt "---------- START ---------- "
column ts format a30
set feedback on
set serveroutput on
create sequence s1
start with 1 increment by 1 cache 20
/
create table t1 (
id number(10) primary key,
name1 varchar2(15 char),
name2 varchar2(15 char)
)
/
create table t1_hist (
ts timestamp(3) not null,
id number(10),
name1 varchar2(15 char),
name2 varchar2(15 char),
constraint pk2 primary key ( id, ts ) using index compress 1
)
/
create global temporary table t1_hist_ptr (
id number(10) primary key,
rid rowid not null -- we use rowid for efficient row access
)
on commit delete rows
/
create trigger t1_insert
after insert or update of id, name1, name2
on t1
for each row
declare
rid rowid;
begin
if updating and :new.id <> :old.id then
-- PK has changed!
update t1_hist_ptr set
id = :new.id
where id = :old.id;
dbms_output.put_line('===> updated id ' || :old.id || ' -> ' || :new.id );
end if;
update t1_hist set
ts = systimestamp,
id = :new.id, -- important!
name1 = :new.name1,
name2 = :new.name2
where t1_hist.rowid = ( select rid from t1_hist_ptr where id = :new.id );
dbms_output.put_line('===> updated rowcount ' || sql%rowcount);
if sql%rowcount = 0 then
-- not there yet, need to insert history record
insert into t1_hist values ( systimestamp, :new.id, :new.name1, :new.name2 )
returning rowid into rid;
insert into t1_hist_ptr values ( :new.id, rid );
end if;
end t1_insert;
/
show errors trigger t1_insert
/*
* MAIN START
*/
-- initially empty
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
-- inserts
set echo on
insert into t1 values ( s1.nextval, '1 original', 'bar' )
/
insert into t1 values ( s1.nextval, '2 original', 'buz' )
/
set echo off
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
-- update
set echo on
update t1 set name1 = '1 updated 1' where id = 1
/
set echo on
update t1 set name1 = '1 updated 2' where id = 1
/
set echo off
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
commit
/
-- update with rollback
set echo on
update t1 set name1 = '2 updated 3' where id = 2
/
set echo off
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
rollback
/
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
-- insert with dummy id
insert into t1 values ( -1, '-1 original', 'dummy entry' )
/
update t1 set id = s1.nextval where id = -1
/
commit
/
select t1_hist.*
from t1_hist
order by t1_hist.ts, t1_hist.id
/
/*
* MAIN END
*/
drop table t1_hist_ptr
/
drop table t1_hist
/
drop table t1
/
drop sequence s1
/
prompt "---------- END ----------"