最近的一个项目要求服务端与UI层分离,业务层以WebApi方式向外提供所有业务服务,服务在数据保密性方面提出了要求,主要包括:

1:客户端认证;

2:服务请求超时(默认5分钟);

3:服务Get请求的参数密文传输。

以上三个需求为一般的网络服务比较常见的简单要求,在WebApi项目中也比较容易实现,以下是我采用的两种实现方式(任意一种即可):

一:继承HttpClient来加入数据(Uri)加密,加入验证Header,继承ApiController来对Header进行合法性验证,并解密Uri

客户端的关键代码包括对HttpClient继承实现:

     public class MyHttpClient : HttpClient
     {
         private static MyHttpClient _myClient;

         public static MyHttpClient Instance
         {
             get
             {
                 if(_myClient!=null)return _myClient;
                 _myClient = new MyHttpClient();
                 _myClient.DefaultRequestHeaders.Add("Mac", MacMd5);
                 return _myClient;
             }
         }

         private MyHttpClient()
         {
         }

         private static readonly IAuthorize _authorize = new MacAuthorize();
         private static string _macMd5 = string.Empty;

         private static string MacMd5
         {
             get
             {
                 if (!string.IsNullOrEmpty(_macMd5)) return _macMd5;
                 var macAddress = _authorize.UniqueId;
                 if (string.IsNullOrEmpty(macAddress)) return string.Empty;
                 _macMd5 = macAddress.GetMD5();
                 return _macMd5;
             }
         }

         private static string TimeAes
         {
             get
             {
                 var time = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
                 var aesTime = time.AesEncrypt();
                 return aesTime;
             }
         }

         private static void AddHeader(ref string url)
         {
             var segs = url.Split('/');
             ];
             if (para.Contains("="))
             {
                 var aes = para.AesEncrypt();
                 url = url.Replace(para, aes.GetMD5());
                 IEnumerable<string> p = new List<string>();
                 if (_myClient.DefaultRequestHeaders.TryGetValues("Param", out p))
                     _myClient.DefaultRequestHeaders.Remove("Param");
                 _myClient.DefaultRequestHeaders.Add("Param",aes);
             }

             IEnumerable<string> time = new List<string>();
             if (_myClient.DefaultRequestHeaders.TryGetValues("Time", out time))
                 _myClient.DefaultRequestHeaders.Remove("Time");
             _myClient.DefaultRequestHeaders.Add("Time", TimeAes);
         }

         public override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request,
             System.Threading.CancellationToken cancellationToken)
         {
             var url = request.RequestUri.AbsoluteUri;
             AddHeader(ref url);
             return base.SendAsync(request, cancellationToken);
         }

         public override int GetHashCode()
         {
             return _myClient.GetHashCode();
         }

     }

其中时间戳为了简便,未转为ms格式进行验证,仅作为字符串进行了验证。

在客户端进行调用时,通过继承的MyHttpClient发送服务请求即可:

 var client = MyHttpClient.Instance;
 var response = client.GetAsync(urlBase.AbsoluteUri + "api/test/GetUser/id=123123&name=jiakai").Result;

服务端的关键代码包括对ApiController的继承实现:

     public class BaseApiController : ApiController
     {
         protected override void Initialize(HttpControllerContext controllerContext)
         {
             base.Initialize(controllerContext);
             //获取请求头信息
             var requestHeader = controllerContext.Request.Headers;
             IEnumerable<string> mac = new List<string>();
             requestHeader.TryGetValues("Mac", out mac);
             IEnumerable<string> time = new List<string>();
             requestHeader.TryGetValues("Time", out time);
             IEnumerable<string> para = new List<string>();
             requestHeader.TryGetValues("Param", out para);
             var paramList = controllerContext.Request.RequestUri.AbsoluteUri.Split('/');
             if (mac == null || time == null || para == null)
             {
                 var resp = Request.CreateResponse<string>(HttpStatusCode.Forbidden, "非法请求", "application/json");
                 throw new HttpResponseException(resp);
             }
             //验证机器MAC地址
             if (mac.Any()&&!string.IsNullOrEmpty(mac.First()))
             {
                 //TODO:验证机器MAC地址是否为已注册MAC地址
             }
             //验证时间戳
             if (time.Any() && !string.IsNullOrEmpty(time.First()))
             {
                 var t = Convert.ToDateTime(time.First().AesDecrypt());
                 ) < DateTime.Now)
                 {
                     var resp = Request.CreateResponse<string>(HttpStatusCode.RequestTimeout, "请求过期", "application/json");
                     throw new HttpResponseException(resp);
                 }
             }
             //验证参数MD5
             if (para.Any() && !string.IsNullOrEmpty(para.First()))
             {
                 ];
                 if (para.First().GetMD5() != newMd5)
                 {
                     var resp = Request.CreateResponse<string>(HttpStatusCode.Forbidden, "参数MD5验证失败", "application/json");
                     throw new HttpResponseException(resp);
                 }
             }
         }
     }

同样,在业务实现的Controller中,继承改为BaseApiController即可:

 public class TestController : BaseApiController
 {
     //TODO:业务服务具体实现
 }

二:通过对WebApi消息处理框架中HttpMessageHandler的自定义实现,并分别注入到Request及Response的消息处理阶段,来实现加密/解密的功能

客户端HttpClient中HttpMessageHandler的实现:

     public class RequestHandler : DelegatingHandler
     {
         private static string TimeAes
         {
             get
             {
                 var time = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
                 var aesTime = time.AesEncrypt();
                 return aesTime;
             }
         }

         protected override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request,
             System.Threading.CancellationToken cancellationToken)
         {
             var uri = request.RequestUri.AbsoluteUri;
             var port = request.RequestUri.Port;
             var para = uri.Split('/').Last();
             if (para.Contains("="))
             {
                 var aes = para.AesEncrypt();
                 uri = uri.Replace(para, aes.GetMD5());
                 IEnumerable<string> p = new List<string>();
                 if (request.Headers.TryGetValues("Param", out p))
                     request.Headers.Remove("Param");
                 request.Headers.Add("Param", aes);
             }
             IEnumerable<string> time = new List<string>();
             if (request.Headers.TryGetValues("Time", out time))
                 request.Headers.Remove("Time");
             request.Headers.Add("Time", TimeAes);
             request.RequestUri = new Uri(uri);

             return base.SendAsync(request, cancellationToken);
         }
     }

在实现了自定义的HttpMessageHandler后,实现一个HttpClient工厂,对外提供一个注入了该HttpMessageHandler的HttpClient单例对象:

     public class AtmHttpClientFactory
     {
         private static HttpClient _atmClient;
         private readonly static HttpClientHandler ClientHandler = new HttpClientHandler();

         public static HttpClient GetClient()
         {
             if (_atmClient != null) return _atmClient;
             _atmClient = new HttpClient(new RequestHandler() {InnerHandler = ClientHandler });
             _atmClient.DefaultRequestHeaders.Add("Mac", MacMd5);
             return _atmClient;
         }

         private AtmHttpClientFactory()
         {
         }

         private static readonly IAuthorize _authorize = new MacAuthorize();
         private static string _macMd5 = String.Empty;

         private static string MacMd5
         {
             get
             {
                 if (!String.IsNullOrEmpty(_macMd5)) return _macMd5;
                 var macAddress = _authorize.UniqueId;
                 if (String.IsNullOrEmpty(macAddress)) return String.Empty;
                 _macMd5 = macAddress.GetMD5();
                 return _macMd5;
             }
         }
     }

这样,在客户端的调用加密过程就完全被封装,简化为如下:

      //调用实现1
      var myClient = AtmHttpClientFactory.GetClient();
      var myResponse = myClient.GetAsync("http://api.ezbatm.com:7777/api/test/GetUser/id=123123&name=jiakai").Result;

