来自阿里云RDS团队的论文“TcpRT: Instrument and Diagnostic Analysis System for Service Quality of Cloud Databases at Massive Scale in Real-time” (TcpRT:面向大规模海量云数据库的服务质量实时采集与诊断系统)被数据库顶会SIGMOD 2018接收,会议将于6/10在美国休斯敦召开。
商南网站建设公司创新互联公司,商南网站设计制作,有大型网站制作公司丰富经验。已为商南1000+提供企业网站建设服务。企业网站搭建\成都外贸网站制作要多少钱,请找那个售后服务好的商南做网站的公司定做!
TcpRT论文介绍了RDS天象系统在云数据库SLA数据采集、服务质量指标计算、异常检测、故障根因分析领域的创新工作,以及大规模部署自动化的服务各类云上客户的实践经验。评委评价“I have plenty of experience with manual anomaly detection. That has wasted much time for me at work, so I liked what you described.”
简介
随着企业上云趋势的日益热化,作为产业核心组件的数据库,已成为各大云计算公司增长最快的在线服务业务。作为中国第一大云数据库厂商,我们RDS团队致力于为用户提供稳定的云数据库服务。从本质上看,RDS是一个多租户DBaaS平台,利用轻量级KVM、Docker镜像等资源隔离技术将用户所购买的数据库实例部署在物理机上,按需分配资源并进行自动升降级,实现一套完全自动化的智能运维管理。
云数据库对客户业务的稳定性至关重要,因此快速发现云数据库性能出现异常,及时定位异常原因是云数据库厂商的一个挑战。TcpRT是阿里云数据库用来监控和诊断数据库服务质量的一个基础设施。TcpRT从主机TCP/IP协议栈的壅塞控制采集trace数据,计算数据库延迟和网络异常,在后台流式计算平台进行大规模实时数据分析和聚合,通过统计指标历史数据的柯西分布发现异常点,并通过同一台主机、交换机、proxy下所有实例一致性趋势的比例来计算不同组件发生异常的概率。
到目前为止,TcpRT以每秒采集2千万条原始trace数据、每天后台处理百亿吞吐数据、秒级检测异常的卓越性能在阿里云持续稳定运行三年。
本文贡献:
问题
从网络架构上看,RDS由控制层和数据链路层两个部分组成,如上图所示。其中,控制层包括一系列管理模块,比如资源管理器,HA管理器,迁移管理器等。资源管理器负责将实例的不同副本分配在独立的物理主机上。当实例发生故障时,HA管理器会探测到主实例的故障,并将服务连接切换到Standby节点上。而当主机负载失衡时,迁移管理器负责将数据库实例进行迁移,保障用户服务质量。
数据链路层主要负责数据的分发和路由。通常,云用户通过ECS和RDS实现业务上云。他们将业务部署在ECS上,并通过VPC和RDS数据库实例进行交互。数据包通过驻留在用户VPC网络中的vSwitch进行打包/解包并路由。而SLB负责将这些数据包进行解析,并将DBaaS IP映射到真实服务器上。目前,SLB同时支持FNAT和DNAT两种模式。由于出流量数据不经过SLB, DNAT比FNAT呈现出更好的性能。除此之外,我们引入NGLB进行优化。在Proxy中,我们解析客户端/服务端协议,并提取查询,将读写进行分离,同时支持横向分区和连接池的功能。
作为客户业务的核心组件,保障云数据库7/24的稳定至关重要。我们通过对用户承诺数据库SLA服务等级协议,来保障用户的权益。然而,云数据库的服务质量受各种因素影响,例如突发性连接断开、数据库延迟发生抖动、吞吐骤然下降等,都有可能带来用户业务指标的下降。因此,快速发现云数据库性能异常,及时定位根因是云数据库厂商的一个挑战。
可能造成云数据库的服务质量下降的原因很多,例如网络上的异常,比如DB主机的上联交换机tor发生丢包,或者load balaner出现TCP incast问题;或者用户侧的问题,用户端网络异常或者丢包;或者多租户机制,DB主机内核缺陷、硬件上SSD的硬件异常等等都会引起。在出现问题时,快速诊断定位解决问题是关键的问题。
传统数据库性能采集只需要采集DBMS内部处理SQL请求的延迟就够了,但云数据库需要end-end数据。因为对云上用户而言,他看到的云数据库的延迟是终端的延迟。真实的场景对trace工具提出这些需求,因此很关键的是要采集end-to-end数据,以及链路每段的延迟,终端用户感受到的延迟是所有路径上每个节点处理延迟以及网络上所有延迟的总和,不仅仅要采集DB上的延迟,还要采集proxy上看到的延迟。这需要在链路上引入trace。还要采集网络上的数据,主机上其实可以看到的上行乱序和下行的重传,这个信息对推测网络有无异常非常重要。发送和接收会走不同的网络路径。
传统的trace手段需要入侵式地在业务代码中添加埋点。第一,我们不能在客户端埋点,因为客户基本上都使用标准数据库客户端来访问,没有嵌入打点代码,我们也不能期望客户会修改自己的业务代码,加入打点逻辑,并将打点数据采集交给我们。因此我们需要一种非侵入式的方法来获取end-to-end性能数据。第二在服务器端埋点也有问题,应用层无法感知真正的数据接收和发送时间。应用层记录的时间是内核把数据交付给应用层和应用层把数据交付给内核的时间。在系统负载很高,应用层进程调度需要花费大量时间的场景下,会导致请求的实际处理时间计算不准确;应用层无法感知到网络链路质量。在网络出现拥塞,大量的重传报文,导致数据发送和接收过程大大延长,无法区分出是对端响应缓慢还是网络链路质量不佳。
在这种挑战下,TcpRT——阿里云数据库监控和诊断服务质量系统,孕育而生。
TcpRT从主机TCP/IP协议栈的拥塞控制采集trace数据,用于监测数据库延迟和网络异常,并利用流技术,在后台实时计算平台上进行大规模在线数据分析,结合离线模型,通过实时异常事件判定,以及RDS网络关系图谱中的趋势一致性概率探测,快速诊断出性能异常并定位原因。
架构
上图是TcpRT的概要架构图,其中包含以下个组件:
TcpRT内核模块
内核模块主要负责对网络数据的传输进行整个生命周期的监控。在遵守停等协议的TCP通信机制下,服务端处理每个请求的过程分成3个阶段,接收阶段(Receive)、处理阶段(Handle)和响应阶段(Response)。如下图所示:
鉴于此,我们需要计算如下时间:
上行时间 = T1 - T0
处理时间 = T2 - T1
下行时间 = T3 - T2
查询时间 = T3 - T0
RTT时间 = T2 - T2'
现有的监控方案,通常在业务层面埋点进行服务耗时监控,但此方法既不能获得真正的数据接收和发送时间、也无法感知到网络链路的质量,更需要在业务代码中入侵式地添加埋点。因此,我们实现了一种通用、低开销的内核模块进行trace监控以得到上述时间。
首先,我们选择修改Linux内核的拥塞控制算法。此算法可以感知内核发送报文的上下文,并且支持热更新(只需增加一个驱动模块而无需更新线上内核)。
此外,拥塞控制算法提供了以下机制:
1. 每个TCP通信是独立的拥塞控制算法,不存在资源竞争情况
2. 可以感知到收到的每个ACK报文的上下文
3. 在已经发送的报文都已经被ACK的情况下,可以感知到当前发送的报文上下文
根据拥塞控制算法提供的事件回调,我们可以获取到每个TCP连接下述事件:
1. 客户端发送给服务端ACK报文
2. 在所有已经发出去的sequence都被确认的情况下,服务端发送报文
3. TCP连接建立
4. TCP连接断开
内核任何线程都可能调用到拥塞控制算法,因此为了性能,拥塞控制算法必须保证是没有数据争抢的。TcpRT内核模块保证了以下四点:
1. TcpRT所有的数据保存在每个TCP连接对象上,所有的数据读写都没有跨TCP连接的共享;
2. TcpRT在每个CPU core都有独立的写缓冲区,防止了多个内核线程争抢写缓冲区加锁;
3. 为了避免内存开销,又保证实时性,TcpRT的写缓冲区会在buffer满或者时间到的情况下,刷新写缓冲区到debugfs,供应用层采集端采集。由于写debugfs的频率很低,debugfs的锁争抢几乎不存在,最大限度保证了性能;
4. TcpRT回写的数据,是binary格式的,对比需要format的字符格式,在实测场景可以提高20%的性能。
此外,通过给linux内核添加setsockopt选项,通知内核一个不需要应答的请求交互过程已经结束,从而支持非停等协议。
针对TcpRT内核模块对用户数据库实例的性能影响,我们基于sysbench模拟了MySQL 400个客户端连接进行压测,结果如下图所示,TcpRT内核模块对系统的负载影响不到1%。
TcpRT内核模块,利用debugfs和用户态通信。每秒以千万的trace数据高速产出并吐入至debugfs中。为了减轻后台在线分析任务的压力,我们构建本地TcpRT聚合器,实现本地trace秒级聚合,并将聚合结果输出到/dev/shm中。Logagent从中读取聚合数据,发送至Kafka,并由后台ETL接手,进行实时数据分析,流程如下图所示:
在本地聚合器中,聚合操作需要保证可交换且可结合,我们采用均值、最大值、请求个数的三元组聚合方法来保证延迟时间类指标满足这一要求。此外,我们采用每秒同客户端出现的不同端口数对活跃连接数进行指标特征提取。同时,抽取请求数作为特征,建立用户长短连接的使用模型,进而对用户使用数据库实例的负载模式进行分析。根据历史数据,当前仍有众多用户采用短连接模式,这对于诸如MySQL线程网络模型的DB是非常不友好的,从而激发我们提供Proxy中间件将短连接转换为长连接服务。
为了最大化聚合效果,我们在内存中维护最近5s的聚合结果,将5s前的数据输出到/dev/shm的文件中。且为了提高查询性能以及长时间段的聚合操作,我们将三种粒度1s, 5s, 1m的聚合结果存入到库中。
TcpRT ETL
下图描绘了ETL任务的拓扑结构:
在线ETL任务主要包括四个子任务:
其中,延迟和乱序到达的处理是一个难点。TcpRT输出的时序数据是以SQL执行结束时间为时间戳,如何做到实时准确地对窗口内的时序数据进行聚合并将结果刷出是一个两难的问题。我们采用了“尽力聚合”的方法,即对窗口设定等待时间,待时间结束后将实时聚合结果刷出,若此后,还有该窗口的时序数据到来,则直接落入到数据库中。如TcpRT聚合器中所述,所有聚合数据具有可再结合特性,这样,我们可以对同时刻的聚合数据进行二次聚合。
在线异常监测
组件的异常往往伴随着相关指标的抖动。比如主机发生IO Hang后,load、dirty、writeback、某些core的iowait、被阻塞的线程数等指标会明显升高,而各实例的write,CPU使用率会明显变低,主机维度的PT指标会明显升高。在部分情况下,连接数会上升,长连接请求数会下降,流量会下降等。在Proxy发生异常的时候,PT可能会升高、CPU、流量、连接数均可能会下降。
传统方法下,我们通过设定阈值进行指标抖动的检测。然而阈值的设定强依赖于专家经验,维护成本高,且一刀切的设定常常会“误伤”健康指标。比如,有些数据库实例是专门用于OLAP,它们的SQL请求处理时间往往都是秒级的,若采用常用的SQL请求处理时间作为异常判定阈值,此类数据库实例就会触发报警。
鉴于此,我们需要一种通用且自适应的智能模型来解决云数据库的异常监测。
为了避免人工设定阈值,我们一开始尝试利用control charts来进行判断,根据样本时间段的均值和标准差,预测未来时间段的置信区间。若实际值超出置信区间,则判定为异常。然而,若样本本身为异常,则此时间段的参数均不可信。
如上图,左上图为(ins1,*,db1)的upsize指标时序图,可以看到(ins1,*,db1)的upsize指标会有周期性的波动。左下的两张图分别是历史时间窗口设定值为30min的(ins1,*,db1)的upsize指标mean&median时序图和SD&MAD时序图。可以清楚看到,当upsize指标发生波动后,mean值和标准差SD值都会发生窗口时间长度的跳变,但median和MAD却几乎不受指标波动的影响,显得更平稳、丝滑。
右上图为(ins2,*,db2)的newConn指标时序图,可以看到在03:40~04:00期间,指标发生异常,出现大量极端值,由于历史的时间窗口设定值为30min,所以可以从左下的图表中看到,很长一段时间内样本的均值和标准差便会发生较大的波动,而中位数和MAD指标却显得平滑,对极端值并不敏感,展出超强的鲁棒性。由于依赖均值和标准差进行预测,control charts具有不稳定性,易造成误判。这启发我们利用中位数和MAD替代均值和标准差作为预测模型参数。
如上图所示,通过大量观察,我们发现,按(*,*,db) 和(*,*,proxy) 粒度聚合后的采样点集合近似正态分布。但是,正态分布依赖平均值和标准差作为参数,如上文所述,这两个参数波动大,会造成严重误判。鉴于柯西分布与正态分布相似,且不依赖均值和标准差,可以使用中位数和MAD这种稳定的统计指标来进行回归。从上上图可以看到当历史时间窗口长度设定合适的时候,中位数和MAD在面对指标波动和异常的情况下,显得平滑且稳定,这样回归出的模型具有更强的稳定性。于是,我们使用中位数和MAD作为参数的回归柯西分布来拟合异常诊断模型。
为了获取回归需要的样本集,我们通过对过去一段时间(比如:最近一个小时)的每一个时间点做采样,得到一段历史窗口的数据点集合S{x0,x1,x2,……}。根据历史窗口数据集,计算中位数M以及MAD,回归出这个数据集的柯西概率分布D。
主机异常检测
RDS主机上承载着众多实例,各实例通常隶属于不同用户的不同业务。在主机正常工作时,由于实例相互独立、并且对应的指标波动具有不确定性,所有实例呈现出一致性升高或降低的概率非常小。当主机发生异常时,由于该主机上的所有实例共享同一资源,某些关键指标会呈现出一致性趋势。当一台主机发生了IO Hang,主机上大部分实例的SQL处理延迟都呈现出升高的趋势,各实例的数据写入量呈现出下降的趋势。这启发我们利用实例趋势一致性概率来判断主机的异常。
首先,设定函数H(curentVal, prevMideanVal,metricType)作为判断指标metric1是否异常的函数,取值只能是-1,0,+1。currentVal代表这一刻该指标的值,prevMideanVal代表过去一段时间(比如:1小时)采样点集合的中位数,metricType代表该指标的类型(比如:rt,rtt,qps,流量等)。取值为-1代表这个指标在跟过去一段时间指标做对比的时候呈现出明显下降的趋势,0代表这个指标没有明显波动,+1代表这个指标与过去一段时间相比明显上升。
我们可以利用上文的算法来实现函数H,这样我们能算出(ins,*,dstIp,metricType)级别数据的函数H值,然后在按(ins,*,dstIp,metricType)->(*,*, dstIp,metricType)做map-reduce(agg类型为sum)。算出的值s反映了这个指标在这台主机上的突升,突降的趋势,用求和值s除以这个主机上的活跃实例数t得出突升/突降实例比例r,当r大于0是,总体趋势上升,当r小于0是,总体趋势下降,且绝对值越大概率越大。
那么我们如何利用r值来判断主机异常呢?如果机器突然由正常变为异常时,各实例由于都依赖这台机器的资源进行工作,或多或少要受其影响,因此相关指标发生突变的实例比例将会发生突变,比如主机上cpu资源突然不足,大部分实例都会受影响,他们的处理时间会突升,流量突降,请求数突降,因此可以通过判断r值是否突变来判断主机异常。
我们可以利用上文的算法来对r值的突变进行判断,对过去一段时间的每一个时间点都做这样的计算,我们可以得到一段历史窗口的突升/突降实例比例r行程的数据点集合S{R0,R1,R2,…}。根据历史窗口数据集,我们算出数据集的中位数M,以及MAD值,回归出这个数据集的柯西概率分布D。
因为比例r的取值是-1~1,而柯西概率分布D的自变量x范围是负无穷到正无穷。我们需要对原来的比率r做转化让他的范围扩充到正负无穷。通过定义的映射函数求出这一时刻下该指标的柯西概率分布的CDF(),如果CDF()非常小,比如小于0.1%,主机hostA的指标metric1呈现总体下降趋势,或者非常大比如99.8%, 主机hostA的指标metric1呈现总体上升趋势。 但是这样做判断,只是判断出了r值的突升和突降,为了减少误判,我们还要判断出r值的突升和突降需要落到警戒范围。因此需要一个必要条件:r值的绝对值至少为20%。
另外,当r值足够高时,无论r值是否突升还是突降,都应该认为是异常,因此还需要一个充分条件:r值的绝对值大于AlarmRatio(主机上的活跃实例t),那么认为是异常。根据我们自己的实际情况,AlarmRatio(t)=0.8*Pow(0.987, t) + 0.2. 曲线如图,当主机上的活跃实例比较少时,AlarmRatio值比较高,这样为了保证判断出异常的主机上异常的实例足够多,这样才有较高的说服力,而随着主机上活跃实例数变多,AlarmRatio值会相应得变小最后收敛到0.2,这样为了不漏掉r比较少但是异常实例数足够多的情况。
当r=-1时,这台机器hostA上所有的实例的指标metric1都出现了下降趋势,当r=1时,hostA上所有实例的指标metric1都出现了上升趋势。除了独占实例,一台主机上有数十甚至上百的实例,他们分属不同的用户,运行着不同的业务,呈现出一致的趋势概率很小,因此在正常的时候r值往往稳定在一个较低的范围内,当r值很高的时候,极有可能主机问题了。
但是这样的判断还不够,因为还存在这两种情况:1. 实例之间公用的某个组件出现了异常。比如网络中间件引起的异常,路由器的异常等等。也会引起r值变高;2.有的主机上因为实例数比较少(比如:小于3)时,单纯根据比例判断还分辨不出是否是主机的问题,因为样本数太少,不具有说服力。针对这两种情况,我们还用了多种方法来提高准确率,比如:
1.结合物理机的资源指标(load,iowait,dirty,writeback等)的加权和来协助判断。2.如果存在上游节点(比如:proxy),并且上游节点存在异常,则优先报上游节点的异常。因为上游节点往往还连接多个下游节点,如果是某个单独的下游节点出现了异常,一般上游节点是不会出现异常的,因为上游节点上连接的下游节点很多,这一个点并不会明显改变整体的指标趋势变化,而某个节点的多个下游节点都出了问题,他们的上游节点出问题的概率更大一些。假设上游节点完全正常,下游节点近似相互独立,一个节点出问题的概率为p,K各节点同时出问题的概率就是p^K,所以问题的原因很可能是概率更大的上游节点。
Proxy异常检测
我们的大部分实例在访问db前要经过proxy的转发,proxy可以帮用户做短链接优化,负载均衡,连接审计,注入检测等。proxy是我们生产集群非常重要的组件,一个proxy节点最多会有上千个实例的请求(requests of thousands of instances)经过。也就是说,如果一个proxy节点发生了故障,将会影响到上千个实例,这样的故障我们需要快速准确地发现并定位出来,否则后果不敢想象。
我们一开始对proxy的qps,连接数等设定阈值的方式来进行报警,但是由于不同分组的proxy业务量大小不一样,因此我们需要对不同分组设定不同的阈值,而且我们的业务增长迅速,proxy会经常进行扩容,这样阈值又要重新调整,维护起来费时费力。
我们虽然可以利用上边所讲的判断db主机异常一样的算法来判断proxy异常,但是由于proxy的处理时间包含了db本地的处理时间,应用这种方式我们是无法评估出请求在单纯在proxy中停留的时间。对于使用了proxy链路的用户,由于在网络中多了proxy转发的代价,所以SQL请求的延时会稍微比不用proxy链路的请求要慢。因此为了不影响用户的体验,我们希望SQL请求在proxy节点中停留的时间越短越好。因此我们以SQL请求在proxy节点中停留和proxy与db之间的传输总时间prT(proxy relay time)作为衡量proxy服务质量的重要指标。如果实例ins1的请求在proxy节点的prT>=ProxyRelayTimeLimit,对于该请求,proxy代价时间过长。
由于有现成的生产集群上proxy节点和db节点都安装了tcprt内核驱动,我们打算proxy节点和db节点的tcprt数据来做类似的工作。
如图是ins1,ins2的SQL请求在proxy节点和db节点上的链路活动图。 ins1实例的一次请求时间rt1= t0+ t1 + t2 + t3 + t4 + t5 + t6, 其中t0是SQL请求从client端到proxy端传输的时间, t1是接收client端发的请求到向db端发送所需的时间,t2是proxy1到db1的网络链路时间,t3是db1的本地处理时间,t4是db1到proxy1的网络链路时间,t5是接受到db1端发的SQL应答到向client端发送的时间,t6是应答从proxy端到client端的传输时间。而(ins1, *, proxy1)的tcprt处理时间proxyInTime1=t1+t2+t3+t4+t5, (ins1, proxy1, *)的tcprt处理时间proxyOutTime1=t3,通过计算diff1 = proxyInTime1-proxyOutTime1 = t1+t2+t3+t5, 可以算出SQL请求在proxy节点中停留和proxy与db之间网络传输的总时间prT。
由于网络延迟正常情况下在内部网络中比较稳定的,如果diff1值变大了,多出的部分,往往是proxy节点贡献的,因此我们可以大致通过diff1估计实例ins1在proxy1停留的的大致时间。对于经过proxy的每个实例,如果prT>=ProxyRelayTimeLimit,则认为prT过大。我们算出proxy1上的各个实例的prT值,得到prT值过长的实例数量占proxy1上活跃实例数的比例r。我们可以根据r的突升和范围来判断proxy及下游网络链路是否对用户形成影响。 为了判断r的突升,首先,利用上面判断db主机异常的方法,回归出这个数据集的柯西概率分布D。
因为比例r的取值是0~1,而柯西概率分布D的自变量x范围是负无穷到正无穷。我们需要对原来的比率r做转化让他的范围扩充到正负无穷。通过映射函数,我们求出这一时刻下该指标的柯西概率分布的CDF(x’),由于r越小表明proxy越健康,所以只有当r>M是才会进一步判断proxy是否异常,如果CDF(x’)非常大比如:大于99.8%, 说明比率r突然明显上升,需要引起注意。为了减少误判,我们还要判断出r值的突升和突降需要落到警戒范围。因此需要一个必要条件:r值的绝对值至少为20%。
不过如果r本身就很大的话比如:由于proxy升级到了一个有bug的版本上,所有的实例从新版本上线后就一直慢,由于数据集的中位数变成了100%,上面的方法就无法判断了。我们还要再加个异常的充分条件,那就是:如果r>MaxRatio(比如:80%),就判断为异常。使用回归分布的方法适合当r发生巨变时来判断异常,主要是为了找到proxy的急性病;而后加的判断异常的充分条件适用于当r从一开始就不正常,或者r缓慢变成不正常的情况,是为了找到proxy的慢性病。
网络异常检测
为了容忍交换机单点故障,每个节点(代理节点和数据库节点)会上联到一对TOR交换机上。TOR中的高丢包率会导致大量TCP数据包重新传输,并导致查询延迟变高、失败连接增加,从而导致用户数据库性能下降。因此,在短时间内,识别网络故障并定位异常网络设备,通过修复或者更换的方式去解决网络异常是至关重要的。 通过TcpRT采集的TCP连接上乱序数据包,重传数据包,RTT抖动和RST数据包的数量,可用于网络故障排查。
如上图所示,在分布式体系结构中,每个节点相互通信,比如,Proxy节点到数据库节点的请求重定向。我们绘制一个二分图来表示节点之间的关系,顶点是Proxy节点和正在通信的数据库节点,如果两个节点存在相互通信,那么这两个节点存在一条链接边。用虚线标记的链接表示在两个节点之间观察到大量网络异常事件(无序,重传等),否则我们使用实线代替。
根据主机到TOR交换机对的连接信息,通过把主机节点替换成相应的TOR交换机对,我们将上图b转换成上图c。直观上,相连虚线数越多的顶点异常可能性越高。因此,我们定义公式count^1.5/total来衡量TOR交换机对发生异常的概率,其中count表示虚线数,total表示线(虚+实)数。count^1.5/total值越大,该TOR交换机对越有可能是异常。
小结
到目前为止,TcpRT以每秒采集2千万条原始trace数据、每天后台处理百亿吞吐数据、秒级检测异常的卓越性能在阿里云持续稳定运行三年。今年TcpRT的监控能力将包装成云产品开放给RDS客户,给客户提供更好的数据库与应用诊断能力。在技术上,我们也在基于TcpRT开发更多的算法,发掘更多的异常行为。
【本文为专栏作者“阿里巴巴官方技术”原创稿件,转载请联系原作者】
分享文章:从没想到监控可以这么做!阿里云RDS智能诊断系统首次公开
URL地址:http://www.csdahua.cn/qtweb/news16/511266.html
网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网