第十九章 Nest multer 文件上传

上章我们了解了Express multer 文件上传的相关操作 本章将了解Nest中的文件上传。用 multer 包处理 multipart/form-data 类型的请求中的 file

新建个 nest 项目:

nest new nest-multer-upload 

1719665836394.png
安装 multer 的 ts 类型的包:

npm install -D @types/multer

1719667026173.png

1、单文件上传

接着创建一个接口,用于文件上传 下面的test1接口就是用于文件上传

import { Body, Controller, Get, Post, UploadedFile, UseInterceptors } from '@nestjs/common';
import { AppService } from './app.service';
import { FileInterceptor } from '@nestjs/platform-express';@Controller()
export class AppController {constructor(private readonly appService: AppService) { }@Get()getHello(): string {return this.appService.getHello();}@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile() file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}
}

使用 FileInterceptor 来提取 file 字段,然后通过 UploadedFile 装饰器把它作为参数传入
接着运行项目可以看到我们的项目根目录下创建了一个uploads名称的文件夹
1719668539032.png
在nestjs项目中设置支持跨域
修改main.ts:

import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';async function bootstrap() {const app = await NestFactory.create(AppModule, {cors: true});await app.listen(3000);
}
bootstrap();

接着 我们来编写前端的代码,创建index.html

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('file', fileInput.files[0]);const res = await axios.post('http://localhost:3000/test1', data);console.log(res);}fileInput.onchange = formData;</script>
</body></html>

运行前端:

npx http-server

1719717782702.png
浏览器访问:
1719718137459.png
上传文件之后可以看控制台看到打印了上传的file 对象 和body数据,文件也保存到了 uploads 目录下
1719718212331.png

2、多文件上传

增加test2接口 注意哦 FilesInterceptor UploadedFiles 是加了s的 和之前的单文件上传不一样

@Post('test2')@UseInterceptors(FilesInterceptor('files', 3, {dest: 'uploads',}))uploadFiles(@UploadedFiles() files: Express.Multer.File[], @Body() body: any) {console.log('body', body);console.log('files', files);}

接着我们在前端代码新增一个函数 formData2:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('file', fileInput.files[0]);const res = await axios.post('http://localhost:3000/test1', data);console.log(res);}async function formData2() {const data = new FormData();data.set('name', '测试');data.set('age', 36);[...fileInput.files].forEach(item => {data.append('file', item);})const res = await axios.post('http://localhost:3000/test2', data, {Headers: { 'content-type': 'multipart/form-data' }});console.log(res);}fileInput.onchange = formData2;</script>
</body></html>

重新运行前端:

npx http-server

接着上传多个文件之后 可以看到控制台打印了 files 文件数组
1719719733484.png

3、多字段上传

接下来我们测试一下多字段上传
新增加 test3接口 使用 FileFieldsInterceptor UploadedFiles

import { Body, Controller, Get, Post, UploadedFile, UploadedFiles, UseInterceptors } from '@nestjs/common';
import { AppService } from './app.service';
import { FileFieldsInterceptor, FileInterceptor, FilesInterceptor } from '@nestjs/platform-express';@Controller()
export class AppController {constructor(private readonly appService: AppService) { }@Get()getHello(): string {return this.appService.getHello();}@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile() file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}@Post('test2')@UseInterceptors(FilesInterceptor('files', 3, {dest: 'uploads',}))uploadFiles(@UploadedFiles() files: Express.Multer.File[], @Body() body: any) {console.log('body', body);console.log('files', files);}@Post('test3')@UseInterceptors(FileFieldsInterceptor([{ name: 'a', maxCount: 2 },{ name: 'b', maxCount: 3 },]))uploadFileFields(@UploadedFiles() files: { aaa?: Express.Multer.File[], bbb?: Express.Multer.File[] }, @Body() body) {console.log('body', body);console.log('files', files);}
}

