经常使用swagger,可以通过设置[ProducesResponseType]标记接口的返回信息;swagger也能通过接口的参数列表,自动获得发送的数据结构信息。

不过有一个例外,就是上传文件的时候,设置了[Consumes]的内容为multi-part/form-data,但是swagger并不能正常感知是上传文件的。代码是这个样子的:

关于文件上传的细节,可以看多年前我写过一篇有关通过WEBAPI上传文件的文章

[Consumes("multipart/form-data")]
[ODataRoute]
[HttpPost]
public async Task<ActionResult> Post(IFormCollection collection)
{
var file = collection.Files[0];
if(file != null)
{
var filename = DateTime.Now.ToString("yyyyMMddHHmmss") + file.FileName;
var path = Path.Combine(_webHostEnvironment.WebRootPath, "Files", filename);
using FileStream fileStream = new FileStream(path, FileMode.Create);
await file.CopyToAsync(fileStream);
var uri = "Files/" + filename;
var fileEntity = new Models.File { Url = uri, LastModified = DateTime.Now };
_homeworkDataContext.Files.Add(fileEntity);
await _homeworkDataContext.SaveChangesAsync();
return Created(WebUtility.UrlEncode(uri), fileEntity);
}
return BadRequest();
}

实际上,swagger一直提示,上传的内容是一个array类型,当然API是没有问题的,可以通过POSTMAN进行发送,不过不能在网页上直接操作,总觉得心里有点不太舒服。

方法

搜索了一下办法,比较靠谱的,就是通过增加一个IOperationFilter来实现目的。

// CODE FROM https://www.talkingdotnet.com/how-to-upload-file-via-swagger-in-asp-net-core-web-api/
public class FileUploadOperation : IOperationFilter
{
public void Apply(Operation operation, OperationFilterContext context)
{
if (operation.OperationId.ToLower() == "apivaluesuploadpost")
{
operation.Parameters.Clear();
operation.Parameters.Add(new NonBodyParameter
{
Name = "uploadedFile",
In = "formData",
Description = "Upload File",
Required = true,
Type = "file"
});
operation.Consumes.Add("multipart/form-data");
}
}
}

然后,在services.ConfigureSwaggerGen()参数中,添加

options.OperationFilter<FileUploadOperation>();

方法的原理是通过重写操作某个特定API的的过滤器,来实现对返回内容的操作。

此方法适用于OAS2,实质上是实现了这里的规范要求。

我已经用上.NET 5.0了,自带了swagger都支持的是OpenAPI 3,这个方法不好用了。不过思想应该相同,首先看看OpenAPI 3的规范,文件上传需要定义为:

requestBody:
content:
multipart/form-data:
schema:
type: object
properties:
fileName:
type: string
format: binary

这个套路和OpenAPI 2完全不一样,需要重新设置requestBody才行。我们按照要求改造代码。

public class FileUploadOperation : IOperationFilter
{
public void Apply(OpenApiOperation operation, OperationFilterContext context)
{
//判断上传文件的类型,只有上传的类型是IFormCollection的才进行重写。
if (context.ApiDescription.ActionDescriptor.Parameters.Any(w => w.ParameterType == typeof(IFormCollection)))
{
Dictionary<string, OpenApiSchema> schema = new Dictionary<string, OpenApiSchema>();
schema["fileName"] = new OpenApiSchema { Description = "Select file", Type = "string", Format = "binary" };
Dictionary<string, OpenApiMediaType> content = new Dictionary<string, OpenApiMediaType>();
content["multipart/form-data"] = new OpenApiMediaType { Schema = new OpenApiSchema { Type = "object", Properties = schema } };
operation.RequestBody = new OpenApiRequestBody() { Content = content };
}
}
}

执行之后,swagger已经可以正常识别了,通过选择文件即可上传,效果如下:

参考资料

最新文章

  1. LoTVideo:只需两步,让HTML5原生态的Video茁壮成长
  2. Extjs.Button 按钮
  3. 2012 Multi-University #10
  4. command not found,系统很多命令都用不了 ,修改环境变量
  5. JS高程2.在HTML中使用Javascript(1)
  6. PHPstorm设置连接FTP,进行文件上传、下载、比较
  7. 【转】JAVA 接口
  8. 【转载】Linux i386+源码中常见宏标识tag的定义
  9. 夺命雷公狗---TP商城----TP之配置环境---1
  10. 跨平台base64数据传输注意问题
  11. [置顶] 博客已迁移至ryantang.me
  12. bzoj2209 2329
  13. Bozo排序
  14. Cocos2dx 学习笔记整理----在项目中使用图片(三)
  15. 实验八 Web基础 SQL注入原理
  16. numpy中的广播机制
  17. CSS预处理语言
  18. Linux系统下本地yum镜像源环境部署-完整记录
  19. ASP.NET Core Identity 实战(3)认证过程
  20. Java httpClient 发送http请求

热门文章

  1. 帆软用工具测试超链接打开弹窗(iframe嵌套),解决js传参带中文传递有乱码问题
  2. Error:Execution failed for task &#39;:app:mergeDebugResources&#39;. &gt; Error: Some file crunching failed, see logs for details
  3. 老猿学5G扫盲贴:3GPP规范中与计费相关的主要规范文档列表及下载链接
  4. 第8.30节 重写Python __setattr__方法实现属性修改捕获
  5. SpringBoot之自定义拦截器
  6. Echarts的食用方式
  7. MySQL入门看这一篇就够了
  8. centos7 mysql 自动补全
  9. Robot Framework+adb框架自动化测试Android设备案例⑹——源码地址、测试报告
  10. 学好Spark/Kafka必须要掌握的Scala技术点(二)类、单例/伴生对象、继承和trait,模式匹配、样例类(case class)