安裝moodle 1.8.2(OS and Web Server: FreeBSD 6.2 and Apache22, PHP5.1.6)上傳中文檔名的檔案後無法正確顯示中文名問題,先到網路搜尋解決方法,在這幾篇看到修正方法:
1.http://moodle.club.tw/moodle/mod/forum/discuss.php?d=66&parent=163
2.http://plog.hlps.tcc.edu.tw/plog/post/1/2945
以下是個人的說明與提出一些修正的看法
於config.php加入
$CFG->unicodecleanfilename = true;
即可正常上傳中文檔名
在lib/moodlelib.php下亦有這段說明
/**
* Cleans a given filename by removing suspicious or troublesome characters
* Only these are allowed: alphanumeric _ - .
* Unicode characters can be enabled by setting $CFG->unicodecleanfilename = true in config.php
*
* WARNING: unicode characters may not be compatible with zip compression in backup/restore,
* because native zip binaries do weird character conversions. Use PHP zipping instead.
*
* @param string $string file name
* @return string cleaned file name
*/
因此似乎雖解決上傳中文檔名問題但zip壓縮卻無法使用中文檔名,以moodle 內建pclzip也許可解決???
以下有兩個問題待解決:
- 下載中文檔名於IE下有問題,會轉成英文檔名,Firefox可正常下載
- 無論在firefox或IE均無法壓縮與解壓縮中文檔名(備份亦然)
產生此問題原因分析:
- IE6對於UTF-8中文檔名無法正常下載,IE7則只對zip的中文檔會產生問題其餘對office或wmv,gif,jpg不會產生問題,所以對於若是IE瀏覽器下載時將檔名轉成Big5應可解決。
- 壓縮與解壓縮問題,原因在於php的basename() function對於UTF-8對中文檔名會回傳成無中文檔名或不正確之檔名(IE6與IE7不一樣,IE7較接近Frefox),改寫basename function應可解決。不知道其他php版本會不會這樣???沒試過。
文章最底下有一小段程式碼支持上述說法。
==================================
詳細改寫程式碼,請參閱以下說明:
1. 下載中文檔名於IE下有問題,Firefox可正常下載
雖然註解lib/file.php下的這一行
//$filename = $args[count($args)-1];
send_file($pathname, $filename, $lifetime, $CFG->filteruploadedfiles, false, $forcedownload);
可解決下載問題,但使用 //$filename = $args[count($args)-1];
則造成下一行程式call send_file(....)的filename參數為null。
trace send_file function(在lib/filelib.php檔案中)以filename無法正確取得mimetype,不知會不會造成隱含的問題?
經過簡化後測試(參閱最下面程式碼)發現應該是以@header('Content-Disposition: attachment; filename='.$filename);
送出UTF-8中文檔名造成IE錯誤(與IE是否設定以UTF-8傳送URL無關)
因此做了以下修改
1. file.php不註解 $filename = $args[count($args)-1];
2. lib/filelib.php加入一小段程式碼:
header('Last-Modified: '. gmdate('D, d M Y H:i:s', $lastmodified) .' GMT');
if (check_browser_version()){
$filename=iconv("UTF-8","Big5",$filename);
}
if ($forcedownload) {
@header('Content-Disposition: attachment; filename='.$filename);
} else {
@header('Content-Disposition: inline; filename='.$filename);
}
紅色部份是加入的程式碼,check_browser_version()是在lib/moodlelib.php內的function用來檢查用戶端使用何種瀏覽器
2. php之basename() 內建function無法正確傳回中文檔名,造成壓縮與解壓縮問題:
於lib/moodlelib.php加一function 來取代原來的basename
function file_basename($file= null) {
if($file=== null || strlen($file)<= 0) {
return null;
}
$file= explode('?', $file);
$file= explode('/', $file[0]);
$basename= $file[count($file)-1];
return $basename;
}
因此需修改的程式碼部份說明如下:
A. files/index.php修改如下
在case "paste": 修改
$shortfile = basename($file);->$shortfile = file_basename($file);
在case "unzip": 修改
$file = basename($file);->$file = file_basename($file);
在case "listzip": 修改
$file = file_basename($file);->$file = file_basename($file);
約825行處$fileurl改成$filesafe(fileurl為未編碼的檔名,filesafe為編碼的檔名), 則不管IE6或 IE7均能正常取得正確的檔名
else if ($icon == "zip.gif") {
$edittext .= "<a href=\"index.php?id=$id&wdir=$wdir&file=$filesafe&action=unzip&sesskey=$USER->sesskey&choose=$choose\">$strunzip</a> ";
//
$edittext .= "<a href=\"index.php?id=$id&wdir=$wdir&file=$filesafe&action=listzip&sesskey=$USER->sesskey&choose=$choose\">$strlist</a> ";
B. 修改lib/moodlelib.php
在lib/moodlelib.php
修改function zip_files(..)
//$destfilename = $path_parts["basename"]; //The name of the zip file
$destfilename = file_basename($destination);
//$filestozip .= escapeshellarg(basename($filetozip));
$filestozip .= escapeshellarg(file_basename($filetozip));
修改 function unzip_file(...)
//$zipfilename = $path_parts["basename"]; //The name of the zip file
$zipfilename = file_basename($zipfile);
經過以上修改無論在Firefox, IE6 or IE7均能正常使用中文檔名了。
簡化測試程式碼
--------------------------------------------
<?
//echo basename("/usr/local/test/中文測試檔.zip");
$filename=iconv("UTF-8","Big5","中文測試檔.zip");
//$filename="中文測試檔.zip";
@header('Content-Disposition: attachment; filename='.$filename);
@header('Content-Type: application/zip');
readfile_chunked("中文測試檔.zip");
//readfile_chunked 為lib/filelib.php內的function
function readfile_chunked($filename, $retbytes=true) {
$chunksize = 1*(1024*1024); // 1MB chunks - must be less than 2MB!
$buffer = '';
$cnt =0;// $handle = fopen($filename, 'rb');
$handle = fopen($filename, 'rb');
if ($handle === false) {
return false;
}
while (!feof($handle)) {
@set_time_limit(60*60); //reset time limit to 60 min - should be enough for 1 MB chunk
$buffer = fread($handle, $chunksize);
echo $buffer;
flush();
if ($retbytes) {
$cnt += strlen($buffer);
}
}
$status = fclose($handle);
if ($retbytes && $status) {
return $cnt; // return num. bytes delivered like readfile() does.
}
return $status;
}
?>
謝謝你的資訊,我試用過沒問題喔。
太感激了!
- 測試PHP 5.2.4以上版本basename已可正常處理中文,因此只要系統PHP版本更新至5.2.4以上,則以上所有basename函式無須取代。
- IE7處理UTF8中文問題似乎已解決,只要把IE6升級至IE7上述問題無須修改程式碼亦可解決。
- zip壓縮問題:在moodle壓縮中文檔名的zip檔無法在windows下正常解壓縮 ; 反之在windows下壓縮的zip檔,在moodle下解壓縮中文檔名出現亂碼(其實是Big5碼) 因為在moodle下之中文檔名為UTF-8而在windows下為Big5碼。
更新1.8.4後發現lib/filelib.php增加程式片斷
if (check_browser_version('MSIE')){
$filename=urlencode($filename);
}
因此本文章原加入
if (check_browser_version()){
$filename=iconv("UTF-8","Big5",$filename);
}
不用再加入
我現在是Apache 2.2.4 +php5.2.5+MySQL 5.0.45
有加
$CFG->unicodecleanfilename = true;
還有哪邊需要改的嗎
約825行處$fileurl改成$filesafe(fileurl為未編碼的檔名,filesafe為編碼的檔名), 則不管IE6或 IE7均能正常取得正確的檔名
else if ($icon == "zip.gif") {
$edittext .= "sesskey&choose=$choose\">$strunzip ";
//
$edittext .= "sesskey&choose=$choose\">$strlist ";
這裡有需要改嗎
謝謝
針對用戶端是IE6
約825行處$fileurl仍需改成$filesafe
因為:
$fileurl與$filesafe的關係為
$filesafe=urlencode($fileurl);
你好我加了$CFG->unicodecleanfilename = true;
上傳中文時會出現
缺乏檔案 - 您確定選擇了上傳檔案了嗎?
備份檔案亦出現
PCLZIP_ERR_READ_OPEN_FAIL (-2) : Unable to open archive 'C:/AppServ/moodledata/temp/backup/1203645681/備份檔-cf101_1-20080222- 1001.zip' in wb mode
請教
缺乏檔案 - 您確定選擇了上傳檔案了嗎
也無法壓縮中文檔
謝謝
我想應該是Windows XP系統儲存中文檔名仍是以Big5存檔(Windows 2003與vista是否也是這樣沒試過),我修改一下lib/uploadlib.php第233行:
原來是:
if (move_uploaded_file($this->files[$i]['tmp_name'], $destination.'/'.$this->files[$i]['name'])) {
修改為 if (move_uploaded_file($this->files[$i]['tmp_name'], iconv("UTF-8","Big5",$destination.'/'.$this->files[$i]['name']))) { 把輸入的檔名改成Big5編碼則能順利上傳,當然瀏覽器編碼需改成Big5就能正確顯示中文檔名。
二, 另一種驗證的方法直接把中文檔案複製到moodledata下,然後進入moodle看該檔案顯示出亂碼,當然瀏覽器編碼改成Big5就能正確顯示中文檔名。
三, 建議將 OS 改成 unix 系統。
四, 是否有辦法將Windows系統設成同時認得Big5與UTF-8編碼的檔名?尚未得到答案!!!
老師
這樣的確能上傳
但如你所說編碼還是...
且要下載時會出現
抱歉,指定的檔案找不到
在課程備份時會出現
備份檔案亦出現
PCLZIP_ERR_READ_OPEN_FAIL (-2) : Unable to open archive 'C:/AppServ/moodledata/temp/backup/1203645681/備份檔-cf101_1-20080222- 1001.zip' in wb mode
無法完成
吳老師你好:
我是在XP下測試
主系統在CENTOS上
依你作法中文檔案可上傳
編碼還是.....
備份資料使用中文亦會有問題
是我在安裝SERVER 哪有設定錯嗎?
謝謝
CentOS版板的確稍舊
再CENTOS上 依你步驟尚未發現大問題
感謝你的指導
老師可否提供聯絡方式
日後請益交流之用
http://plog.ptes.tp.edu.tw/post/97/716
在上列文章迴響內容對新版1.8.4修正說明,只提高memory_limit即可。
將php.ini 中memory_limit調高到128M以上試試。




