延續(xù)《Windows Server 2003中iis配置php》一文
服務(wù)器上使用Apache2+PHP正常運(yùn)行,換成IIS+PHP,先后出現(xiàn)了php.ini的環(huán)境變量無(wú)法讀取,php中驗(yàn)證碼無(wú)法顯示的問(wèn)題,如今又有人反應(yīng)無(wú)法上傳圖片的問(wèn)題。
從IIS替換Apache2的過(guò)程僅僅是開(kāi)啟IIS,關(guān)閉Apache2,其它的沒(méi)什么變化,但是卻發(fā)生了如此多的差異,看樣子IIS支持PHP還是有很多要進(jìn)行修改的。
分析:
根據(jù)上面的描述,我懷疑問(wèn)題出在IIS的權(quán)限配置上,IUSR_MACHINE的帳戶(hù)對(duì)upload沒(méi)有寫(xiě)入的權(quán)限,于是進(jìn)行權(quán)限修改,IIS下的權(quán)限,NTFS下的權(quán)限都進(jìn)行修改,但是終究都沒(méi)用,查找網(wǎng)絡(luò)上的資料也沒(méi)有相應(yīng)的,對(duì)上傳頁(yè)面進(jìn)行測(cè)試,流程為:
swf文件調(diào)用save.php上傳文件---->swf文件對(duì)上傳的文件進(jìn)行重命名--->名字返回給save.php--->顯示出最后的名字。
現(xiàn)在的問(wèn)題一直停留在swf對(duì)文件重命名的這里,一直沒(méi)有到顯示出最后的名字,并且swf文件不參與上傳過(guò)程,那就只能在save.php文件中進(jìn)行問(wèn)題查找了,在該文件中進(jìn)行測(cè)試,最后顯示的名字所使用的變量為fileName,于是插入下面的語(yǔ)句進(jìn)行測(cè)試:
echo "fileName=2008*****.gif";
這句話(huà)的作用就是使得fileName有值,save.php能正常顯示,先把原來(lái)的語(yǔ)句一句一句的進(jìn)行屏蔽測(cè)試,都正常的返回了,但是當(dāng)測(cè)試到:
if (mailto:!@move_uploaded_file($f[%22tmp_name"], $dest_dir.'/'.$fileName)) header("HTTP/1.0 404 Not Found");
這句話(huà)的時(shí)候問(wèn)題出現(xiàn)了,不能上傳,查找上下文,一直沒(méi)發(fā)現(xiàn)tmp_name的變量,不過(guò)看意思是先把文件上傳到一個(gè)臨時(shí)文件,再挪動(dòng)到目的位置,那這個(gè)tmp位置在哪里呢?是不是這個(gè)位置不可寫(xiě),
才導(dǎo)致了無(wú)法上傳文件?
查找網(wǎng)上資料,發(fā)現(xiàn)php.ini下面有2個(gè)地方關(guān)于上傳的配置:
file_uploads = On 這里設(shè)置是否允許HTTP上傳,默認(rèn)應(yīng)該為ON的
;upload_tmp_dir= 這里設(shè)置上傳文件存放的臨時(shí)位置
網(wǎng)上對(duì)于這2個(gè)地方的相關(guān)資料有:
I try to set up file uploading under IIS 7 and PHP 5.
First problem was to set 2 variables in php.ini
file_uploads = On //這里是說(shuō)php.ini文件這個(gè)地方設(shè)置成On
upload_tmp_dir = "C:\Inetpub\wwwroot\uploads" //這個(gè)路徑就是自己設(shè)置的上傳文件臨時(shí)存儲(chǔ)路徑
For some reasons such directory name works,
but "upload_tmp" won't work.
The second problem was to set correct user rigths for upload folders where you try to save your file. I set my upload folder rights for the "WORKGROUP/users" for the full access. You may experiment by yourselves if you not need execute access, for example.
我的php.ini中upload_tmp_dir是被注釋的,沒(méi)有啟用,更沒(méi)有設(shè)置,可是為什么Apache2卻可以正常上傳呢?難道問(wèn)題真的出在這里?
解決:
新建一個(gè)文件夾做臨時(shí)上傳目錄,按照上面的英文說(shuō)明修改php.ini中相應(yīng)的那2項(xiàng),把臨時(shí)上傳目錄upload_tmp_dir設(shè)置成剛才建立的文件夾,把該文件夾的權(quán)限賦予“IUSR_計(jì)算機(jī)名”用戶(hù)可寫(xiě),重新啟動(dòng)IIS,上傳試試,問(wèn)題真的就這樣解決了。
最終的分析答案:
上面的內(nèi)容寫(xiě)于09年,但是現(xiàn)在2010年7月我新增一臺(tái)服務(wù)器,又出現(xiàn)了這個(gè)問(wèn)題,同時(shí)再次按照上面的解決方法實(shí)施,在操作的過(guò)程中大概是由于哪里出了錯(cuò),竟然沒(méi)有成功,不得不抽出點(diǎn)時(shí)間來(lái)研究具體原因,找到了最終產(chǎn)生這個(gè)問(wèn)題的原因如下。
無(wú)法上傳文件,不代表所有文件都無(wú)法上傳,因?yàn)槲业囊粋€(gè)網(wǎng)站,flash調(diào)用fwrite()傳頭像之類(lèi)的成功了,但是調(diào)用@move_uploaded_file($f["tmp_name"], $dest_dir.'/'.$fileName)這樣的函數(shù)傳照片的時(shí)候仍舊無(wú)法上傳。
經(jīng)過(guò)我的分析,原因是由于fwrite()是傳的二進(jìn)制文件,而move_uploaded_file()傳的是文本文件,而windows操作系統(tǒng)是區(qū)分這2種文件的 [參考php手冊(cè)fwrite()函數(shù)的說(shuō)明],這也就是說(shuō)這2種不同的文件在php環(huán)境下上傳時(shí)所存儲(chǔ)的臨時(shí)上傳目錄是不同的,由于在配置IIS環(huán)境下的PHP的時(shí)候,設(shè)置的臨時(shí)目錄為E:\tmp,設(shè)置該目錄的iusr用戶(hù)可寫(xiě),二進(jìn)制文件即可上傳,所以我懷疑該目錄就是二進(jìn)制文件上傳臨時(shí)文件的存儲(chǔ)位置,那么move_uploaded_file()傳的文本文件的臨時(shí)文件存儲(chǔ)位置在哪里呢?其實(shí)就是在上面的那段英文里面,upload_tmp_dir設(shè)置的路徑就是了,但是我的幾臺(tái)服務(wù)器中,每臺(tái)服務(wù)器的這個(gè)設(shè)置的值都是被注釋掉的“no value”,為什么有的服務(wù)器可以上傳,而有的服務(wù)器不可以上傳呢?這也就回到了以前我提出的問(wèn)題,為什么Apache2可以上傳而iis不可以上傳呢?
這次我再次分析upload.php文件,分析其中造成該故障的代碼具體內(nèi)容如下:
復(fù)制代碼 代碼如下:
// 檢查是否有文件上傳
if (! $_FILES['upload'.$num]['name'] == ""){
if ($_FILES['upload'.$num]['size'] < $max_size) {
1、 echo "文件上傳路徑:".$location.$_FILES['upload'.$num]['name'];
2、echo "文件臨時(shí)文件名:".$_FILES['upload'.$num]['tmp_name'];
3、 move_uploaded_file($_FILES['upload'.$num]['tmp_name'],$location.$_FILES['upload'.$num]['name']) or $event = "Failure";
} else {
$event = "File too large!";
}
其中正常代碼中第2句是不存在的,為了測(cè)試方便我加上來(lái)的,它的主要目的就是測(cè)試我的php.ini沒(méi)有設(shè)置upload_dir_tmp的值的時(shí)候,上傳的文件臨時(shí)保存在哪里的,經(jīng)過(guò)這個(gè)測(cè)試發(fā)現(xiàn)原來(lái)在不配置php.ini的upload_dir_tmp的值的時(shí)候,默認(rèn)的存儲(chǔ)位置是在C:\windows\temp目錄,并且臨時(shí)文件是以.tmp為后綴存儲(chǔ)的,該文件馬上就會(huì)被刪除,所以你想通過(guò)操作系統(tǒng)的文件修改搜索功能是無(wú)法找到的,也就無(wú)法找到upload_dir_tmp的默認(rèn)路徑是哪里。
既然找到了upload_dir_tmp的默認(rèn)路徑了,那么修改c:\windows\temp的訪(fǎng)問(wèn)權(quán)限,賦予IUSR_用戶(hù)可寫(xiě),重啟動(dòng)IIS Admin服務(wù),上傳文件,終于成功了。這就是為什么我的多臺(tái)服務(wù)器upload_dir_tmp的值都為空的時(shí)候有的可傳,有的不可傳的原因。
現(xiàn)在已經(jīng)修改了IIS使用環(huán)境變量方式精簡(jiǎn)配置php的那篇文章了,因?yàn)楫?dāng)時(shí)沒(méi)有注意這個(gè)位置的權(quán)限設(shè)置,造成了如今的問(wèn)題,不過(guò)最終解決也是好的。