PHP文件上传到MySQL mediumblob的错误

时间:2013-03-05 16:06:18

标签: php mysql file-upload blob multipartform-data

我有一个脚本,允许用户上传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
....

然而,在上传然后下载之后,它看起来像一个长字符串,其中一些字符丢失了:

%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 forum上遇到了类似的问题。摘录:

  

我一直尝试上传和阅读Mac OS文件失败   一个Linux服务器。很多记录只显示了一个只使用了   以下:

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

然而,这确实有效:

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

我确实试过了这个并不适合我。

1 个答案:

答案 0 :(得分:1)

如果您不想对其内容进行任何更改,请使用fopen($tmpName, 'rb');

然后这个:

protected function insert($table, $data ) {
  //Build sql
  $data = $this -> clean($data); //<---- here......