nestjs上传文件multer详解

前言

最近在开发一个商城项目, 后端使用的nestjs+mongodb, 上传文件在一个系统里肯定是必不可少的。multer是官方推荐的上传文件方案, 虽然官方文档写的很详细, 但实战中还是遇到了一些坑, 在此码文记录一下, 希望对你快速上手有所帮助。

memory or disk?

multer假如没有任何配置默认就是储存在内存中, 上传文件的数据结构大概是这样

js 复制代码
{
  fieldname: 'file',
  originalname: 'unnamed.png',
  encoding: '7bit',
  mimetype: 'image/png',
  buffer: <Buffer 89 50 4e 47 0d 0a 1a 0a 00 00 00 0d 49 48 44 52 00 00 00 c0 00 00 00 c0 08 03 00 00 00 65 02 9c 35 00 00 00 03 73 42 49 54 08 08 08 db e1 4f e0 00 00 ... 10782 more bytes>,
  size: 10832
}

一般来说, 不涉及到文件转换的逻辑不会用到这个, 硬盘存储应该是我们更常见的需求, 我们需要额外配置一些选项

js 复制代码
import { MulterModule } from '@nestjs/platform-express';

MulterModule.register({
  dest: './public', // 配置文件硬盘存储的地址, destination 指向项目根目录
});

硬盘存储上传文件数据格式这样的

js 复制代码
{
  fieldname: 'file', 
  originalname: 'unnamed.png',             //原文件名 
  encoding: '7bit',                        
  mimetype: 'image/png',
  destination: './public',                 //文件的储存的文件夹位置
  filename: '1704774383763-unnamed.png',   //为了不重名, 上传后的文件名中加了一个时间戳
  path: 'public/1704774383763-unnamed.png', 
  size: 10832                              //文件大小
}

硬盘存储的配置(三种维度)

  1. 全局配置: 假如项目比较小, 所有文件只需要塞一个文件夹里即可的话, 全局无疑是最佳选择
js 复制代码
import { Module, Global } from '@nestjs/common';
import { AppController } from './app.controller';
import { AppService } from './app.service';
import { diskStorage } from 'multer';
import { MulterModule } from '@nestjs/platform-express';  // nest底层就是express, 而express又内置multer, 所以不需要再安装, 只需要安装一下类型即可npm i -D @types/multer

@Global()  // 全局导出的decoration
@Module({
  imports: [
    MulterModule.register({
      storage: diskStorage({
        destination: './public',
        filename: (req, file, cb) => {
          return cb(null, `${Date.now()}-${file.originalname}`);
        },
      }),
    }),
  ],
  controllers: [AppController],
  providers: [],
  exports: [MulterModule],  // 导出MulterModule
})
export class AppModule {}

使用了@global就不需要一个个在子module中去import MulterModule这个模块了

  1. 模块配置: 子模块导入的代码基本和上述一致, 针对于每个模块存放的文件位置不一致, 可以每个module各自配置, 而且不需要exports

  2. 拦截器配置: 当然还有一种情况, 就是在同一个模块中,不同接口的存储位置也是不同的, 所以multer颗粒度可以精确到每个接口

js 复制代码
import { diskStorage } from 'multer';
import { FileInterceptor } from '@nestjs/platform-express';

@UseInterceptors(FileInterceptor('file', {
  storage: diskStorage({
    destination: './another_uploads',
    filename: (req, file, cb) => {
      cb(null, `${Date.now()}-${file.originalname}`);
    },
  }),
}))
uploadFile(@UploadedFile() file) {
  console.log(file);
}

nestjs 静态文件服务

上述我们讲了nestjs上传文件的代码, 上传之后必然是需要访问的, 这就涉及到了nestjs的静态文件服务, 也很简单,首先安装包

js 复制代码
npm install --save @nestjs/serve-static

然后在app.module.ts中导入即可

js 复制代码
import { Module, Global } from '@nestjs/common';
import { MulterModule } from '@nestjs/platform-express';
import { ServeStaticModule } from '@nestjs/serve-static';
import { diskStorage } from 'multer';
import { join } from 'path';

@Global()
@Module({
  imports: [
    MulterModule.register({
      storage: diskStorage({
        destination: './public',
        filename: (req, file, cb) => {
          return cb(null, `${Date.now()}-${file.originalname}`);
        },
      }),
    }),
    ServeStaticModule.forRoot({
      rootPath: join(__dirname, '..', 'public'),
    }),
  ],
  controllers: [AppController],
  providers: [],
  exports: [MulterModule],
})
export class AppModule {}

这样你上传的文件, 就可以通过your.domain.com/filename.su... 访问了

后序

最近打算做一个鞋类商城全栈项目,移动端flutter + react-native,管理后端arco + vue3 + mongo + nodejs, 感兴趣的朋友可以关注我。

相关推荐
汪子熙26 分钟前
Angular 服务器端应用 ng-state tag 的作用介绍
前端·javascript·angular.js
Envyᥫᩣ35 分钟前
《ASP.NET Web Forms 实现视频点赞功能的完整示例》
前端·asp.net·音视频·视频点赞
Мартин.5 小时前
[Meachines] [Easy] Sea WonderCMS-XSS-RCE+System Monitor 命令注入
前端·xss
昨天;明天。今天。6 小时前
案例-表白墙简单实现
前端·javascript·css
数云界6 小时前
如何在 DAX 中计算多个周期的移动平均线
java·服务器·前端
风清扬_jd6 小时前
Chromium 如何定义一个chrome.settingsPrivate接口给前端调用c++
前端·c++·chrome
安冬的码畜日常6 小时前
【玩转 JS 函数式编程_006】2.2 小试牛刀:用函数式编程(FP)实现事件只触发一次
开发语言·前端·javascript·函数式编程·tdd·fp·jasmine
ChinaDragonDreamer6 小时前
Vite:为什么选 Vite
前端
小御姐@stella6 小时前
Vue 之组件插槽Slot用法(组件间通信一种方式)
前端·javascript·vue.js
GISer_Jing6 小时前
【React】增量传输与渲染
前端·javascript·面试