WebAPi的可视化输出模式(RabbitMQ、消息补偿相关)所有webapi似乎都缺失的一个功能

来源:互联网 发布:我想在淘宝上买话费 编辑:程序博客网 时间:2024/05/21 09:04

最近的工作我在做一个有关于消息发送和接受封装工作。大概流程是这样的,消息中间件是采用rabbitmq,为了保证消息的绝对无丢失,我们需要在发送和接受前对消息进行DB落地。在发送前我会先进行DB的插入,单表插入,所以在性能上也是能接受的,单表插入做了压测基本上是一到两毫秒的时间,加上消息的发送(有ACK)再加上集群是两个节点的高可用(一个磁盘持久化节点),单台TPS基本上是在2000-3000左右。这对于我们的业务场景来说是够用了。一旦当消息丢失或者由于网络问题、集群问题业务不会中断,消息就算发不出去也没关系,我们会进行消息的补偿或者同步api调用补偿。这是架构设计的必须要考虑的A计划、B计划、C计划,这是敬畏或者危机意识。

你可能又要说两个节点或者三个节点的集群怎么会有问题,那你就错了,大错特错。只能说明你并不了解什么叫分布式系统及分布式系统的特性。你也许不会知道网络抖动、网络闪断导致socket断开如何进行心跳重试已保持有效的Rabbitmq Connection。当你的网络极不稳定,你的linux keepalived VIP 来回漂移,导致你的ARP根本无法成效,可能就连广播都传不出去,而客户端则在一直使用一个无用的IP地址。当你的集群节点之间无法连接成一个整体的时候各种奇葩的问题又来了。这些都是可能导致你的集群出问题的原因,所以不要大意。

(后面我会整理一篇专门讲解“rabbitmq高可用、故障转移集群架构“文章,所以这里我们就不继续介绍了)

这是一个铺垫,本文的重点是介绍下我在尝试使用可视化webapi的输出模式,这比原本json的输出模式看起来会方便许多。如果你的api提供两种输出模式,人性化绝对很好。现在很多后端api都是没有界面的都是只提供了一个json输出。然而,我们其实很需要一个可读性很强的输出模式。

我在开发消息补偿程序的时候,我借鉴了这一思想进行了尝试。先来看下整体架构蓝图:

1

本篇文章要介绍的是有关于这个补偿程序的api的可视化输出内容。不涉及到消息相关太多的东西,只是为了让这个可视化输出看起来容易理解点。这个补偿程序需要对发送的消息和接受的消息进行查询和比较然后输出,用来确定消息的发送是失败了还是成功的。简单逻辑就是比较某个时间段内的消息发送表和接受表,然后进行消息id的匹配。

我在想这个数据反馈到api上是个什么样子的,按照常规设计就是两个字段:

/// <summary>    /// 接受的消息对象。    /// </summary>    public class ReceiveMessage    {        /// <summary>        /// 发送消息ID。        /// </summary>        public string SendMessageId { get; set; }        /// <summary>        /// 接受消息ID。        /// </summary>        public string ReceiveMessageId { get; set; }    }

这表示一个消息从发送到接受的一个过程。如果失败了,可能是只有SendMessageId而没有ReceiveMessageId。然后我才会针对没有ReceiveMessageId的消息进行自动补偿。在开发的时候只有几十条消息,输出到postman中的看起来也还行,但是不直观。

2

GetReceiveMessage是获取接受消息列表,就是查看当前消息发送到接受是个什么状态。

/// <summary>    /// 处理成功消息对象。    /// </summary>    public class SuccessMessage    {        /// <summary>        /// 发送消息ID        /// </summary>        public string SendMessageId { get; set; }        /// <summary>        /// 接受消息ID        /// </summary>        public string ReceiveMessageId { get; set; }        /// <summary>        /// 处理成功消息ID        /// </summary>        public string SuccessMessageId { get; set; }    }

SuccessMessage表示处理成功消息情况。此时有可能是有SendMessageId,ReceiveMessageId消息,但是SuccessMessageId可能是没有的。就会针对处理成功的消息进行发送。

3

突然受到ElasticSearch的_cat endpoint 启发。似乎这里我可以尝试下,webapi带有两种输出模式,一种是针对程序使用的json输出模式,另外一种是针对人可以阅读的模式text/plain模式,而第二种模式可以简单的理解为是行列转换缺省模式。

4