修改前端代码 增加 formData3 实现上传4个文件 前两个文件上传的时候在a字段 后两个文件上传的时候在b字段实现上传:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('file', fileInput.files[0]);const res = await axios.post('http://localhost:3000/test1', data);console.log(res);}async function formData2() {const data = new FormData();data.set('name', '测试');data.set('age', 36);[...fileInput.files].forEach(item => {data.append('files', item);})const res = await axios.post('http://localhost:3000/test2', data, {Headers: { 'content-type': 'multipart/form-data' }});console.log(res);}async function formData3() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.append('a', fileInput.files[0])data.append('a', fileInput.files[1])data.append('b', fileInput.files[2])data.append('b', fileInput.files[3])const res = await axios.post('http://localhost:3000/test3', data);console.log(res);}fileInput.onchange = formData3;</script>
</body></html>

重新运行前端:

npx http-server

接着打开 http://127.0.0.1:8080/ 上传4个文件
1719723208742.png
可以看到 a 字段收到了2个文件 b字段收到了2个文件

4、任意字段上传

如果我们不知道有哪些字段是上传的 可以使用AnyFilesInterceptor,新建接口test4:

@Post('test4')@UseInterceptors(AnyFilesInterceptor({dest: 'uploads',}))uploadAnyFiles(@UploadedFiles() files: Array<Express.Multer.File>, @Body() body) {console.log('body', body);console.log('files', files);}

修改前端代码 增加 formData4 函数:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('file', fileInput.files[0]);const res = await axios.post('http://localhost:3000/test1', data);console.log(res);}async function formData2() {const data = new FormData();data.set('name', '测试');data.set('age', 36);[...fileInput.files].forEach(item => {data.append('files', item);})const res = await axios.post('http://localhost:3000/test2', data, {Headers: { 'content-type': 'multipart/form-data' }});console.log(res);}async function formData3() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.append('a', fileInput.files[0])data.append('a', fileInput.files[1])data.append('b', fileInput.files[2])data.append('b', fileInput.files[3])const res = await axios.post('http://localhost:3000/test3', data);console.log(res);}async function formData4() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('aaa', fileInput.files[0]);data.set('bbb', fileInput.files[1]);data.set('ccc', fileInput.files[2]);data.set('ddd', fileInput.files[3]);const res = await axios.post('http://localhost:3000/test4', data);console.log(res);}fileInput.onchange = formData4;</script>
</body></html>

重新运行前端:

npx http-server

接着打开 http://127.0.0.1:8080/ 上传4个文件 可以看到识别了所有字段
1719724483848.png
同时也可以指定storage,转换上传的文件名称 创建 storage.ts

import * as multer from "multer";
import * as fs from 'fs';
import * as path from "path";const storage = multer.diskStorage({destination: function (req, file, cb) {try {fs.mkdirSync(path.join(process.cwd(), 'my-uploads'));}catch(e) {}cb(null, path.join(process.cwd(), 'my-uploads'))},filename: function (req, file, cb) {const uniqueSuffix = Date.now() + '-' + Math.round(Math.random() * 1E9) + '-' + file.originalnamecb(null, file.fieldname + '-' + uniqueSuffix)}
});export { storage };

修改之前的test4接口:

import { Body, Controller, Get, Post, UploadedFile, UploadedFiles, UseInterceptors } from '@nestjs/common';
import { AppService } from './app.service';
import { AnyFilesInterceptor, FileFieldsInterceptor, FileInterceptor, FilesInterceptor } from '@nestjs/platform-express';
import { storage } from './storage ';@Controller()
export class AppController {constructor(private readonly appService: AppService) { }@Get()getHello(): string {return this.appService.getHello();}@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile() file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}@Post('test2')@UseInterceptors(FilesInterceptor('files', 3, {dest: 'uploads',}))uploadFiles(@UploadedFiles() files: Express.Multer.File[], @Body() body: any) {console.log('body', body);console.log('files', files);}@Post('test3')@UseInterceptors(FileFieldsInterceptor([{ name: 'a', maxCount: 2 },{ name: 'b', maxCount: 3 },]))uploadFileFields(@UploadedFiles() files: { aaa?: Express.Multer.File[], bbb?: Express.Multer.File[] }, @Body() body) {console.log('body', body);console.log('files', files);}@Post('test4')@UseInterceptors(AnyFilesInterceptor({dest: 'uploads',storage: storage}))uploadAnyFiles(@UploadedFiles() files: Array<Express.Multer.File>, @Body() body) {console.log('body', body);console.log('files', files);}}