服务端也是类似的方法,在app的config中注入我们自己实现的服务端HttpMessageHandler即可。response的HttpMessageHandler实现:

     public class ResponseHandler : DelegatingHandler
     {
         protected override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request,
             System.Threading.CancellationToken cancellationToken)
         {
             //获取请求头信息
             IEnumerable<string> mac = new List<string>();
             request.Headers.TryGetValues("Mac", out mac);
             IEnumerable<string> time = new List<string>();
             request.Headers.TryGetValues("Time", out time);
             IEnumerable<string> para = new List<string>();
             request.Headers.TryGetValues("Param", out para);
             var paramList = request.RequestUri.AbsoluteUri.Split('/');
             //验证Header是否完整
             if (mac == null || time == null || para == null)
             {
                 return RequestNotAcceptable();
             }
             //验证机器MAC地址
             if (mac.Any() && !string.IsNullOrEmpty(mac.First()))
             {
                 //TODO:验证机器MAC地址是否为已注册MAC地址
             }
             //验证时间戳
             if (time.Any() && !string.IsNullOrEmpty(time.First()))
             {
                 var t = Convert.ToDateTime(time.First().AesDecrypt());
                 ) < DateTime.Now)
                 {
                     return RequestTimeOut();
                 }
             }
             //验证参数MD5
             if (para.Any() && !string.IsNullOrEmpty(para.First()))
             {
                 ];
                 //if (para.First().GetMD5() != newMd5)
                 {
                     return RequestForbidden();
                 }
             }
             return base.SendAsync(request, cancellationToken);
         }

         private static Task<HttpResponseMessage> RequestForbidden()
         {
             return Task.Factory.StartNew(() =>
             {
                 return new HttpResponseMessage(HttpStatusCode.Forbidden)
                 {
                     Content = new StringContent("请求未通过认证")
                 };
             });
         }

         private static Task<HttpResponseMessage> RequestTimeOut()
         {
             return Task.Factory.StartNew(() =>
             {
                 return new HttpResponseMessage(HttpStatusCode.RequestTimeout)
                 {
                     Content = new StringContent("请求已超时")
                 };
             });
         }

         private static Task<HttpResponseMessage> RequestNotAcceptable()
         {
             return Task.Factory.StartNew(() =>
             {
                 return new HttpResponseMessage(HttpStatusCode.NotAcceptable)
                 {
                     Content = new StringContent("请求为非法请求")
                 };
             });
         }
     }    

之后,在(owin框架)Startup中注入该自定义HttpMessageHandler即可:

         public void Configuration(IAppBuilder app)
         {
             // Configure Web API for self-host.
             var config = new HttpConfiguration();
             config.Routes.MapHttpRoute(
                 name: "DefaultApi",
                 routeTemplate: "api/{controller}/{action}/{param}",
                 defaults: new { id = RouteParameter.Optional }
             );
             //注入response handler
             config.MessageHandlers.Add(new ResponseHandler());

             app.UseWebApi(config);
         }

完成了以上客户端+服务器端的工作后,即可按照我们的实际需求对Request及Response消息进行自定义的处理。

总结:以上两种方法中,客户端及服务器端的方案都是解耦的,也就是是,你可以采用方法一的客户端方案+方法二的服务器端方案,等等以此类推。

