当前位置: 首页 > news >正文

网站开发与建设课程网络服务投诉

网站开发与建设课程,网络服务投诉,网站建设从建立服务器开始,广告设计与制作主修课程最后更新时间 2023-01-24. 背景 笔者所在公司技术栈为 Golang PHP,目前部分项目已经逐步转 Go 语言重构,部分 PHP 业务短时间无法用 Go 重写。 相比 Go 语言,互联网公司常见的 Nginx PHP-FPM 模式,经常会出现性能问题—— 特…

最后更新时间 2023-01-24.

背景

笔者所在公司技术栈为 Golang + PHP,目前部分项目已经逐步转 Go 语言重构,部分 PHP 业务短时间无法用 Go 重写。

相比 Go 语言,互联网公司常见的 Nginx + PHP-FPM 模式,经常会出现性能问题——

特别是我们的活动业务,尽管底层用了鸟哥的 Yaf 框架,

但由于业务逻辑繁重,即使框架层面上完全零损耗,常常支撑不了流量高峰。

一旦某个时间段开启活动,虚拟机的扩容真的非常痛苦。

SRE、开发、QA 三方经常需要因为某个运营活动的进行,提前压测预估容量。

目前活动业务已经逐步用 Go 语言改造,此处不具体展开,防止跑题哈哈。

正因为 PHP 虚拟机模式,每次扩容需要流量剔除、克隆、操作负载均衡、验证流量等等,

推进 PHP 容器化就显得格外重要。

公司在去年年中,已经开始进行 PHP 容器化,不过由于项目优先级以及人力原因,进度较为迟缓。

事情经过

  1. 某项目进行 PHP 容器化改造,切换少许流量到容器中
  2. 逐步加大灰度流量
  3. 某一天开发上线新功能,发现滚动部署过程中存在 502 错误

分析原因

nginx 发生了 502,很多时候是后端,也就是 php-fpm 不在工作。

我们的 PHP 业务的 Pod,由以下 5 个容器组成:

  • nginx
  • php-fpm
  • metric(监控)
  • jaeger(链路追踪)
  • log(日志收集)

滚动时存在关闭旧 Pod 启动新 Pod 的过程,借助 K8s 官方文档 的描述,我们看看 Pod 结束的一个例子:

  1. 你使用 kubectl 工具手动删除某个特定的 Pod,而该 Pod 的体面终止限期是默认值(30 秒)。
  2. API 服务器中的 Pod 对象被更新,记录涵盖体面终止限期在内 Pod 的最终死期,超出所计算时间点则认为 Pod 已死(dead)。 如果你使用 kubectl describe 来查验你正在删除的 Pod,该 Pod 会显示为 “Terminating” (正在终止)。 在 Pod 运行所在的节点上:kubelet 一旦看到 Pod 被标记为正在终止(已经设置了体面终止限期),kubelet 即开始本地的 Pod 关闭过程。
  3. 在 kubelet 启动体面关闭逻辑的同时,控制面会将关闭的 Pod 从对应的 EndpointSlice(和 Endpoints)对象中移除,过滤条件是 Pod 被对应的服务以某 选择算符选定。 ReplicaSet 和其他工作负载资源不再将关闭进程中的 Pod 视为合法的、能够提供服务的副本。 关闭动作很慢的 Pod 也无法继续处理请求数据, 因为负载均衡器(例如服务代理)已经在终止宽限期开始的时候将其从端点列表中移除。
  4. 超出终止宽限期限时,kubelet 会触发强制关闭过程。容器运行时会向 Pod 中所有容器内仍在运行的进程发送 SIGKILL 信号。 kubelet 也会清理隐藏的 pause 容器,如果容器运行时使用了这种容器的话。
  5. kubelet 触发强制从 API 服务器上删除 Pod 对象的逻辑,并将体面终止限期设置为 0 (这意味着马上删除)。
  6. API 服务器删除 Pod 的 API 对象,从任何客户端都无法再看到该对象。

