API Platform多应用环境下Swagger文档缓存冲突问题解析
2025-05-26 01:33:06作者:滕妙奇
api-platform
🕸️ Create REST and GraphQL APIs, scaffold Jamstack webapps, stream changes in real-time.
问题背景
在使用API Platform构建多应用系统时,开发者可能会遇到一个棘手的问题:当系统包含多个独立应用(每个应用都有自己的API文档)时,首次加载各应用的Swagger文档显示正常,但在刷新页面后,所有应用的文档会显示相同内容。
问题现象
在多应用架构中(例如基于Symfony的多内核配置),每个应用都有自己独立的API文档配置。初始状态下:
- 应用A的Swagger文档正确显示A应用的API接口
- 应用B的Swagger文档正确显示B应用的API接口
但在刷新任意一个应用的页面后:
- 所有应用的Swagger文档都会显示相同内容(通常是最后加载的那个应用的API接口)
问题根源
经过深入分析,发现问题的根源在于API Platform的缓存机制。具体表现为:
- API Platform使用
CachedResourceNameCollectionFactory来缓存资源名称集合 - 默认的缓存键生成策略没有考虑应用之间的区分
- 导致不同应用在访问Swagger文档时,可能读取到相同的缓存结果
解决方案
方案一:自定义缓存键生成策略
通过继承并重写CachedResourceNameCollectionFactory类,可以修改缓存键的生成逻辑,确保每个应用有独立的缓存空间:
namespace Shared\ApiPlatform\Metadata\Resource\Factory;
use ApiPlatform\Metadata\Resource\Factory\CachedResourceNameCollectionFactory as BaseCachedResourceNameCollectionFactory;
use Symfony\Component\HttpFoundation\RequestStack;
class CachedResourceNameCollectionFactory extends BaseCachedResourceNameCollectionFactory
{
private $requestStack;
public function __construct($cache, $decorated, RequestStack $requestStack)
{
parent::__construct($cache, $decorated);
$this->requestStack = $requestStack;
}
protected function getCacheKey(): string
{
// 添加主机名作为缓存键的一部分
$request = $this->requestStack->getCurrentRequest();
return $request ? $request->getHost().parent::getCacheKey() : parent::getCacheKey();
}
}
方案二:服务配置覆盖
在服务配置中覆盖默认的缓存工厂服务:
services:
api_platform.metadata.resource.name_collection_factory.cached:
class: Shared\ApiPlatform\Metadata\Resource\Factory\CachedResourceNameCollectionFactory
arguments:
- '@api_platform.cache.metadata.resource'
- '@api_platform.metadata.resource.name_collection_factory.cached.inner'
- '@request_stack'
实现原理
这种解决方案的核心思想是通过在缓存键中加入应用特定的标识(如主机名),确保:
- 不同应用的API文档请求会生成不同的缓存键
- 每个应用独立维护自己的API文档缓存
- 避免了缓存冲突问题
最佳实践建议
- 明确应用边界:在多应用架构中,确保每个应用有明确的域名或路径区分
- 缓存隔离:不仅API文档,其他缓存也应考虑应用隔离
- 环境区分:开发、测试、生产环境的缓存也应相互隔离
- 监控机制:建立缓存命中/失效的监控,确保缓存机制正常工作
总结
API Platform作为强大的API开发框架,在多应用环境下使用时需要注意缓存隔离问题。通过自定义缓存键生成策略,可以有效地解决Swagger文档在多应用间的冲突问题,确保每个应用能够正确显示自己的API文档。这种解决方案不仅适用于Swagger文档,也可以推广到其他需要应用隔离的缓存场景中。
api-platform
🕸️ Create REST and GraphQL APIs, scaffold Jamstack webapps, stream changes in real-time.
登录后查看全文
热门项目推荐
相关项目推荐
atomcodeClaude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed. Get StartedRust0139- DDeepSeek-V4-ProDeepSeek-V4-Pro(总参数 1.6 万亿,激活 49B)面向复杂推理和高级编程任务,在代码竞赛、数学推理、Agent 工作流等场景表现优异,性能接近国际前沿闭源模型。Python00
GLM-5.1GLM-5.1是智谱迄今最智能的旗舰模型,也是目前全球最强的开源模型。GLM-5.1大大提高了代码能力,在完成长程任务方面提升尤为显著。和此前分钟级交互的模型不同,它能够在一次任务中独立、持续工作超过8小时,期间自主规划、执行、自我进化,最终交付完整的工程级成果。Jinja00
MiniCPM-V-4.6这是 MiniCPM-V 系列有史以来效率与性能平衡最佳的模型。它以仅 1.3B 的参数规模,实现了性能与效率的双重突破,在全球同尺寸模型中登顶,全面超越了阿里 Qwen3.5-0.8B 与谷歌 Gemma4-E2B-it。Jinja00
MiniMax-M2.7MiniMax-M2.7 是我们首个深度参与自身进化过程的模型。M2.7 具备构建复杂智能体应用框架的能力,能够借助智能体团队、复杂技能以及动态工具搜索,完成高度精细的生产力任务。Python00
MusicFreeDesktop插件化、定制化、无广告的免费音乐播放器TypeScript00
热门内容推荐
最新内容推荐
项目优选
收起
deepin linux kernel
C
29
16
暂无描述
Dockerfile
727
4.66 K
Ascend Extension for PyTorch
Python
599
751
Claude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed.
Get Started
Rust
1.02 K
139
🎉 (RuoYi)官方仓库 基于SpringBoot,Spring Security,JWT,Vue3 & Vite、Element Plus 的前后端分离权限管理系统
Vue
1.66 K
971
暂无简介
Dart
970
246
openEuler内核是openEuler操作系统的核心,既是系统性能与稳定性的基石,也是连接处理器、设备与服务的桥梁。
C
427
377
旨在打造算法先进、性能卓越、高效敏捷、安全可靠的密码套件,通过轻量级、可剪裁的软件技术架构满足各行业不同场景的多样化要求,让密码技术应用更简单,同时探索后量子等先进算法创新实践,构建密码前沿技术底座!
C
1.09 K
610
AI 将任意文档转换为精美可编辑的 PPTX 演示文稿 — 无需设计基础 | 包含 15 个案例、229 页内容
Python
122
7
本项目是CANN提供的数学类基础计算算子库,实现网络在NPU上加速计算。
C++
992
988