WebApi服务Uri加密及验证的两种方式的更多相关文章

  1. 使用使用dockerfile构建webapi镜像然后使用link和bridge两种方式进行桥接

    首先新增一个webapi的项目 项目核心代码 UserContext using Microsoft.EntityFrameworkCore; using System; using System.C ...

  2. SQL Server验证的两种方式

    1.Windows身份验证:本机连接或者受信的局域网连接(一般在忘记管理员密码或者做系统配置的情况下使用). 2.SQLServer验证:使用用户名.密码验证(推荐使用). 启用方法:以Windows ...

  3. WCF服务使用(IIS+Http)和(Winform宿主+Tcp)两种方式进行发布

    1.写在前面 刚接触WCF不久,有很多地方知其然不知其所以然.当我在[创建服务->发布服务->使用服务]这一过程出现过许多问题.如客户端找不到服务引用:客户端只在本机环境中才能访问服务,移 ...

  4. 服务容错保护断路器Hystrix之一:入门示例介绍(springcloud引入Hystrix的两种方式)

    限流知识<高可用服务设计之二:Rate limiting 限流与降级> 在微服务架构中,我们将系统拆分成了一个个的服务单元,各单元间通过服务注册与订阅的方式互相依赖.由于每个单元都在不同的 ...

  5. 不停止MySQL服务增加从库的两种方式

    不停止MySQL服务增加从库的两种方式 转载自:http://lizhenliang.blog.51cto.com/7876557/1669829 现在生产环境MySQL数据库是一主一从,由于业务量访 ...

  6. XFire构建服务端Service的两种方式(转)

    XFire构建服务端service的两种方式,一是用xfire构建,二是和spring集成构建. 一,xifre构建,确保把xfire的jar包导入到工程中或classpath. 1,service的 ...

  7. XFire构建服务端Service的两种方式

    1.原声构建: 2.集成spring构建 http://blog.csdn.net/carefree31441/article/details/4000436XFire构建服务端Service的两种方 ...

  8. 不停止MySQL服务增加从库的两种方式【转载】

    现在生产环境MySQL数据库是一主一从,由于业务量访问不断增大,故再增加一台从库.前提是不能影响线上业务使用,也就是说不能重启MySQL服务,为了避免出现其他情况,选择在网站访问量低峰期时间段操作. ...

  9. springboot 注册服务注册中心(zk)的两种方式

    在使用springboot进行开发的过程中,我们经常需要处理这样的场景:在服务启动的时候,需要向服务注册中心(例如zk)注册服务状态,以便当服务状态改变的时候,可以故障摘除和负载均衡. 我遇到过两种注 ...

随机推荐

  1. windows 如何查看端口占用情况?

    原文来自:http://www.iteye.com/topic/1117270 开始--运行--cmd 进入命令提示符 输入netstat -ano 即可看到所有连接的PID 之后在任务管理器中找到这 ...

  2. java 22 - 9 多线程之 代码实现的方式2

    多线程的代码实现: 方式2:实现Runnable接口 步骤: A:自定义类MyRunnable实现Runnable接口 B:重写run()方法 C:创建MyRunnable类的对象 D:创建Threa ...

  3. EditText限制输入字符类型的几种方式

    最近的项目上需要限制EditText输入字符的类型,就把可以实现这个功能的方法整理了一下: 1.第一种方式是通过EditText的inputType来实现,可以通过xml或者Java文件来设置.假如我 ...

  4. virtual box使用

    1.工具栏菜单显示 用的是mac电脑.开始发现分辨率小,选了视图->全屏模式的菜单之后发现工具栏菜单不见了. 解决办法:用一个外置键盘,右ctrl+c进行恢复 virtualBox菜单栏和状态栏 ...

  5. 字符串截取函数substr()

    substr(参数1,参数2[,参数3]); 该系统函数返回被截后的子字符串,它接受2个必选参数,参数1为要截取的字符串,参数2为截取的开始位置,参数3可选,表示截取长度. 例子:substr(&qu ...

  6. usb驱动开发23之驱动生命线

    关于字符串描述符的地位仅次于设备/配置/接口/端点四大描述符,那四大设备必须得支持,而字符串描述符对设备来说则是可选的,这并不是就说字符串描述符不重要,对咱们来说,提供字符串描述符的设备要比没有提供的 ...

  7. [转]PHP 下使用 ZeroMQ 和 protobuf

    FROM : http://www.68idc.cn/help/makewebs/php/20150118175432.html 前言 这个记录总的来说分两部分: 搭建环境. 简单使用教程. 搭建环境 ...

  8. OkHttp使用教程

    Android系统提供了两种HTTP通信类,HttpURLConnection和HttpClient.关于HttpURLConnection和HttpClient的选择>>官方博客尽管Go ...

  9. KMS10流氓软件

    win10想激活,结果中了流氓软件的当... (关键是win10家庭单语言版居然还激活不了....白吃亏了) 把我的chrome 和 firefox 主页改成hao.qquu8.com,该网址重定向到 ...

  10. Laravel 安装多国语言包后,phpstorm 还是报错

    问题: 解决办法: vagrant@homestead:~/Code/awbeci$ composer require "overtrue/laravel-lang:~3.0" 总 ...