免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 4570 | 回复: 3
打印 上一主题 下一主题

请教,如何在oracle的存储过程中调用OS命令? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2006-08-07 16:54 |只看该作者 |倒序浏览
想在oracle的存储过程中调用OS命令去执行OS中的脚本.

请教各位大侠有什么较好的方法实现;

论坛徽章:
0
2 [报告]
发表于 2006-08-07 19:13 |只看该作者
HOST('xxx')

原帖由 Emerbor 于 2006-8-7 16:54 发表
想在oracle的存储过程中调用OS命令去执行OS中的脚本.

请教各位大侠有什么较好的方法实现;

论坛徽章:
0
3 [报告]
发表于 2006-08-07 20:43 |只看该作者
楼上说的其实也是对的.

PLSQL执行shell
[Typeset page]

转载请著名: www.xiaobaicai.com

CREATE OR REPLACE AND COMPILE JAVA SOURCE NAMED "Hawk_OSCommand" AS
import java.io.*;                                                               
public class Hawk_OSCommand{                                                         
  public static String Run(String Command){                                    
     try{                                                                       
        Runtime.getRuntime().exec(Command);                                    
        return("0");                                                                  
     }                                                                          
     catch (Exception e){                                                      
        System.out.println("Error running command: " + Command +
                                 "\n" + e.getMessage());   
        return(e.getMessage());  
     }                                                                          
  }                                                                             
}

/   

CREATE OR REPLACE FUNCTION Hawk_OSCommand_Run(
    Command   IN   STRING )
    RETURN VARCHAR2
IS
    LANGUAGE JAVA
    NAME 'Hawk_OSCommand.Run(java.lang.String) return int';

/

SELECT * FROM user_errors;

/

Execute SYS.dbms_java.grant_permission( 'APPS', 'SYS:java.io.FilePermission', '<<ALL FILES>>','execute');

EXECUTE dbms_java.grant_permission( 'APPS','SYS:java.lang.RuntimePermission','writeFileDescriptor','*' );

EXECUTE dbms_java.grant_permission( 'APPS', 'SYS:java.lang.RuntimePermission','readFileDescriptor','*' );


--底下的测试脚本可以在navigator 运行,也可以在unix上的sqlplus,还可以在本地的sqlplus运行.

--底下这段调用有问题. 会告诉你ls不存在
Declare
    v Varchar2(2000);
BEGIN
    v := Hawk_OSCommand_Run('ls -l');  
    DBMS_OUTPUT.Put_Line(v);
End;

-- 这段执行没问题.记住命令要写全,用/usr/bin/mv
Declare
    v Varchar2(2000);
BEGIN
    v := Hawk_OSCommand_Run('/usr/bin/mv /home/testuser/joodhawk/shell/test_outrename2.txt

/home/testuser/joodhawk/shell/rename.txt');  
    DBMS_OUTPUT.Put_Line(v);
End;

--这段执行没问题,可以创建一个文件touch.txt
Declare
    v Varchar2(2000);
BEGIN
    v := Hawk_OSCommand_Run('/usr/bin/touch /home/testuser/joodhawk/shell/touch.txt');  
    DBMS_OUTPUT.Put_Line(v);
End;

--这段执行有问题,似乎不支持管道
Declare
    v Varchar2(2000);
BEGIN
    v := Hawk_OSCommand_Run('/usr/bin/ls -e > /home/testuser/joodhawk/shell/test_cal.txt');  
    DBMS_OUTPUT.Put_Line(v);
End;

后来高人点拨,终于明白实质是不支持管道的:
Input, Ouput redirection (and Pipe) would not work for the Java Class Below .
Those features are actually the features of your shell environment.
(If you're curios why it won't work, here's some good explanation

http://www.ensta.fr/~diam/java/online/io/javazine.html)

If you want to access standard output/error, the first code piece from askTom  
would give you an idea.

http://asktom.oracle.com/pls/ask ... PLAYID:952229840241

You can even improve this snippet to put the standart output in a String array

and return back to your wrapper function (use a IN OUT parameter)
25       // Echo back what the program spit out
26       while ((len = bis.read(buffer, 0, bufSize)) != -1)
27          System.out.write(buffer, 0, len);

Hope this helps
-erdal

Note: 这种调用还是存在一些问题,最好的方式是用java读取外部的文件,然后传递到command里头的.
调用方式采用Hawk_OSCommand(外部文件的名字)

还有c 调用的方式,可以参考metalink: 130799.1
java 参考
http://www.ensta.fr/~diam/java/online/io/javazine.html
asktom参考
http://asktom.oracle.com/pls/ask ... PLAYID:952229840241

论坛徽章:
0
4 [报告]
发表于 2006-08-07 20:45 |只看该作者
另外,这个是我在做的时候的一点笔记.

转载著名来自: www.xiaobaicai.com

调试了一个晚上,也没调试成功,有机会哪天继续一下.

create or replace procedure hawk_host( cmd in varchar2 )
as
    status number;
    v_cmd VARCHAR2(4000);   
begin
    dbms_pipe.pack_message( cmd );
    status := dbms_pipe.send_message( 'HOST_PIPE' );
    if ( status <> 0 ) then
      raise_application_error( -20001, 'Pipe error' );
    ELSE
     dbms_output.put_line('good '|| DBMS_PIPE.receive_message( 'HOST_PIPE' ));
     dbms_pipe.unpack_message( v_cmd );  
     dbms_output.put_line(v_cmd);   
    end if;
end;

DECLARE
BEGIN
    hawk_host('chmod 555 /home/testuser/joodhawk/testdir/*.*');
EXCEPTION
    WHEN OTHERS THEN
        dbms_output.put_line('error');
END;


参考;: tom

昨天尝试没有成功,最后我提交了问题.maillist 有人给我做答.
有时间我会用java来尝试一下的.先做到这.

Any of these options will work.  I wouldn't use DBMS_PIPE unless I was running
Oracle 7, which doesn't have external procedures.  It requires you to make sure
that your pipe daemon is running anytime the database is running and it has to
be started separately.  External procedures are a bit more difficult to install
and configure than Java stored procedures, but perform well and are
not bad for someone who knows C better than Java.  I have one application
that uses an external procedure for this, but more recent ones use Java,
and we'll eventually convert the oldest one, because the Java version has
proven to be more portable and easier to configure.

No matter how you do this, consider VERY carefully the security ramifications
Misuse of this capability can wreak havoc on your server and database.  Our
version uses the SQL_PRODUCT_PROFILE table to control who may do what with a
shell script.  Yes, this is the same table that SQL*Plus can use to control
what users may do.

Do some searches on the WWW, and I think you'll find sample code from several
people for each of these approaches.  Sample code for the DBMS_PIPE method used
to be listed in the Oracle Application Developer's guide.  I'll have a look and
see if I have published our version.  If not, I may do so, but I may need to
add some documentation before I do.
--------------------------------------------------------------------------------
From: ml-errors@fatcity.com [mailto:ml-errors@fatcity.com] On Behalf Of Joodhawk Lin
Sent: Wednesday, July 12, 2006 6:13 AM
To: Multiple recipients of list ODTUG-JAVA-L
Subject: I want to inovke shell program from pl/sql, is any good idea?

Hello,

I want to invoke shell program from pl/sql, is any good idea to complete it?
1) We can use external procedure (c)
2) Call java procedure à invoke shell.
3) Use DBMS_PIPE package.
Any other method?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP