使用Multer重命名上传的文件无法正常工作(Express.js)

18

我正在尝试使用Express.js和Multer从HTML表单上传文件。我已经成功将文件保存到所需位置(名为uploads的文件夹中)。

然而,我想在上传时重命名文件,因为默认情况下,Multer会给它一个奇怪的名称,如:

5257ee6b035926ca99923297c224a1bb

这可能是十六进制时间戳之类的东西,但我需要一个更明确的名称,以便稍后调用脚本。

我遵循了这里找到的解释,但它除了上传具有十六进制名称的文件外,什么都没有做。

此外,两个事件onFileUploadStartonFileUploadComplete似乎从未被触发,因为我在控制台中没有看到任何日志。

我正在使用两个单独的文件作为服务器和路由:

app.js

/**
 * Dependencies
 */

var express = require('express');
var path = require('path');
var logger = require('morgan');
var cookieParser = require('cookie-parser');
var bodyParser = require('body-parser');

/**
 * Importation of routes
 */
var routes = require('./routes/index');
var recog = require('./routes/recog');

/**
 * Express
 */
var app = express();

app.use(logger('dev'));
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({extended: false}));
app.use(cookieParser());
app.use(express.static(path.join(__dirname, 'public')));

// pour contrer les erreurs de cross domain
app.use(function (req, res, next) {

    // Website you wish to allow to connect
    res.setHeader('Access-Control-Allow-Origin', '*');

    // Request methods you wish to allow
    res.setHeader('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE');

    // Request headers you wish to allow
    res.setHeader('Access-Control-Allow-Headers', 'X-Requested-With,content-type');

    // Set to true if you need the website to include cookies in the requests sent
    // to the API (e.g. in case you use sessions)
    res.setHeader('Access-Control-Allow-Credentials', true);

    // Pass to next layer of middleware
    next();
});

/**
 * Routes
 */
app.use('/', routes);
app.use('/recog', recog);

module.exports = app;

recog.js

/**
 * Requirements
 */
var express = require('express');
var router = express.Router();
var multer = require('multer');
var uploads = multer({
    dest: 'uploads/',
    rename: function (fieldname, filename) {
        console.log("Rename...");
        return filename + Date.now();
    },
    onFileUploadStart: function () {
        console.log("Upload is starting...");
    },
    onFileUploadComplete: function () {
        console.log("File uploaded");
    }
});

/**
 * Upload d'une image
 */
router.post('/upload', uploads.single('image'), function (req, res, next) {
    console.log("Front-end is calling");
    res.json({status: 'success', data: 'Fichier chargé.\nOrgane sélectionné : ' + req.body.organ});
});

module.exports = router;

我一直在搜索,但由于我对Node.js和JavaScript还比较陌生,所以无法找出问题所在。

感谢各位的帮助!

6个回答

28

Multer的用法已经改变。

现在Multer构造函数只接受三个选项:

  1. dist/storage
  2. fileFilter
  3. limits

现在,rename、onFileUploadStart和onFileUploadComplete将不起作用。

然而,可以使用DiskStorage来进行重命名操作。

var storage = multer.diskStorage({
    destination: function (req, file, cb) {
        cb(null, '/tmp/my-uploads')
    },
    filename: function (req, file, cb) {
        cb(null, file.fieldname + '-' + Date.now())
  }
})

var upload = multer({ storage: storage })

请查看以下链接:


13

我知道这篇文章有些过时了。我想为那些后来可能会到达的人做出贡献。下面是一个完全功能的服务器脚本,用于处理多个上传的图片,并具有随机保存图片名称和文件扩展名的功能。

var express = require("express");
var multer = require("multer");
var app = express();
var path = require("path");
var uuid = require("uuid");

// Allow cross origin resource sharing (CORS) within our application
app.use(function(req, res, next) {
    res.header("Access-Control-Allow-Origin", "*");
    res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
    next();
});

var storage = multer.diskStorage({
  destination: function (req, file, cb) {
    cb(null, 'uploadedimages/')
  },
  filename: function (req, file, cb) {
    cb(null, uuid.v4() + path.extname(file.originalname));
  }
})