再次打开游览器 http://127.0.0.1:8080/ 上传4个文件
1719730782749.png

5、文件上传限制

接下来我们对上传的文件进行显示 文件大小、类型等,这部分我们在pipe里实现

nest g pipe file-size-validation-pipe --no-spec --flat

1719730968608.png
修改ize-validation-pipe.pipe.ts 设置文件要小于20kb

import { PipeTransform, Injectable, ArgumentMetadata, HttpException, HttpStatus } from '@nestjs/common';@Injectable()
export class FileSizeValidationPipe implements PipeTransform {transform(value: Express.Multer.File, metadata: ArgumentMetadata) {if(value.size > 20 * 1024) {throw new HttpException('文件大于 20k', HttpStatus.BAD_REQUEST);}return value;}
}

添加到接口里 FileSizeValidationPipe:

@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile(FileSizeValidationPipe) file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}

接着把前端代码修改一下

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('file', fileInput.files[0]);const res = await axios.post('http://localhost:3000/test1', data);console.log(res);}async function formData2() {const data = new FormData();data.set('name', '测试');data.set('age', 36);[...fileInput.files].forEach(item => {data.append('files', item);})const res = await axios.post('http://localhost:3000/test2', data, {Headers: { 'content-type': 'multipart/form-data' }});console.log(res);}async function formData3() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.append('a', fileInput.files[0])data.append('a', fileInput.files[1])data.append('b', fileInput.files[2])data.append('b', fileInput.files[3])const res = await axios.post('http://localhost:3000/test3', data);console.log(res);}async function formData4() {const data = new FormData();data.set('name', '测试');data.set('age', 36);data.set('aaa', fileInput.files[0]);data.set('bbb', fileInput.files[1]);data.set('ccc', fileInput.files[2]);data.set('ddd', fileInput.files[3]);const res = await axios.post('http://localhost:3000/test4', data);console.log(res);}fileInput.onchange = formData;</script>
</body></html>

重新运行前端: 上传文件一张大于20kb的图片或文件

npx http-server

可以发现接口报错 这样就可以实现文件的校验
1719731591999.png
其实 Nest 内置了文件大小、类型的校验 接下来我们测试一下:
修改test1接口 使用Nest 内置的 ParseFilePipe,它的作用是调用传入的 validator 来对文件做校验。
比如 MaxFileSizeValidator 是校验文件大小、FileTypeValidator 是校验文件类型。

@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile(new ParseFilePipe({validators: [new MaxFileSizeValidator({ maxSize: 1000 }),new FileTypeValidator({ fileType: 'image/jpeg' })]})) file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}

接下来我们再上传文件测试一下 可以看到返回了400和具体错误信息
1719731967044.png
接下来我们自定义一下返回的错误信息: 使用 exceptionFactory 自定义错误信息

@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile(new ParseFilePipe({validators: [new MaxFileSizeValidator({ maxSize: 1000 }),new FileTypeValidator({ fileType: 'image/jpeg' })],exceptionFactory(error) {throw new HttpException('测试' + error, 400);},})) file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}

再次上传可以看到返回了自定义的错误信息
1719732165228.png
我们也可以自己实现Validator
创建 my-file-validator.ts

import { FileValidator } from "@nestjs/common";export class MyFileValidator extends FileValidator{constructor(options) {super(options);}isValid(file: Express.Multer.File): boolean | Promise<boolean> {if(file.size > 10000) {return false;}return true;}buildErrorMessage(file: Express.Multer.File): string {return `文件 ${file.originalname} 大小超出 10k`;}
}

修改接口使用自定义的validator:

@Post('test1')@UseInterceptors(FileInterceptor('file', {dest: 'uploads',}))uploadFile(@UploadedFile(new ParseFilePipe({validators: [new MyFileValidator({}),],exceptionFactory(error) {throw new HttpException('测试' + error, 400);},})) file: Express.Multer.File, @Body() body: any) {console.log('body', body);console.log('file', file);}

上传文件:
1719732568683.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/870332.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

