前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >shell动态脚本和pl/sql动态脚本的比较

shell动态脚本和pl/sql动态脚本的比较

作者头像
jeanron100
发布2018-03-13 17:17:34
1.3K0
发布2018-03-13 17:17:34
举报
文章被收录于专栏:杨建荣的学习笔记

最近项目有一个需求,需要在多个数据库的schema上跑一些脚本。希望dba能够提供一个脚本,能够根据需求在环境中执行指定的脚本。 乍一听,没什么技术难点,为了更明白的说明问题,我举个例子。 有4个DB Instance: DB1,DB2,DB3,DB4 有6个DB Schemas, 5个table,分布如下:

代码语言:javascript
复制
                         db schemas        tables
                          user1@DB1        table1, table2, table3, table4,table5
                          user2@DB2        table1, table2, table3, table4,table5
                          user3@DB3        table1, table3
                          user4@DB4        table1, table4
                          user5@DB1        table1, table5
                          user6@DB2        table2, table3, table4

实现的目标如下,对于同时含有table1--5的db schema才需要执行指定的脚本,脚本内容都是些dml操作。 目前的情况只能够得到db schema的列表,对于里面是否还有5个表,还没有细粒度的管理。 脚本需要从db schema的列表中筛选出符合的 db schema,然后执行脚本内容。

pl/sql执行情况 :

代码语言:javascript
复制
#!/bin/ksh
export ScriptName=`basename $0`
export ScriptDir=`dirname $0`
echo $ScriptName
echo $ScriptDir
rep_conn=$1
Env_Code=$2
sqlplus -s ${rep_conn} <<EOS
set serveroutput on
set feedback off
spool app_change_tmp.log
declare
conn_str   varchar2(100);
target_conn varchar2(200);
tmp_cnts number;
cursor cur_conn_strs is
select distinct ''||username||'/'||password||'@'||db_instance||''   conn_str  from xxxx   --查询制定的配置表,从里面得到一个基本的db schema列表
group by username,password,db_instance ;
begin
for cur_conn_str in cur_conn_strs  loop
dbms_output.put_line('conn '||cur_conn_str.conn_str);
dbms_output.put_line('set serveroutput on');
dbms_output.put_line('set feedback on');
dbms_output.put_line('set echo on');
dbms_output.put_line('declare');
dbms_output.put_line('tmp_cnt number;');
dbms_output.put_line('begin');
dbms_output.put_line('select count(*) into tmp_cnt from user_synonyms where synonym_name');
dbms_output.put_line(' in('||chr(39)||'T1'||chr(39)||','||chr(39)||'T2'||chr(39)||','||chr(39)||'T3'||chr(39)||','||chr(39)||'T4'||chr(39)||','||chr(39)||'T5'||chr(39)||');');
dbms_output.put_line('dbms_output.put_line(tmp_cnt);');
dbms_output.put_line('if(tmp_cnt>=5) then ');
dbms_output.put_line('dbms_output.put_line('||chr(39)||'app POST SCRIPTS RUNNING...'||chr(39)||');');
dbms_output.put_line('@script/script1.ps ');
dbms_output.put_line('@script/script2.ps ');
dbms_output.put_line('@script/script3.ps ');
dbms_output.put_line('dbms_output.put_line('||chr(39)||'app POST SCRIPTS RUNNING...'||chr(39)||');');
dbms_output.put_line('end if;');
dbms_output.put_line('end;');
dbms_output.put_line('/');
dbms_output.put_line(tmp_cnts);
end loop;
end;
/
spool off;
@app_change_tmp.log
EOS

如上的Pl/sql生成的动态pl/sql如下, 先判断是否还有T1--T5,如果条数符合,就执行脚本内容,但是有个限制就是执行脚本的时候如果脚本中有“set linesize... set define off之类的设置的话,脚本是运行不了的,对于ddl的执行也有一些限制。 生成的动态 pl/sql 如下 :