var upload = multer({ storage: storage })

// "files" should be the same name as what's coming from the field name on the client side.
app.post("/upload", upload.array("files", 12), function(req, res) {
    res.send(req.files);
    console.log("files = ", req.files);
});

var server = app.listen(3000, function() {
    console.log("Listening on port %s...", server.address().port);
});

destination can also simply be a string when a dynamic upload directory isn't needed. e.i. destination: "uploadedImages/" instead of destination: function (... - Justin Taddei

5

尝试使用我正在使用的方法

  var storage = multer.diskStorage({
    destination: function (req, file, cb) {
      cb(null, 'uploads/')
    },
    filename: function (req, file, cb) {
      console.log(file);
      var fileObj = {
        "image/png": ".png",
        "image/jpeg": ".jpeg",
        "image/jpg": ".jpg"
      };
      if (fileObj[file.mimetype] == undefined) {
        cb(new Error("file format not valid"));
      } else {
        cb(null, file.fieldname + '-' + Date.now() + fileObj[file.mimetype])
      }
    }
  })

  var upload = multer({ storage: storage })

4
我们可以通过日期随机生成文件名,并使用file.mimetype来添加原始文件扩展名。尝试使用console.log(file.mimetype),您将获得由“/”分隔的文件名和扩展名,然后我们将其拆分为数组并从中获取扩展名。请尝试以下代码。
let storage = multer.diskStorage({
  destination: function (req, file, cb) {
    cb(null, './uploads')
  },
  filename: function (req, file, cb) {
    let extArray = file.mimetype.split("/");
    let extension = extArray[extArray.length - 1];
    cb(null, file.fieldname + '-' + Date.now()+ '.' +extension)
  }
})
const upload = multer({ storage: storage })

3
文件的结构如下所示:
{
"fieldname": "avatar",
"originalname": "somefile.pdf",
"encoding": "7bit",
"mimetype": "application/pdf",
"destination": "./uploads",
"filename": "36db44e11b83f4513188f649ff445a2f",
"path": "uploads\\36db44e11b83f4513188f649ff445a2f",
"size": 1277191

下面的示例会将文件保存为其原始名称和扩展名,而不是默认情况下的奇怪名称。 (您可以根据需要将其保存为“file.originalname”)

var storage = multer.diskStorage({
  destination: function (req, file, cb) {
    cb(null, './uploads') //Destination folder
  },
  filename: function (req, file, cb) {
    cb(null, file.originalname) //File name after saving
  }
})

var upload = multer({ storage: storage })

虽然这段代码片段可能解决了问题,但包括解释真的有助于提高您的帖子质量。请记住,您正在为未来的读者回答问题,而这些人可能不知道您的代码建议原因。 - Clijsters

2

个人而言,我实现了以下解决方案,为文件生成一个随机名称,并附加原始文件扩展名(假设我的扩展名在最后一个“.”之后)。

var path = require('path');

    var options = multer.diskStorage({ destination : 'uploads/' ,
      filename: function (req, file, cb) {
        cb(null, (Math.random().toString(36)+'00000000000000000').slice(2, 10) + Date.now() + path.extname(file.originalname));
      }
    });

    var upload= multer({ storage: options });

    router.post('/cards', upload.fields([{ name: 'file1', maxCount: 1 }, { name: 'file2', maxCount: 1 }]), function(req, res, next) {
    /*
      handle files here
      req.files['file1']; //First File
      req.files['file2']; //Second File
      req.body.fieldNames;//Other Fields in the form

    */
    });

MULTER 文档中,你会发现以下内容:
磁盘存储引擎可以让你完全控制文件的存储到磁盘上。
提供了两个选项:目标和文件名。它们都是用于确定文件应该存储在哪里的函数。
注意:如果将目标作为函数提供,则需要负责创建目录。如果传递字符串,则 multer 会确保为您创建目录。
文件名用于确定文件夹内的文件名称。如果没有给出文件名,则每个文件将被赋予一个不包含任何文件扩展名的随机名称。
注意:Multer 不会为您追加任何文件扩展名,您的函数应返回包含文件扩展名的文件名。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接