通常情况下,容器运行时会发送一个 TERM 信号到每个容器中的主进程。很多容器运行时都能够注意到容器镜像中 STOPSIGNAL 的值,并发送该信号而不是 TERM。一旦超出了体面终止限期,容器运行时会向所有剩余进程发送 KILL 信号,之后 Pod 就会被从 API 服务器上移除。 如果 kubelet 或者容器运行时的管理服务在等待进程终止期间被重启,集群会从头开始重试,赋予 Pod 完整的体面终止限期。

所以,我们可以发现:

  1. nginx、php-fpm 收到 TERM 信号后,不做请求的优雅处理,直接强制退出了!强制退出的原因,可以移步这俩文档:nginx - http://nginx.org/en/docs/control.html、php-fpm - https://linux.die.net/man/8/php-fpm;
  2. 参考上面的第 3 点,在容器运行时发送 TERM 信号后,也同时移除 endpoint,此处不是串行的。一旦 endpoint 移除的时间晚了,流量就会剔不干净,到达了 nginx 后,php-fpm 进程已经退出从而导致 502 的产生。

解决办法

有了上面的分析,解决起来就方便多了!

查看上面文档,我们可以了解到,nginx 和 php-fpm 喜欢 QUIT 信号,均可做到 graceful shutdown。

只需要在 Dockerfile 指定 STOPSIGNAL SIGQUIT 即可。

但我记得之前封装的 php-fpm 镜像使用的是社区维护版本,应该加上了才对。

而我看了线上的 Dockerfile,nginx 使用的是社区维护的,已经配置了 STOPSIGNAL SIGQUIT,没问题!

但是 PHP 由于之前的 alpine linux 因为监控扩展、链路追踪扩展编译环境的原因,使用了 CentOS 镜像。

镜像的来源都是自己打包的,并没有指定退出信号!

加上了之后,发现不会有 502 了!

至此,问题解决。

延伸思考

本来文章到此结束,突然想到线上的 Go 服务会不会有同样的问题?

想了一下,也好办!

要么自行处理 TERM 信号,做好优雅退出的姿势!

要么学 nginx,也用 QUIT 信号并做好优雅退出处理,Dockerfile 指定 STOPSIGNAL。

至于怎么处理信号,Go 实现起来非常舒服:

ch := make(chan os.Signal, 1)
signal.Notify(ch, syscall.SIGTERM, syscall.SIGQUIT)
<-ch
// 收到信号了(还可以根据信号类型做不同的处理逻辑),自行处理剩余任务实现优雅退出。

嗯,channel 大法好。


文章来源于本人博客,发布于 2022-09-03,原文链接:https://imlht.com/archives/398/

http://www.yayakq.cn/news/69137/

相关文章:

  • 义乌网站制作公司公司网站建设长春
  • 游戏门户网站模板如果做夺宝网站
  • 网站轮播动态图如何做s.w.g wordpress
  • 北京建设工程交易网站官网设计网页用什么语言
  • 陕西省建设厅的网站桂阳网页定制
  • 网站开发 报价广西网站建设公司招聘
  • 哈尔滨做网站快速优化seo软件推广方法
  • 营销型网站上海制作百度邮箱注册申请免费注册
  • 东营网站关键字优化建设内网网站流程
  • 二七网建站网站建设的主要特征
  • 企业整站推广成都微信公司地址在哪里
  • 网站空间怎么做3d建模软件推荐
  • 网站的布局龙华网站建设设计制作公司
  • 网站的界面设计网站的免费空间是什么
  • 做坑人网站二维码免费婚纱摄影网站模板
  • 360的网站怎么做制造业人才网
  • 小型网站开发要多少钱wordpress wiki模板
  • 什么软件做网站好些深圳插画设计公司
  • 网站网站服务器温州最好的网站建设公司
  • 山西省建设局官方网站慈溪网站制作哪家最便宜
  • 万网可以做网站吗做网站哪个语言强
  • 织梦做双语网站影视公司名字取名
  • 高大上强企业网站表白网页生成器软件
  • 邯郸建站公司龙港网
  • 海外房地产网站建设id创建网站
  • 网站主页图片尺寸网址制作网站
  • 呼和浩特企业网站制作1小时赚8000元游戏
  • 陕西省住房和城乡建设厅网站南宁网络广播电视台
  • 苏州代做淘宝网站江宁网站制作
  • 淘宝客网站根目录wordpress改域名修改哪里