2

ユーザーが DOC、DOCX、PDF、および TXT タイプのファイルをアップロードできるようにするためのスクリプトがあります。改行はありませんが、ファイルはアップロードされます。たとえば、TXT ファイルは次のようになります。

//////////
THIS IS A TEXT FILE
//////////

次のようにダウンロードされます。

//////////THIS IS A TEXT FILE//////////

現在、これはテキスト ファイルではそれほど「大きな」問題ではありませんが、この改行の損失により、PDF、DOC、および DOCX タイプが完全に破損します。

ここでも、メモ帳で表示された PDF は元は次のようになります。

%PDF-1.5
%µµµµ
1 0 obj
<</Type/Catalog/Pages 2 0 R/Lang(en-US) /StructTreeRoot 10 0 R/MarkInfo<</Marked true>>>>
endobj
2 0 obj
....

ただし、アップロードしてからダウンロードすると、一部の文字が欠落した 1 つの長い文字列のように見えます。

%PDF-1.5%µµµµ1 0 obj<>>>endobj2 0 obj

そのため、Adobe PDF リーダーでファイルを開くと、ファイルが破損していると表示されます。


これは、ファイルのアップロードに使用している PHP スクリプトの抜粋です。

$_POST['request_id'] = $_POST['rid'];
    $tmpName = $_FILES['exam_file']['tmp_name'];
    $_POST['name'] = $_FILES['exam_file']['name'];
    $_POST['type'] = $_FILES['exam_file']['type'];
    $_POST['size'] = $_FILES['exam_file']['size'];
    //File types allowed are PDF, DOC, DOCX, TXT
    if ( ( $_POST['type'] == "application/pdf") || ($_POST['type'] == "application/msword") || ($_POST['type'] == "application/vnd.openxmlformats-officedocument.wordprocessingml.document") || ($_POST['type'] == "text/plain") ){
        $fp = fopen($tmpName, 'r');
        $content = fread($fp, filesize($tmpName));
        fclose($fp);
        $file_id = $model -> addFiles($_POST, $content);    
    }

$model の関数 addFiles() は次のとおりです。

function addFiles($data, $file){
    $table = 'faculty_files';
    $data = parent::clean($data);
    $keys = array('request_id', 'name', 'type', 'size', 'content');
    $data = parent::cleanAndPick( $keys, $data );
    $data['content'] = $file;
    $data['faculty_user_id'] = $_SESSION['user_details']['id'];
    $data['inserted_on'] = date('Y-m-d H:i:s');
    parent::insert( $table, $data );
    return mysql_insert_id();
}

parent::insert()テーブル名やフィールドなどの関連するすべての詳細を取得し、それらを SQL ステートメントに入れるだけです。

これは関数parent::clean()ですが、$file実行されていません:

public function clean($data = NULL){
  $data = str_replace("\n", "", $data);
  $data = str_replace("\r", "", $data);
  $data = stripslashes($data);
  return mysql_real_escape_string($data);
}

関数parent::cleanAndPick()

public function cleanAndPick($keys, $data = NULL) {

    $clean = array();
    foreach ($keys as $key) {
        if (array_key_exists($key, $data))
            $clean[$key] = $this -> clean($data[$key]);
    }
    return $clean;

}

関数parent::insert()

protected function insert($table, $data ) {

    //Build sql
    $data = $this -> clean($data);
    $sql = "INSERT INTO " . $this -> clean($table);
    $sql .= " (" . implode(",", array_keys($data)) . ") VALUES (";
    foreach ($data as $value)
        $sql .= ($value == NULL ? "NULL," : "'" . $value . "',");
    $sql = substr($sql, 0, -1) . ")";
    return $this -> query($sql);

}

これは、ファイルをダウンロードするために使用している PHP スクリプトです。

$file = $model -> retrieveFile($_POST['fid']);
header("Content-length: " . $file[0]['size'] . "");
header("Content-type: " . $file[0]['type'] . "");
header("Content-Disposition:attachment; filename=" . $file[0]['name'] . "");
print $file[0]['content'];

この問題の原因がわかりません。ご意見をいただければ幸いです。

編集: 誰かがPHP フォーラムで同様の問題を抱えていました。抜粋:

Linux サーバーに Mac OS ファイルをアップロードして読み取ろうとして失敗しました。多くのレコードは、次のみを使用して 1 つの大きなものとして表示されます。

<?php $fhandle = fopen($file, 'r'); ?>    or  <?php $fhandle =
> fopen($file, 'rb'); ?> 

ただし、次のように機能します。

> <?php  ini_set('auto_detect_line_endings', TRUE);  $fhandle =
> fopen($file, 'r');  ?>

私はこれを試しましたが、うまくいきませんでした。

4

1 に答える 1

1

fopen($tmpName, 'rb');その内容を変更したくない場合に使用します。

そして、これ:

protected function insert($table, $data ) {
  //Build sql
  $data = $this -> clean($data); //<---- here......
于 2013-03-05T16:12:41.307 に答える