HTTP CODE 状态码500|502|504分析

给别人轻松讲明白一个问题,才能算自己真正了解这个问题。

Origin Header 头让我熟悉了一次sheme

从HTTP的头Origin说起,想起之前客户端定义scheme,因为不了解,问了开发的同事“scheme是什么?”反正我当时是不明白他们讲的。

在了解HTTP Origin语法的时候,我其实才真正明白:scheme 指请求所使用的协议,通常是HTTP、HTTPS或者其他。

Origin: <scheme> "://" <host> [":" <port>]

Origin表示请求来至哪个站点。在WebSocket通信的时候,明确指明要校验这个参数。

状态码预览

500

服务器内部错误。比如:服务端处理出现异常。同时,在PHP错误日志中可以查看异常发生的调用栈信息。

502

作为网关或者代理工作的服务器尝试执行请求时,从上游服务器接收到无效的响应。比如nginxphp-fpm接收到了不完整的response数据。

比如:服务端尝试连接mysql,但长时间链接不上,就会返回502错误。

可以浏览一些具体的文章:

  1. http 502 和 504 的区别
  2. Nginx一次奇怪的502 报错探究

504

网关超时。为了完成您的 HTTP 请求,该服务器访问一个上游服务器,但没得到及时的响应

比如:nginx超过了自己设置的超时时间,不等待php-fpm的返回结果,直接给客户端返回504错误。但是此时php-fpm依然还在处理请求(在没有超出自己的超时时间的情况下)。

当Go服务发生Panic时的状态码

500

首先我们启动一个Go的原生服务,然后在服务前设置Nginx作为服务的代理。这里配置中配置 Nginx 到 Go 服务使用长链接。

直接在代码中手动调用 panic,然后请求接口,观察接口的状态码返回值。

r.POST("/panic", func(context *gin.Context) {
	panic("this panic")
})

使用 curl 命令请求接口,同时,打印接口的详细信息。在程序处理请求的过程中发生panic的话,服务返回的500。这里,建议加上 -v 参数。

