通过本文章,可以完成多级缓存架构中的Lua缓存。
一、nginx服务
在docker/docker-compose.yml
中添加nginx服务块。
1 | nginx: |
删除原来docker里的multiCache
项目并停止springboot
应用。
nginx
部分配置如下,监听端口为8080
,并且将请求反向代理至172.30.3.11
,下一小节,将openresty
固定在172.30.3.11
。
1 | upstream nginx-cluster { |
重新启动multiCache
看看nginx前端网页效果。
1 | docker-compose -p multi-cache up -d |
访问http://localhost:8080/item.html?id=10001
查询id=10001商品页
这里是假数据,前端页面会向/api/item/10001
发送数据请求。
二、OpenResty服务
1. 服务块定义
在docker/docker-compose.yml
中添加openresty1
服务块。
1 | openresty1: |
2. 配置修改
前端向后端发送/api/item/10001
请求关于id=10001商品信息。
根据nginx
的配置内容,这个请求首先被nginx
拦截,反向代理到172.30.3.11
(即openresty1
)。
1 | upstream nginx-cluster { |
openresty1
收到的也是/api/item/10001
,同时,openresty
将/api/item/(\d+)
请求代理到指定lua
程序,在lua
程序中完成数据缓存。
因此,openresty
的conf/conf.d/default.conf
如下
1 | upstream tomcat-cluster { |
conf/nginx.conf
在http
块最后添加3行,引入依赖。
1 | #lua 模块 |
3. Lua程序编写
common.lua
被挂载到lualib
,表示可以被其他lua当做库使用,内容如下
1 | -- 创建一个本地缓存对象item_cache |
item.lua
是处理来自形如/api/item/10001
请求的程序,内容如下
1 | -- include |
4. 总结
- 这里
lua
对item
(tb_item表)和stock
(tb_stock表)两个信息都有缓存,并使用cjson
库将两者合并后返回到前端。 - 关于
expire
时效性的问题,如果后台改变了数据,但是openresty
关于此数据的缓存未过期,前端得到的是旧数据
。 - 大致来说openresty = nginx + lua,不仅具有
nginx
反向代理的能力,还能介入lua
程序进行扩展。
三、运行
到此为止,docker-compose.yml应该如下
1 | version: '3.8' |
启动各项服务
1 | docker-compose -p multi-cache up -d |
启动springboot
程序。
四、测试
清空openresty
容器日志。
访问http://localhost:8080/item.html?id=10001
查看openresty
容器日志,可以看到两次commonUtils.read_data
都没有缓存,于是代理到tomcat
,可以看到springboot
日志出现查询相关记录。
1 | 2024-01-12 11:45:53 2024/01/12 03:45:53 [error] 7#7: *1 [lua] common.lua:99: read_data(): redis cache miss, try tomcat, key: item:id:10001, client: 172.30.3.3, server: localhost, request: "GET /api/item/10001 HTTP/1.0", host: "nginx-cluster", referrer: "http://localhost:8080/item.html?id=10001" |
再次访问此网址,强制刷新+禁用浏览器缓存+更换浏览器
间隔超过4s但小于1800s时,日志如下,只出现一次miss。
1 | 2024-01-12 11:48:04 2024/01/12 03:48:04 [error] 7#7: *4 [lua] common.lua:99: read_data(): redis cache miss, try tomcat, key: item:stock:id:10001, client: 172.30.3.3, server: localhost, request: "GET /api/item/10001 HTTP/1.0", host: "nginx-cluster", referrer: "http://localhost:8080/item.html?id=10001" |
再次访问此网址,强制刷新+禁用浏览器缓存+更换浏览器
间隔小于4s,日志如下,未出现miss。
1 | 2024-01-12 11:49:16 172.30.3.3 - - [12/Jan/2024:03:49:16 +0000] "GET /api/item/10001 HTTP/1.0" 200 486 "http://localhost:8080/item.html?id=10001" "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36 Edg/121.0.0.0" |
五、高可用集群
1. openresty
对于openresty
高可用,可以部署多个openresty docker
实例,并在nginx
的docker/nginx/conf/conf.d/default.conf
的upstream nginx-cluster
将多个openresty地址添加进去即可。比如
1 | upstream nginx-cluster { |
多个openresty
无论是conf还是lua都保持一致即可。
并且使用hash $request_uri
负载均衡作为反向代理策略,防止同一请求被多个实例缓存数据。
2. tomcat
对于springboot
程序高可用,也是类似。可以部署多个springboot docker
实例,并在openresty
的docker/openresty1/conf/conf.d/default.conf
的upstream nginx-cluster
将多个springboot地址添加进去即可。比如
1 | upstream tomcat-cluster { |