记录一次莫名奇妙的跨域502(badgateway)错误

这里图片加载不了,原文请访问:原文链接

公司的项目,这几天添加了一个统计功能, 本地测试没太大问题,上线后有一个问题,具体现象描述如下:

  • 统计首页接口大约有5-6个,也就是同时需要访问6个接口拿首页数据
  • 这些接口浏览器访问会随机的,偶尔的出现跨域错误,提示CROS。但偶尔又能全部正常访问
  • 本地测试没有问题CROS问题

基于以上描述,首先想到可能nginx会有问题,因为线上和测试环境就只加了一层转发。而测试环境没有这个问题, 于是直接暴露线上服务端口访问,故障依旧。

这里没太想明白为啥是cros错误,对比了下options请求响应,都是正常请求响应的,响应数据也是正确的。所以这个问题暂记下

但是本地启动为什么没有CROS问题呢,看了下线上启动脚本,线上启动有添加堆大小限制的启动命令(限制为512MB)。本地同步添加,果然问题就在本地复现了。

访问浏览器提示跨域时,服务后台日志如下:

java 复制代码
14:50:41.003 [http-nio-8985-exec-60] ERROR o.a.c.h.Http11NioProtocol - [log,175] - Failed to complete processing of a request
java.lang.OutOfMemoryError: Java heap space
14:50:41.210 [http-nio-8985-exec-58] ERROR o.a.c.h.Http11NioProtocol - [log,175] - Failed to complete processing of a request
java.lang.OutOfMemoryError: Java heap space
14:50:41.211 [http-nio-8985-exec-60] ERROR o.a.t.u.n.NioEndpoint - [log,175] - Error running socket processor
java.lang.NullPointerException: Cannot invoke "java.nio.ByteBuffer.limit(int)" because "this.byteBuffer" is null
	at org.apache.coyote.http11.Http11InputBuffer.recycle(Http11InputBuffer.java:262)
	at org.apache.coyote.http11.Http11Processor.recycle(Http11Processor.java:1418)
	at org.apache.coyote.AbstractProtocol$ConnectionHandler.release(AbstractProtocol.java:1085)
	at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:1060)
	at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1744)
	at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:52)
	at org.apache.tomcat.util.threads.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1191)
	at org.apache.tomcat.util.threads.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:659)
	at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:63)
	at java.base/java.lang.Thread.run(Thread.java:842)
14:50:41.211 [http-nio-8985-exec-58] ERROR o.a.t.u.n.NioEndpoint - [log,175] - Error running socket processor
java.lang.NullPointerException: Cannot invoke "java.nio.ByteBuffer.limit(int)" because "this.byteBuffer" is null
	at org.apache.coyote.http11.Http11InputBuffer.recycle(Http11InputBuffer.java:262)
	at org.apache.coyote.http11.Http11Processor.recycle(Http11Processor.java:1418)
	at org.apache.coyote.AbstractProtocol$ConnectionHandler.release(AbstractProtocol.java:1085)
	at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:1060)
	at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1744)
	at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:52)
	at org.apache.tomcat.util.threads.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1191)
	at org.apache.tomcat.util.threads.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:659)
	at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:63)
	at java.base/java.lang.Thread.run(Thread.java:842)

堆溢出,我擦。第一时间想的就是哪里内存泄漏了? 导出堆空间,分析下内存。 这里可以看到有一个对象堆占用特别多

接下来看下堆中各个对象占用情况:

这里可以看到 bytes 数组占用太大,有点不正常。 继续跟进去看看谁在引用这些数组。

把这个字节数组导出来,看看是写什么。 这里面只有http请求头,而且后面很多都是空。 大小125MB.

继续跟踪字节数组被谁持有

这里可以看到字节数组被一个http11InputBuffer对象的inputBuffer持有, 而inputBuffer里面headerBufferSize属性,这个属性值等于131072000 也就是前面的 125MB

这个headerbuffer怎么这么大啊,谷歌搜索下,这个buffer是用来装http请求的header数据内容,也就是说http每次请求就需要申请125MB的空间,怪不得请求会报堆栈溢出。

再看了下文章,这个属性可以通过 max-http-request-header-size 属性进行配置。

所以接下来就简单了,项目中搜索这个配置。有人配置为:max-http-request-header-size: 131072000 换算下来 1GB。 我的天

这也太大了,调整为10MB吧。 重启项目,问题消失。

结语就是,绕了一大圈,这个跨域错误居然是内存泄漏,这个提示真的有方向性误导啊。哈哈

相关推荐
一切尽在,你来5 分钟前
C++ 零基础教程 - 第 6 讲 常用运算符教程
开发语言·c++
泉-java6 分钟前
第56条:为所有导出的API元素编写文档注释 《Effective Java》
java·开发语言
0思必得07 分钟前
[Web自动化] Selenium处理滚动条
前端·爬虫·python·selenium·自动化
沈浩(种子思维作者)20 分钟前
系统要活起来就必须开放包容去中心化
人工智能·python·flask·量子计算
2301_7903009625 分钟前
Python数据库操作:SQLAlchemy ORM指南
jvm·数据库·python
weixin_4997715526 分钟前
C++中的组合模式
开发语言·c++·算法
初级代码游戏26 分钟前
套路化编程 C# winform 自适应缩放布局
开发语言·c#·winform·自动布局·自动缩放
_waylau30 分钟前
鸿蒙架构师修炼之道-架构师的职责是什么?
开发语言·华为·harmonyos·鸿蒙
2的n次方_41 分钟前
CANN Ascend C 编程语言深度解析:异构并行架构、显式存储层级与指令级精细化控制机制
c语言·开发语言·架构
m0_7369191041 分钟前
用Pandas处理时间序列数据(Time Series)
jvm·数据库·python