是不是看起来会很舒服。这在进行消息的时间段查看非常有帮助,如果还按照原本的json输出模式可能看起来会比较吃力。

来看下基本的api的设计,为了保证你的所有api支持?v可视化模式,需要一定的抽象:

5

需要定义一种ViewModel,所有的数据都输出这种对象,当然我这里也只是简单地封装。如果可以,其实可以专门提取出一个库出来,包括对文本的输出自动化。

我们看下BaseApiController:

public class BaseApiController : ApiController    {        public class ViewModel        {            public string Content { get; set; }            public object JsonObject { get; set; }            public bool Success = true;        }        protected bool IsView;        private const string ViewQuerystring = "?v";        public ViewModel ResultModel;        private const string CheckToken = "CheckToken";        private const string Token = "49BBD022-CDBF-4F94-80E4-5BCACB1192EC";        private bool _checkStatus;        public override Task<HttpResponseMessage> ExecuteAsync(HttpControllerContext controllerContext, CancellationToken cancellationToken)        {            //验证token            if (controllerContext.Request.Headers != null && controllerContext.Request.Headers.Contains(CheckToken))            {                var requestToken = controllerContext.Request.Headers.GetValues(CheckToken).FirstOrDefault();                if (requestToken != null && requestToken.Equals(Token))                {                    this._checkStatus = true;                }            }            if (!_checkStatus)            {                var checkResult = new Task<HttpResponseMessage>(() => new HttpResponseMessage                {                    Content = new StringContent("非法访问,缺少token", Encoding.UTF8, "text/plain")                }, cancellationToken);                checkResult.Start();                return checkResult;            }            if (controllerContext.Request.RequestUri.Query.Equals(ViewQuerystring))                this.IsView = true;            base.ExecuteAsync(controllerContext, cancellationToken);            //text模式            if (this.IsView)            {                var textResult = new Task<HttpResponseMessage>(() => new HttpResponseMessage                {                    Content = new StringContent(this.ResultModel.Content, Encoding.UTF8, "text/plain")                }, cancellationToken);                textResult.Start();                return textResult;            }            //json模式            var resultData = new Result<object>            {                Data = this.ResultModel.JsonObject,                Type = this.ResultModel.Success ? ResultType.Successfully : ResultType.Failure            };            var jsonResult = new Task<HttpResponseMessage>(() => new HttpResponseMessage            {                Content = new ObjectContent(typeof(Result), resultData, new JsonMediaTypeFormatter(), "application/json")            }, cancellationToken);            jsonResult.Start();            return jsonResult;        }    }


代码很简单,这里给我们一个启发,webapi是不是真的缺少了一个可视化模式。

0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 六个月宝宝拉肚子怎么办拉水 婴儿拉的是沫怎么办 婴儿拉泡沫屎是怎么办 宝宝吃奶粉上火大便干燥怎么办 没满月的宝宝拉肚子怎么办 婴儿吃奶粉上火不大便怎么办 四个月宝宝拉沫怎么办 4个月小孩拉肚子怎么办 3个月的宝宝拉稀怎么办 宝宝没喝过奶粉怎么办 宝宝喝冰酸奶咳嗽了怎么办 7个月不爱喝奶怎么办 宝宝不认妈妈该怎么办 九个月宝宝不喝牛奶怎么办 吃羊肉和西瓜后怎么办 娃儿感冒了很咳怎么办 5岁娃儿经常感冒发烧怎么办 娃儿冷得发抖感冒怎么办 吃羊肉和茶后怎么办 宝宝胆汁酸高22怎么办 5个月发烧38度怎么办 28个月宝宝不愿把尿怎么办 16个月宝宝咳嗽怎么办 宝宝晚上不用纸尿裤要尿床怎么办 宝宝头型睡偏了怎么办 初生婴儿鼻子被奶块堵住怎么办 月经排的不顺畅怎么办 四个月的小孩拉肚子怎么办 月经期做了水光怎么办 4个多月的宝宝拉肚子怎么办 2个月婴儿积食怎么办 3个月婴儿积食怎么办 2个月的婴儿干呕怎么办 脚上的皮烂了怎么办 嘴皮里面烂了怎么办 未满月婴儿拉屎很费劲怎么办 新生儿血钙1.7怎么办啊 3个月宝宝胃口小怎么办 3个月宝宝缺钙怎么办 五个月宝宝吃手睡觉怎么办 2个月宝宝有蛲虫怎么办