代码语言:javascript
复制

conn user1/user1@DB1
set serveroutput on
set feedback on
set echo on
declare
tmp_cnt number;
begin
select count(*) into tmp_cnt from user_tables where table_name
in('T1','T2','T3','T4','T5');
dbms_output.put_line(tmp_cnt);
if(tmp_cnt>=5) then
dbms_output.put_line('app POST SCRIPTS RUNNING...');
@script/script1.ps
@script/script2.ps
@script/script3.ps
dbms_output.put_line('app POST SCRIPTS RUNNING...');
end if;
end;
/


conn user2/user2@DB1
set serveroutput on
set feedback on
set echo on
declare
tmp_cnt number;
begin
select count(*) into tmp_cnt from user_tables where table_name
in('T1','T2','T3','T4','T5');
dbms_output.put_line(tmp_cnt);
if(tmp_cnt>=5) then
dbms_output.put_line('app POST SCRIPTS RUNNING...');
@script/script1.ps
@script/script2.ps
@script/script3.ps
dbms_output.put_line('app POST SCRIPTS RUNNING...');
end if;
end;
/

pl/sql执行情况: shell 脚本实现动态shell :

代码语言:javascript
复制

echo 'app CHANGE START....'
cat $ScriptDir/script1.ps > $ScriptDir/app_all.ps
cat $ScriptDir/script2.ps >> $ScriptDir/app_all.ps
cat $ScriptDir/script3.ps>> $ScriptDir/app_all.ps
echo `sqlplus -s  ${rep_conn} <<EOF
set feedback off  pages 0
select distinct ''||username||'/'||password||'@'||db_instance||''   conn_str  from xxxxxx   --查询制定的配置表,从里面得到一个基本的db schema列表
group by username,password,db_instance ;
EOF`|awk  '{for (i=1;i<=NF;i++){ print "echo `sqlplus -s  " $i " <<EOS";
print  "set feedback off pages 0";
print "select (case when (select count(*) from user_synonyms where synonym_name in ('\''T5'\'','\''T1'\'','\''T2'\'','\''T3'\'','\''T4'\''))>=5 then  '\''Y @app_all.ps'\'' else '\''N no_need_to_run_app_script'\'' end) from dual; ";print "EOS` " $i}}'> $ScriptDir/dynamic_tmp.ksh
ksh $ScriptDir/dynamic_tmp.ksh |awk '{ if( $1 =="Y" ){ print "sqlplus -s " $3 " <<EOS";print "set echo on"; print $2; print "EOS"}}' >$ScriptDir/app_change_tmp.ksh
ksh $ScriptDir/app_change_tmp.ksh
rm $ScriptDir/dynamic_tmp.ksh
echo 'app CHANGE ENDED....'
rm $ScriptDir/app_change_tmp.ksh

生成的动态shell脚本1内容如下:

代码语言:javascript
复制

echo `sqlplus -s user1/user1@DB1 <<EOS
set feedback off pages 0
select (case when (select count(*) from user_tables where table_name in in('T1','T2','T3','T4','T5');)>=5 then  'Y @adj_all.ps' else 'N no_need_to_run_adj_script' end) from dual;
EOS`  user1/user1@DB1

echo `sqlplus -s  user2/user2@DB2 <<EOS
set feedback off pages 0
select (case when (select count(*) from user_tables where table_name in in('T1','T2','T3','T4','T5');)>=5 then  'Y @adj_all.ps' else 'N no_need_to_run_adj_script' end) from dual;
EOS` user2/user2@DB2
代码语言:javascript
复制

执行动态shell脚本1后生成的脚本2内容如下:

sqlplus -s user1/user1@DB1 <<EOS
@adj_all.ps
EOS
sqlplus -s user2/user2@DB2 <<EOS
@adj_all.ps
EOS

############## shell 脚本实现动态shell ################################
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2014-03-04,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 杨建荣的学习笔记 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档