进阶版智能家居系统Demo[C#]:整合AI和自动化

引言 在基础智能家居系统的基础上&#xff0c;我们将引入更多高级功能&#xff0c;包括AI驱动的自动化控制、数据分析和预测。这些进阶功能将使智能家居系统更加智能和高效。 目录 高级智能家居功能概述使用C#和AI实现智能家居自动化实现智能照明系统的高级功能 自动调节亮度…

Linux C语言基础 day7

目录 思维导图&#xff1a; 学习目标&#xff1a; 学习内容&#xff1a; 1. 数组 1.1 对数组元素的常规操作 1.1.1 逆序 1.1.2 挑选数据 1.1.3 排序 1. 冒泡排序 2. 选择排序 2. 二维数组 2.1 二维数组的概念 2.1.1. 定义格式 2.2.2.初始化 2.2 二维数组的相关操…

2.4G芯片开发的遥控玩具方案介绍 东莞酷得

玩具从早期的简单功能&#xff0c;到现如今各种各样的智能操作&#xff0c;发展的速度也是飞速的。随着玩具市场的逐步完善与推进&#xff0c;中国的智能玩具市场也出现了很多远程遥控玩具。遥控玩具也是从最初的有线到现在的无线&#xff0c;从地上跑的到天上飞的&#xff0c;…

Go 1.19 工具链升级:go命令与工具改进详解

Go 1.19 工具链升级&#xff1a;go命令与工具改进详解 1. 引言 1.1 Go 1.19 简介 Go 1.19 是 Go 语言的一次重大更新&#xff0c;它带来了许多新特性和改进&#xff0c;特别是在工具链方面。 1.2 工具链的重要性 工具链是任何编程语言生态中的重要组成部分&#xff0c;它直…

编程语言一般学几种语言:探索编程语言的广度与深度

编程语言一般学几种语言&#xff1a;探索编程语言的广度与深度 在编程的广阔领域中&#xff0c;编程语言的选择和学习是每位初学者和进阶者都需要面对的问题。那么&#xff0c;一般应该学习几种编程语言呢&#xff1f;这个问题看似简单&#xff0c;实则充满了困惑和深度。接下…

Wireshark 对 https 请求抓包并展示为明文

文章目录 1、目标2、环境准备3、Wireshark 基本使用4、操作步骤4.1、彻底关闭 Chrome 进程4.2、配置 SSLKEYLOGFILE [核心步骤]4.3、把文件路径配置到 Wireshark 指定位置4.4、在浏览器发起请求4.5、抓包配置4.6、过滤4.6.1、过滤域名 http.host contains "baidu.com4.6.2…

UNI_App平台调试指南 debug(十五)

App平台调试指南 debug 常规开发里,在 HBuilderX 的运行菜单里运行 App,手机端的错误或 console.log 日志信息会直接打印到控制台。 如果需要更多功能,比如审查元素、打断点 debug,则需要启动调试模式。自 HBuilderX 2.0.3+ 版本起开始支持 App 端的调试。 #打开调试窗口…

响应式建站公司企业官网源码系统 带源代码以及搭建部署教程

系统概述 响应式建站公司企业官网源码系统是一套集设计、开发、部署于一体的综合性解决方案。它旨在为企业提供一个易于定制、功能强大、适应各种设备屏幕的官方网站平台。 该系统采用先进的技术架构&#xff0c;确保网站的稳定性和性能。它能够与各种后端数据库和服务器环境…

python找因子

【问题描述】 输入一个大于1的整数&#xff0c;返回一个列表&#xff0c;包含所有能够整除该整数的因子&#xff08;不包含1和它本身&#xff09;&#xff0c;并且从小到大排序。如果这个数是素数&#xff0c;则输出“(整数) is prime”。 【样例输入】 number:6 【样例输出…

TCP四次挥手:为什么四次?原理大揭密!

我是小米,一个喜欢分享技术的29岁程序员。如果你喜欢我的文章,欢迎关注我的微信公众号“软件求生”,获取更多技术干货! Hello, 大家好,我是你们的技术小伙伴小米!今天我们来聊一聊网络基础中的一个重要环节——TCP四次挥手过程。大家都知道,TCP连接的建立和断开是网络通…

2024年10款免费的项目管理软件推荐

本文向大家推荐10款2024年免费使用的项目管理软件&#xff0c;其中包括桌面应用和基于Web平台的多种产品&#xff0c;同时还涵盖了一些优秀的开源软件。 1.禅道开源项目管理软件 禅道是一款开源的、基于Web的项目管理软件&#xff0c;其功能丰富且操作简便&#xff0c;为团队提…

孟加拉最受欢迎的slot游戏推广okspin海外网盟广告优势

孟加拉最受欢迎的slot游戏推广okspin海外网盟广告优势 在当今全球化日益加剧的时代&#xff0c;游戏产业正迎来前所未有的发展机遇。孟加拉国&#xff0c;作为一个充满活力和潜力的新兴市场&#xff0c;其游戏出海之路也愈发受到业界的关注。在这一过程中&#xff0c;广告投放…

C++入门——命名空间与输入输出与缺省参数与重载函数与引用与内联

文章目录 命名空间——namespace命名空间的用处命名空间的定义命名空间的使用命名空间的嵌套命名空间的别名 输入与输出原理概述输入输出的使用 缺省参数定义缺省参数的方式使用缺省参数的价值和优势 函数重载定义与使用价值与优势 引用定义与使用价值与优势注意事项常量引用函…

【TOOLS】Chrome扩展开发

Chrome Extension Development 1. 入门教程 入门案例&#xff0c;可以访问【 谷歌插件官网官方文档 】查看官方入门教程&#xff0c;这里主要讲解大概步骤 Chrome Extenson 没有固定的脚手架&#xff0c;所以项目的搭建需要根据开发者自己根据需求搭建项目&#xff08;例如通过…

2021 RoboCom 世界机器人开发者大赛-本科组(复赛):拼题A打卡奖励

拼题 A 的教超搞打卡活动&#xff0c;指定了 N 张打卡卷&#xff0c;第 i 张打卡卷需要 mi​ 分钟做完&#xff0c;完成后可获得 ci​ 枚奖励的金币。活动规定每张打卡卷最多只能做一次&#xff0c;并且不允许提前交卷。活动总时长为 M 分钟。请你算出最多可以赢得多少枚金币&a…

生物素-十一聚乙二醇-沙利度胺;Biotin-PEG11-Thalidomide

Biotin-PEG11-Thalidomide&#xff0c;即生物素-十一聚乙二醇-沙利度胺&#xff0c;是一种结合了生物素、十一聚乙二醇&#xff08;PEG11&#xff09;和沙利度胺的复杂化合物。以下是对该化合物的详细分析&#xff1a; 一、组成成分及特性 生物素&#xff08;Biotin&#xff09…

美间·AI创意商拍——面向全球电商从业者提供AI背景图、AI真实增强、AI智能抠图、AI扩图、AI智能消除等AI生成及设计工具

一、产品介绍 「美间AI创意商拍」是群核科技推出的电商AIGC设计平台&#xff0c;依托群核前沿技术研究院自研大模型和美间2D智能设计引擎&#xff0c;面向全球电商从业者提供AI背景图、AI真实增强、AI智能抠图、AI扩图、AI智能消除等AI生成及设计工具&#xff0c;帮助用户高效…

odoo 继承原生domain进行修改

有需求是 编辑条件,进行对条件的路由拼接,进行展示对应报表 在原生的 domain widget 效果不是很理想,并想展示只是编辑条件以及展示条件,并不能满足。所以继承开发并另命名widget 代码如下: js 修改 在原有代码基础上复制进行修改 export class XCDomainField extends Co…

3ds Max 软件介绍基本操作方法

3ds Max 是一款功能强大的三维建模、动画和渲染软件&#xff0c;广泛应用于游戏开发、影视特效、建筑可视化、工业设计等领域。 功能特点&#xff1a; 强大的建模工具 提供了多种建模方式&#xff0c;包括多边形建模、曲面建模、数字雕刻等&#xff0c;能够创建各种复杂的三维模…

薄冰英语语法学习--代词1

人称代词&#xff1a; 做主语。 你、我、他、她、它、你们、他们、我们 you、I、he、she、it、you、they、we 做宾语 you、me、him、her、it、you、them、us 形容词性&#xff1a;物主代词 我的美貌&#xff0c;你的丑陋&#xff0c;她的悲伤&#xff0c;他们的可爱&…