WebApi过滤器使用和请求实体合法性验证

1、 在WebApiConfig中注册 请求数据处理(如做签名认证)注册 (继承MessageProcessingHandler)

2、 在WebApiConfig中注册 对请求数据实体注解验证注册(继承ActionFilterAttribute)

3、 在WebApiConfig中注册  如红色部分

public static void Register(HttpConfiguration config)
        {           
            var jsonFormatter = new JsonMediaTypeFormatter {
                UseDataContractJsonSerializer = false
            };
            var serializerSettings = jsonFormatter.SerializerSettings;       
            config.Services.Replace(typeof(IContentNegotiator), new JsonContentNegotiator(jsonFormatter));

           // Web API 路由
            config.MapHttpAttributeRoutes();

            config.Routes.MapHttpRoute(
                name: "DefaultApi",
                routeTemplate: "api/{controller}/{id}",
                defaults: new { id = RouteParameter.Optional }
            );
             // 对请求数据做验证处理
            config.MessageHandlers.Add(new CmbMessageProcesssingHandler ());
             // 对接收的实体对象参数做合法性验证         
            config.Filters.Add(new CmbActionFilterAttribute());
            config.Filters.Add(new ExceptionFilterAttribute());
        }


  

4、详细说明 CmbMessageProcesssingHandler和CmbActionFilterAttribute

CmbMessageProcesssingHandler:对请求数据和响应数据做处理  
public class CmbMessageProcesssingHandler : MessageProcessingHandler
    {
        private static readonly ILog Logger = LogManager.GetLogger(typeof(CmbMessageProcesssingHandler));

 // 数据结构 不是重点
        protected class AbcDataTransferDto
        {
            public string VisitCode { get; set; }

            public string Sign { get; set; }

            public string Content { get; set; }
        }

// 请求数据做处理是重点
        protected override HttpRequestMessage ProcessRequest(HttpRequestMessage request, CancellationToken cancellationToken)
        {
            var contentType = request.Content.Headers.ContentType;
            var stringContent = string.Empty;
            var formData = request.Content.ReadAsFormDataAsync(cancellationToken).Result;
           

            var sign = formData.Get("Sign");
            var visitCode = formData.Get("VisitCode");
            var content = formData.Get("Content");

            if (!string.IsNullOrWhiteSpace(sign) && !string.IsNullOrWhiteSpace(visitCode) && !string.IsNullOrWhiteSpace(content))
            {
                var baseContent = CommonUtils.RijndaelDecrypt(content, Consts.EncryptKey);
                if (VertifySign(visitCode, baseContent, sign))
                {
                    stringContent = baseContent;
                    Logger.Info($"api 接口请求参数解密报文:{stringContent}");
                }
                else
                {
                    Logger.Error($"api 接口请求参数签名无效:{request.Content.ReadAsStringAsync().Result}");
                }
            }
            else
            {
                Logger.Error($"api 接口请求参数格式无效:{request.Content.ReadAsStringAsync().Result}");
            }

            // 将解密后的BODY数据 重置
            request.Content = new StringContent(stringContent);
            //此contentType必须最后设置 否则会变成默认值
            // contentType.MediaType = "application/json"; 当实际内容content是json时 Sing=22&VisitCode=ssss&Content=加密({"/Yurref":22,"/Dbtacc"/:yinghang01})
            //当实际内容content是Yurref=22&Dbtacc=yinghang01 时 application/x-www-form-urlencoded   Sing=22&VisitCode=ssss&Content=加密(Yurref=22&Dbtacc=yinghang01)不修改情况content-type:application/x-www-form-urlencoded  现在是这种方案
            request.Content.Headers.ContentType = contentType;
            return request;
        }
// 响应数据做处理是重点
        protected override HttpResponseMessage ProcessResponse(HttpResponseMessage response, CancellationToken cancellationToken)
        {
            if (response.StatusCode != HttpStatusCode.OK)
                return response;
            var result = response.Content.ReadAsStringAsync().Result;
            var visitCode = Consts.VisitCode;//后续可以考虑随机颁发
            var encodeResult = CommonUtils.RijndaelEncrypt(result, Consts.EncryptKey);
            var dto = new AbcDataTransferDto {
                VisitCode = visitCode,
                Content = encodeResult,
                Sign = GetEncryptSign(visitCode, result)
            };
            response.Content = new StringContent(JsonConvert.SerializeObject(dto));
            return response;
        }        
}

5、 CmbActionFilterAttribute:对请求实体对象验证做处理

public class CmbActionFilterAttribute : ActionFilterAttribute
    {
        private static readonly ILog Logger = LogManager.GetLogger(typeof(CmbActionFilterAttribute));

        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            base.OnActionExecuting(actionContext);
            // 查看接收到的实体对象各属性是否合法
            if (actionContext.ModelState.IsValid)
                return;
            var errorMessage = actionContext.ModelState.ErrorMessage();
            Logger.Error($"模型绑定参数验证无效-RequestUri:{actionContext.Request.RequestUri} errorMessage:{errorMessage}");
            // 如果不合法直接返回 结果信息
            actionContext.Response = actionContext.Request.CreateResponse(HttpStatusCode.OK, new {
                status = -1,
                message = errorMessage
            });
        }
    }
补充说明如实体对象验证信息
        /// <summary>
        /// 业务参考号
        /// </summary>  
        [Required(AllowEmptyStrings = false, ErrorMessage = "业务参考号不能为空")]
        [StringLength(30, ErrorMessage = "业务参考号长度无效,最多30位")]
        public string Yurref { get; set; }

 

 

 

posted @ 2019-12-26 09:25  xiaoBai1001  阅读(561)  评论(0编辑  收藏  举报