WebAPI接口设计:SwaggerUI文档 / 统一响应格式 / 统一异常处理 / 统一权限验证
为什么还要写这类文章?因为我看过网上很多讲解的都不够全面,而本文结合实际工作讲解了swaggerui文档,统一响应格式,异常处理,权限验证等常用模块,并提供一套完善的案例源代码,在实际工作中可直接参考使用。
一、先看看最终效果
这是最后生成的swagerui文档,大家可以直接访问这个地址体验:
http://sapi.daimali.com/swagger/ui/index
(若无法访问,请公众号CodeL联系)
git源码地址:https://gitee.com/daimali/WebApiDemo (推荐直接看源码)
文档效果图:
响应的内容:
注意看红色部分,所有的响应内容都将自动封装成如下格式:由code,msg,data三部分组成
- {
- "code": ,
- "msg": "OK",
- "data": {
- "List": [
- {
- "OrderId": ,
- "UserName": "绿巨人",
- "OrderDate": "2018-11-18T09:39:36.0404502+08:00"
- },
- {
- "OrderId": ,
- "UserName": "钢铁侠",
- "OrderDate": "2018-11-17T09:39:36.0404502+08:00"
- }
- ],
- "total":
- }
- }
实现以上API的整体思路是:
1. 使用SwaggerUI自动生成接口文档、便于团队协作,减少工作量
2. 通过ActionFilter实现权限控制与响应格式的统一封装
3. 通过ExceptionFilter实现异常的统一处理
我觉得大部分人阅读到这里就可以了,剩下的时间去看看源码,需要用的时候边用边学就好了
二、接口文档 - SwaggerUI注意点
1. swagger汉化,注意swagger_lang.js 属性生成操作需要选择"嵌入的资源"
2. 项目右键-属性-生成:输出项勾选XML文档文件
三、统一响应格式说明
通过 ApiResultFilterAttribute 类实现响应参数的统一封装:ApiResultFilterAttribute继承自ActionFilterAttribute
这里封装的响应格式由如下三部分组成: code:跟随HttpCode,msg:返回的描述信息, data:接口返回的业务数据统一放在data中
- {
- "code": ,
- "msg": "OK",
- "data": null
- }
- /// <summary>
- /// 响应数据
- /// </summary>
- /// <typeparam name="T">自定义响应的内容</typeparam>
- public class ResponseApi<T>
- {
- /// <summary>
- /// 错误代码
- /// </summary>
- public int code { get; set; }
- /// <summary>
- /// 错误信息
- /// </summary>
- public string msg { get; set; }
- /// <summary>
- /// 响应数据
- /// </summary>
- public T data { get; set; }
- }
通过actionfilter统一封装:
- public class ApiResultFilterAttribute : ActionFilterAttribute
- {
- /// <summary>
- /// 进入action之前做权限验证
- /// </summary>
- public override void OnActionExecuting(HttpActionContext actionContext)
- {
- var token = true;
- //权限验证省略
- if (token)
- {
- ResponseApi<object> result = new ResponseApi<object>();
- // 取得由 API 返回的状态代码
- result.code = (int)HttpStatusCode.Unauthorized;
- // 取得由 API 返回的资料
- result.data = null;
- result.msg = "invalid ticket value";
- HttpResponseMessage response = new HttpResponseMessage
- {
- Content = new StringContent(JsonConvert.SerializeObject(result),
- Encoding.GetEncoding("UTF-8"), "application/json")
- };
- //结果转为自定义消息格式
- HttpResponseMessage httpResponseMessage = response;
- // 重新封装回传格式
- actionContext.Response = httpResponseMessage;
- }
- }
- /// <summary>
- /// 统一响应格式
- /// </summary>
- public override void OnActionExecuted(HttpActionExecutedContext filterContext)
- {
- base.OnActionExecuted(filterContext);
- if (filterContext.ActionContext.Response != null)
- {
- ResponseApi<object> result = new ResponseApi<object>();
- // 取得由 API 返回的状态代码
- result.code = (int)filterContext.ActionContext.Response.StatusCode;
- // 取得由 API 返回的资料
- result.data = filterContext.ActionContext.Response.Content.ReadAsAsync<object>().Result;
- HttpResponseMessage response = new HttpResponseMessage
- {
- Content = new StringContent(JsonConvert.SerializeObject(result),
- Encoding.GetEncoding("UTF-8"), "application/json")
- };
- //结果转为自定义消息格式
- HttpResponseMessage httpResponseMessage = response;
- // 重新封装回传格式
- filterContext.Response = httpResponseMessage;
- }
- }
- }
ApiResultFilterAttribute.cs
四、自定义异常信息
针对于所有的异常信息,接口也会返回对应的code,msg,data的格式:
通过CustomException和CustomExceptionFilterAttribute实现:
- public class CustomExceptionFilterAttribute : ExceptionFilterAttribute
- {
- /// <summary>
- /// 统一对调用异常信息进行处理,返回自定义的异常信息
- /// </summary>
- /// <param name="context">HTTP上下文对象</param>
- public override void OnException(HttpActionExecutedContext context)
- {
- //自定义异常的处理
- if (context.Exception is CustomException)
- {
- var exception = (CustomException)context.Exception;
- ResponseApi<object> result = new ResponseApi<object>()
- {
- code = exception.GetErrorCode(),
- msg = exception.Message
- };
- throw new HttpResponseException(new HttpResponseMessage(HttpStatusCode.OK)
- {
- //封装处理异常信息,返回指定JSON对象
- Content = new StringContent(JsonConvert.SerializeObject(result),
- Encoding.GetEncoding("UTF-8"), "application/json"),
- ReasonPhrase = "InternalServerErrorException",
- });
- }
- else
- {
- ResponseApi<object> result = new ResponseApi<object>()
- {
- code = -,
- msg = context.Exception.Message
- };
- throw new HttpResponseException(new HttpResponseMessage(HttpStatusCode.InternalServerError)
- {
- //封装处理异常信息,返回指定JSON对象
- Content = new StringContent(JsonConvert.SerializeObject(result)),
- ReasonPhrase = "InternalServerErrorException"
- });
- }
- }
- }
CustomExceptionFilterAttribute.cs
看源码
需要说的东西太多,直接看源码更方便:
接口预览地址:http://sapi.daimali.com/swagger/ui/index
(若无法访问,请公众号联系)
git源码地址:https://gitee.com/daimali/WebApiDemo
继续看详细步骤:
1. 新建空ASP.NET MVC空应用程序,选择WebApi
2. Nuget引用Swashbuckle.Core (demo目前用的v5.6.0最新稳定版)
3. 将App_Start中的类复制到你的新项目中,然后更改命名空间为你自己项目
4. 按需调整SwaggerConfig.cs 配置
5. 将Scripts文件复制到你的项目中,同时设置 swagger_lang.js 文件 属性- 生成操作为"嵌入的资源",按需调整 swagger_lang.js文件内容
6. 注意你的WebApiConfig中需要添加 ApiResultFilterAttribute 和 CustomExceptionFilterAttribute
7. 项目右键-属性-生成:输出项勾选XML文档文件
此时,你新建的webapi控制器已经支持swagger,并且会统一封装成code,msg,data的格式了
修正:
1. HttpGet 请求接收参数时需要加 [FromUri]
2. token验证不应该放在OnActionExecuted中,应该在接口执行之前,比如:AuthorizeAttribute中,或者 ActionFilterAttribute里面的OnActionExecuting方法(源码已调整为OnActionExecuting )
评论区已知问题(待解决,后续将持续更新Demo,感兴趣的同学多多关注):
1. 解决swagger 文件上传问题
相关资源获取或其他疑问可在公众号CodeL留言。
WebAPI接口设计:SwaggerUI文档 / 统一响应格式 / 统一异常处理 / 统一权限验证的更多相关文章
- 服务器文档下载zip格式 SQL Server SQL分页查询 C#过滤html标签 EF 延时加载与死锁 在JS方法中返回多个值的三种方法(转载) IEnumerable,ICollection,IList接口问题 不吹不擂,你想要的Python面试都在这里了【315+道题】 基于mvc三层架构和ajax技术实现最简单的文件上传 事件管理
服务器文档下载zip格式 刚好这次项目中遇到了这个东西,就来弄一下,挺简单的,但是前台调用的时候弄错了,浪费了大半天的时间,本人也是菜鸟一枚.开始吧.(MVC的) @using Rattan.Co ...
- 【奇淫技巧】API接口字段table文档转代码工具
今天做一个视频接口对接,发现对方提供的文档没有json格式,无法自动生成请求和响应对象 json自动生成C#类的工具 http://tool.sufeinet.com/Creater/JsonClas ...
- [转]支付宝接口程序、文档及解读(ASP.NET)
本文转自:http://www.cnblogs.com/blodfox777/archive/2009/11/03/1595223.html 最近需要为网站加入支付宝的充值接口,而目前关于支付宝接口开 ...
- <<海闻电子发票接口 ESB 封装文档>>
<<海闻电子发票接口 ESB 封装文档>> 章节目录结构: 发票验证接口 发票开具接口 ESB请求地址: 发票验证接口: http://10.15.22.120:8866/42 ...
- 支付宝接口程序、文档及解读(ASP.NET)
最近需要为网站加入支付宝的充值接口,而目前关于支付宝接口开发的资料比较杂乱,这里就我此次开发所用到的资料进行汇总整理,希望能够帮助需要的朋友. 开发步骤: 1. 确定签约类型 支付宝的接口有多种类型, ...
- 数字麦克风PDM转PCM与STM32 I2S接口应用----重要文档列表
数字麦克风PDM脉冲到PCM信号需要一个二次采样,ST 提过了PDM2PCM的软件包,可以完成上面的工作.软件包源码没有开源,使用手册也简洁的让人抓狂,我觉得可能是因为ST更高级的MCU直接带了硬解码 ...
- Anakia 转换xml文档为其他格式
一.简介 Anakia 使用JDOM 和Velocity将XML文档转换为特定格式的文档 二.解析xml文档方法 1.DOM java jdk,xml-api.jar 需要加载整个xml文档来构建层次 ...
- WebApi生成在线API文档--Swagger
1.前言 1.1 SwaggerUI SwaggerUI 是一个简单的Restful API 测试和文档工具.简单.漂亮.易用(官方demo).通过读取JSON 配置显示API. 项目本身仅仅也只依赖 ...
- webAPI 自动生成帮助文档
之前在项目中有用到webapi对外提供接口,发现在项目中有根据webapi的方法和注释自动生成帮助文档,还可以测试webapi方法,功能很是强大,现拿出来与大家分享一下. 先看一下生成的webapi文 ...
随机推荐
- mysql数据库单表只有一个主键自增id字段,ibatis实现id自增
mysql数据库单表只有一个主键自增id字段,ibatis实现id自增 <insert id="autoid"> insert into user_id ...
- MFC控件编程之组合框跟列表框
MFC控件编程之组合框跟列表框 一丶简介 如果要使用组合框跟列表框.那么就要知道.组合框列表框是最核心的东西就是索引. 索引是从0开始的. 二丶组合框列表框常用的方法 AddString(字符串) 添 ...
- Win32之隐藏DLL隐藏模块技术
Win32之隐藏DLL隐藏模块技术 这一讲涉及到windows底层技术.跟汇编内容. 我们才可以实现模块隐藏(也称为DLL隐藏) 一丶API反汇编勾引兴趣 我们都用过Windows的进程跟线程API ...
- 动态规划法(十)最长公共子序列(LCS)问题
问题介绍 给定一个序列\(X=<x_1,x_2,....,x_m>\),另一个序列\(Z=<z_1,z_2,....,z_k>\)满足如下条件时称为X的子序列:存在一个严格 ...
- Python GUI
1.flexx Flexx 是一个纯Python工具包,用来创建图形化界面应用程序.其使用 Web 技术进行界面的渲染.你可以用Flexx来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档 ...
- php常用函数搜集
搜集了几个php常用函数方法....相信项目中肯定会用到吧... <?php /** * @param $arr * @param $key_name * @return array * 将数据 ...
- PHP中获取当前页面的URL信息
<? //获取当前的域名: echo $_SERVER['SERVER_NAME']; //获取来源网址,即点击来到本页的上页网址 echo $_SERVER["HTTP_REFERE ...
- 10.QT-定时器
QObject定时器 需要头文件#include <QTimerEvent> 需要函数 int QObject::startTimer(int interval); //启动定时器,并设 ...
- js实现ctrl+v上传图片
描述:实现类似QQ截图删上传图片的功能 a.需要的js插件 paste.image.js 地址:https://github.com/iyangyuan/pasteimg b.paste.image. ...
- js 面向对象 ES5 AND ES6
1. ES5实现 父类: // 职员类 function Employees(id,name,salary) { // 属性 this.id = id; this.name = name; this. ...