- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我遵循了一些关于如何设置 HTTP 服务器的教程,并在本地 Kubernetes 集群中对其进行了测试(使用 minikube
)。
我还从我发现的一些示例中实现了正常关闭,并预计 Kubernetes 滚动重启不会造成停机。
为了验证这一点,我开始执行负载测试(使用 Apache Benchmark ,通过运行 ab -n 100000 -c 20 <addr>
)并运行 kubectl rollout restart
在基准测试期间,但 ab
执行滚动重启后立即停止运行。
这是我当前的项目设置:
Dockerfile
FROM golang:1.13.4-alpine3.10
RUN mkdir /app
ADD . /app
WORKDIR /app
RUN go build -o main src/main.go
CMD ["/app/main"]
package main
import (
"context"
"fmt"
"log"
"net/http"
"os"
"os/signal"
"syscall"
"github.com/gorilla/mux"
)
func main() {
srv := &http.Server{
Addr: ":8080",
Handler: NewHTTPServer(),
}
idleConnsClosed := make(chan struct{})
go func() {
sigint := make(chan os.Signal, 1)
signal.Notify(sigint, os.Interrupt, syscall.SIGTERM, syscall.SIGINT)
<-sigint
// We received an interrupt signal, shut down.
if err := srv.Shutdown(context.Background()); err != nil {
// Error from closing listeners, or context timeout:
log.Printf("HTTP server Shutdown: %v", err)
}
close(idleConnsClosed)
}()
log.Printf("Starting HTTP server")
running = true
if err := srv.ListenAndServe(); err != http.ErrServerClosed {
// Error starting or closing listener:
log.Fatalf("HTTP server ListenAndServe: %v", err)
}
<-idleConnsClosed
}
func NewHTTPServer() http.Handler {
r := mux.NewRouter()
// Ping
r.HandleFunc("/", handler)
return r
}
func handler(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "Hello World!")
}
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: myapp
name: myapp
spec:
replicas: 10
selector:
matchLabels:
app: myapp
strategy:
rollingUpdate:
maxSurge: 1
maxUnavailable: 5
type: RollingUpdate
template:
metadata:
labels:
app: myapp
spec:
containers:
- name: myapp
image: dickster/graceful-shutdown-test:latest
imagePullPolicy: Never
ports:
- containerPort: 8080
apiVersion: v1
kind: Service
metadata:
labels:
app: myapp
name: myapp
spec:
ports:
- port: 8080
protocol: TCP
selector:
app: myapp
sessionAffinity: None
type: NodePort
rollingUpdate
策略,应该至少有五个正在运行的 Pod 来处理传入的请求,但是
ab
以
apr_socket_recv: Connection reset by peer (54)
退出错误。我也尝试添加就绪/活跃度探测器,但没有运气。我怀疑这里也不需要它们。
最佳答案
为了在不停机的情况下工作,您需要让 Pod 停止接收新连接,同时允许 Pod 优雅地完成处理当前连接。这意味着 pod 需要运行,但尚未准备好,因此不会向其发送新请求。
您的服务将使用您配置的标签选择器(我假设 app: myapp
)匹配所有 pod,并将使用任何处于就绪状态的 pod 作为可能的后端。只要 Pod 通过 readinessProbe,它就会被标记为就绪。由于您没有配置探针,因此只要 pod 正在运行,它的状态就会默认为就绪。
只需一个 readinessProbe配置将有很大帮助,但不会提供 100% 的正常运行时间,这将需要在您的代码中进行一些调整,以导致 readinessProbe 失败(因此不会发送新请求),同时容器正常完成当前连接。
编辑:根据@Thomas Jungblut 提到的,消除网络服务器错误的很大一部分是应用程序如何处理 SIGTERM。当 pod 处于终止状态时,它将不再通过服务接收请求。在此阶段,您的网络服务器需要配置为正常完成和终止连接,而不是突然停止并终止请求。
请注意,这是在应用程序本身中配置的,而不是 k8s 设置。只要网络服务器优雅地排空连接并且您的 pod 规范包含足够长的 gracefulTerminationPeriod 以允许网络服务器排空,您应该基本上看不到任何错误。尽管这仍然不能保证 100% 的正常运行时间,尤其是在使用 ab 轰炸服务时。
关于docker - 在 Kubernetes 推出期间在 HTTP 服务器上测试正常关闭,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58751354/
我有一个传统上作为 MSI 文件提供的产品。它由我们的各种客户通过某种形式的 SMS 推送部署到数千个桌面。我们用来创建这些安装程序的软件正在使用中,我们正在寻找替换它。由于我们支持许多操作系统,我们
我们有一个使用 Gradle/Android Studio 构建的 Android 应用程序,并使用 JaCoCo 为我们的单元测试生成代码覆盖率报告;这很好用。我们也有兴趣能够为手动测试生成覆盖率报
我正在寻找有关数据库脚本修改的最佳实践的见解,这些修改与软件系统的其他代码更改一起进行。 我曾经在一家公司工作,该公司坚持每次推出都要准备好回滚,以防出现问题。这听起来很合理,但在我看来,通过脚本部署
我使用下面的代码通过 Intent 启动 Twitter,但它不起作用。我的手机上安装了 twitter 应用程序。 Intent shareIntent = new Intent(android.c
我用 1Password存储我的密码,以及 fluid.app创建一些特定于站点的应用程序/浏览器 虽然 1Password与 Chrome、Safari 和其他浏览器很好地集成,它不能很好地与 Fl
我如何启动 google maps transit get directions,以便当我按坐标或地点名称输入源和目的地时,google maps 会建议公交(公交路线)方向。这是骑自行车的代码,但我
我已经为 Android 应用程序开发了一个 SDK。我们有很多客户在应用程序中使用这个 SDK。现在我已经更新了我的 SDK。我正在寻找一种方法可以在应用程序中反射(reflect)这些更改,而无需
我有一个带 child 的 flex 容器,每个 child 都有一些内容(没有固定宽度的),并且在一个 child 上我使用 flex-grow: 1;将其填充到剩余的可用空间。 有时,正在成长的
我们即将推出使用 java web start 构建的客户端服务器应用程序。我们的主服务器位于欧洲的一个国家,第一天就会有许多用户从世界各地下载我们的客户端。由于客户端应用程序相当大(MB),因此我们
我已经尝试了很多组合,而这个是最接近我需要的组合。 我需要的是一个图像 (img1),上面有另一个(较小的)图像 (img2),当我翻转 img1 时,我希望它淡入(并在滚出时淡出),而 img2 保
北京时间 1 月 21 日晚间消息,知情人士今日称,苹果公司的首款 VR/AR(虚拟现实 / 增强现实)头显将是一款昂贵的小众产品,有望于 2022 年推出。这款产品主要为了将来更主流的 “AR
搜索 Apple Pay/Passkit/Wallet 文档,我发现代码示例很少,文档也很差。我们尝试提供我们已配置的支付通行证,而不仅仅是使用 openPaymentSetup() 启动钱包。 根据
我需要这个用于 API11 之前的项目。 我正在寻找一种从右侧添加 View 的方法,这将导致当前全屏 View 向左移出,以便在屏幕中显示新 View 。 新 View 大约是屏幕宽度的 1/3,这
Electron v1.8.2 Spectron v3.8.0 添加光谱测试,但即使是非常基本的测试也会失败。 尝试运行: const { Application } = require('spect
这是我最简单的问题。让我试着把它弄清楚。 我有一个 div,在本例中称为 "testdiv",它附加了一个类名 "menulink"。 div 中有一个链接。当我滚动链接时,我希望 div 类更改为
我想将 li 元素排列在任意数量的列中。 Yellow Block Red Block Green
我试图将 child 推到 parent 的盒子外面,但是,似乎 parent 只是在长度上扩展并获得一个滚动条,所以 child 留在里面,并且在 parent 外面不可见。 .parent {
Google Play 游戏服务:SIGN_IN 状态:ERROR_NOT_AUTHORIZED,推出 Beta 版时。还有这个:- W Auth : [GetToken] GetToken f
我正在尝试使用 Jetpack Compose,但我对 Row 的行为感到困惑。我在图标按钮旁边有一个文本,我希望图标按钮锚定在最小宽度为 48dp 的一侧,并在其周围环绕文本。像这样: 但是文本没有
我从第四行的字符 1 中收到上述错误...不知道是什么抛出了它?我确信这是我看不到的简单事情...... out = out + "Select Box Information The name of
我是一名优秀的程序员,十分优秀!