➜  ~ curl -v -X POST  http://127.0.0.1:8081/panic
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to 127.0.0.1 (127.0.0.1) port 8081 (#0)
> POST /panic HTTP/1.1
> Host: 127.0.0.1:8081
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 500 Internal Server Error
< Date: Sat, 03 Oct 2020 02:01:14 GMT
< Content-Length: 0
<
* Connection #0 to host 127.0.0.1 left intact
* Closing connection 0

通过网络的请求情况,我们来佐证一下这个过程,对这个过程抓包分析。图中标志①的地方是因为程序中发生了 panic 导致,可以看出,虽然程序发生了panic,但是整个链接(63509端口到8001端口)并没有因为panic而断开;之后是②部分,服务端主动发起了3次心跳(默认间隔15秒);最后,在③阶段,客户端主动断开了链接。

在这里插入图片描述

如果对 panic的信息进行了 recover 呢,我们调整一下代码,在 router 前加一个中间件来专门 recover 请求中的 panic信息:

func MiddleRecover(ctx *gin.Context) {
	defer func() {
		if r := recover(); r != nil {
			log.Println("panic occur", r)
		}
	}()
	
	ctx.Next()
}

我们继续使用curl 请求 /panic 接口,接口正常返回了,状态码是200。

➜  ~ curl -v -X POST  http://127.0.0.1:8081/panic
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to 127.0.0.1 (127.0.0.1) port 8081 (#0)
> POST /panic HTTP/1.1
> Host: 127.0.0.1:8081
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sat, 03 Oct 2020 02:08:39 GMT
< Content-Length: 0
<
* Connection #0 to host 127.0.0.1 left intact
* Closing connection 0

502

那么,如何触发502的状态码呢,通过前面对状态码的介绍:“从上游服务器接收到无效的响应”。从网络传输来看,在传输的过程中,如果服务端主动断开链接,就会导致502。服务端怎么样才会主动断开链接呢,有一种比较常见的情况,就是整个服务 crash。

我们在新建一个请求,用来导致整个服务发生 panic,这里采用 map 并发读写的panic,因为这个panic 是runtime层发生的panic,整个服务只能crash,不能被recover。代码中模拟了 map 的并发读写操作。注意,并不一定每次都能触发panic

result := make(map[int]int)
r.POST("/crash_panic", func(context *gin.Context) {
	go func() {
		for {
			result[1] = 1
		}
	}()
	_ = result[1]
	context.JSON(http.StatusOK, "")
})

我们还是通过 curl 来请求几次,看一下结果。在系统崩溃时,返回的状态码确实是502。

➜  nginx curl -v -X POST http://127.0.0.1:1144/crash_panic
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to 127.0.0.1 (127.0.0.1) port 1144 (#0)
> POST /crash_panic HTTP/1.1
> Host: 127.0.0.1:1144
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 502 Bad Gateway
< Server: nginx/1.19.1
< Date: Sat, 03 Oct 2020 08:21:45 GMT
< Content-Type: text/html
< Content-Length: 157
< Connection: keep-alive
<
<html>
<head><title>502 Bad Gateway</title></head>
<body>
<center><h1>502 Bad Gateway</h1></center>
<hr><center>nginx/1.19.1</center>
</body>
</html>
* Connection #0 to host 127.0.0.1 left intact
* Closing connection 0

依旧通过网络抓包来看一下连接断开的过程。从图中的②可以看出,服务端主动断开了网络的连接。因为服务Down机了,在Down机的时候会主动断开服务端的连接。

在这里插入图片描述

504

最后,我们再来看一下504,没有得到及时的响应。怎么样才算是网络超时呢,看一下实验的 nginx 配置文件。

upstream mysvr {
        server 127.0.0.1:8081;
        keepalive 300;       #这个很重要!
}

server {

        keepalive_requests 120; #单连接请求上限次数。

        listen  1144;      #监听端口
        location / {       #请求的url过滤,正则匹配,~为区分大小写,~*为不区分大小写。
                proxy_pass  http://mysvr;  #请求转向mysvr 定义的服务器列表
                proxy_set_header Host $host;
                proxy_set_header Connection "";   #设置Connection为长连接(默认为no)
                proxy_connect_timeout 30;         #与upstream server的连接超时时间
                proxy_read_timeout 60s;           #nginx会等待多长时间来获得请求的响应
                proxy_send_timeout 12s;           #发送请求给upstream服务器的超时时间
                proxy_http_version 1.1;
        }
}

对配置文件做一下改动,将proxy_read_timeout从60秒调整到3秒,然后我在程序中sleep 3秒看一下效果:

r.POST("/timeout", func(context *gin.Context) {
	time.Sleep(time.Second * 4)
	context.JSON(http.StatusOK, "")
})

下面是 curl 请求的过程以及网络抓包的截图:

➜  nginx curl -v -X POST http://127.0.0.1:1144/timeout
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to 127.0.0.1 (127.0.0.1) port 1144 (#0)
> POST /timeout HTTP/1.1
> Host: 127.0.0.1:1144
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 504 Gateway Time-out
< Server: nginx/1.19.1
< Date: Sat, 03 Oct 2020 08:33:54 GMT
< Content-Type: text/html
< Content-Length: 167
< Connection: keep-alive
<
<html>
<head><title>504 Gateway Time-out</title></head>
<body>
<center><h1>504 Gateway Time-out</h1></center>
<hr><center>nginx/1.19.1</center>
</body>
</html>
* Connection #0 to host 127.0.0.1 left intact
* Closing connection 0

通过第二列的时间项,我们可以计算出,大概在3秒后,nginx主动断开了链接,然后客户端收到了504的状态码。
在这里插入图片描述

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
基于C语言的词法分析器的生成程序 基于C语言的词法分析器的生成程序是一个能够根据特定的词法规则生成词法分析器的程序。这种程序通常接受一组正则表达式定义的词法规则,并输出一个能够识别这些规则的C语言程序。下面是一个简化的项目介绍,描述了如何实现这样一个生成程序。 ### 项目介绍: **目标**:创建一个程序,能够生成用于C语言源代码的词法分析器。 **主要任务**: 1. **解析词法规则**: - 定义一种格式,用于描述词法规则和正则表达式。 - 编写解析器,将这种格式的文件解析为内部数据结构。 2. **生成NFA**: - 根据正则表达式生成非确定性有限自动机(NFA)。 - 实现NFA的状态和转换函数。 3. **NFA确定化为DFA**: - 实现子集构造算法,将NFA转换为确定性有限自动机(DFA)。 - 实现DFA的状态和转换函数。 4. **DFA最小化**: - 实现DFA最小化算法,如Hopcroft算法。 - 简化DFA的状态和转换函数。 5. **生成C语言代码**: - 根据最小化的DFA,生成C语言词法分析器的源代码。 - 生成代码应该能够读取源代码,并输出词法单元。 **技术要求**: - 熟悉C编程语言。 - 了解编译原理中的词法分析概念。 - 熟悉有限自动机理论和正则表达式的使用。 **开发工具**: - C编译器,如GCC、Clang或MSVC。 - 代码编辑器或IDE,如Visual Studio、Code::Blocks或Eclipse。 ### 适合人员: - 计算机科学或相关领域的学生:此项目能够帮助他们实践编译原理和C编程知识。 - 软件开发者:特别是那些对编译器和解释器如何工作感兴趣的程序员。 - 语言处理领域的研究者:此项目可以作为自然语言处理和编译技术的一个研究起点。 ### 额外建议: - 从一个简单的词法规则集开始,逐步增加支持的复杂性。 - 使用单元测试和集成测试来验证生成器的正确性。 - 编写详细的文档,记录设计决策、实现细节和测试结果。 - 考虑使用版本控制系统(如Git)来管理项目代码。 通过实现这样一个生成程序,开发者不仅能够深入理解编译器构建的过程,还能够提高对程序语言的理解和C编程技能。此外,这个项目对于希望进入编译器设计、程序分析和代码生成等领域的人来说,是一个很好的实践机会。
词法分析程序生成器实现将正则表达式、NFA、DFA、DFA最小化词法分析程序 词法分析程序生成器是一个工具,它能够根据给定的正则表达式自动生成词法分析器。这个生成器通常包括以下几个步骤:正则表达式的转换、NFA(非确定性有限自动机)的构建、DFA(确定性有限自动机)的构建、DFA的最小化,以及最终生成词法分析程序。以下是一个基于这个概念的项目介绍。 ### 项目介绍: **目标**:实现一个能够将正则表达式转换为词法分析程序的生成器。 **主要任务**: 1. **正则表达式解析**: - 实现一个解析器,用于解析复杂的正则表达式。 - 将正则表达式转换为内部表示,如抽象语法树(AST)。 2. **NFA构建**: - 根据正则表达式的内部表示,构建对应的NFA。 - 实现NFA的状态和转换函数。 3. **DFA构建**: - 将NFA转换为等价的DFA。 - 实现DFA的状态和转换函数。 4. **DFA最小化**: - 实现DFA的最小化算法,如Hopcroft算法。 - 简化DFA的状态和转换函数。 5. **词法分析程序生成**: - 根据最小化的DFA,生成可执行的词法分析程序代码。 - 生成代码应该能够读取源代码,并输出词法单元。 **技术要求**: - 熟悉正则表达式、有限自动机理论和算法。 - 掌握编译原理中的词法分析概念。 - 熟悉至少一种编程语言(如Java、C++、Python等)。 **开发工具**: - 代码编辑器或IDE(例如Visual Studio Code, Eclipse, IntelliJ IDEA等)。 - 编程语言(Java、C++、Python等)。 ### 适合人员: - **计算机科学或相关领域的学生**:此项目能够帮助他们实践编译原理和理论计算机科学的知识。 - **软件开发者**:特别是那些对编译器设计和自动化工具感兴趣的程序员。 - **研究者**:在编译技术、程序分析或语言处理领域的研究者。 ### 额外建议: - 从简单的正则表达式和NFA开始,逐步增加复杂度。 - 使用单元测试和集成测试来验证生成器的正确性。 - 编写详细的文档,记录设计决策、实现细节和测试结果。 - 考虑使用版本控制系统(如Git)来管理项目代码。 通过这个项目,开发者不仅能够深入理解编译器构建的过程,还能够提高对程序语言的理解和编程技能。此外,这个项目对于希望进入编译器设计、程序分析和代码生成等领域的人来说,是一个很好的实践机会。
要使用curl获取状态码,可以在执行curl_exec后,通过curl_getinfo函数来获取。例如: ``` $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, 'http://www.example.com'); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $response = curl_exec($ch); $status_code = curl_getinfo($ch, CURLINFO_HTTP_CODE); curl_close($ch); ``` 其中,`$ch`是一个curl的句柄,`CURLOPT_URL`设置请求的URL,`CURLOPT_RETURNTRANSFER`设置为true以返回请求的结果,`$response`保存了curl的执行结果,`CURLINFO_HTTP_CODE`是一个常量,用于获取HTTP状态码,`$status_code`保存了获取到的状态码。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [PHP curl 获取响应的状态码的方法](https://download.csdn.net/download/weixin_38739164/14846196)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [通过curl获取HTTP状态返回码](https://blog.csdn.net/weixin_46686835/article/details/113761418)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [curl分析http请求各阶段耗时情况](https://blog.csdn.net/zhaikaiyun/article/details/117467144)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值