Quarkus集成Keycloak卡顿解决方法
最近发现不少小伙伴都对文章很感兴趣,所以今天继续给大家介绍文章相关的知识,本文《Quarkus集成Keycloak阻塞问题与解决方法》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

本教程探讨在Quarkus响应式应用程序中使用`quarkus-keycloak-admin-client-reactive`扩展时,`ServerRequestFilter`中遇到的`BlockingNotAllowedException`问题。尽管扩展名暗示响应式,但底层Keycloak客户端仍执行阻塞操作。文章将详细解释此问题,并提供一个基于Vert.x `executeBlocking`的有效解决方案,确保在响应式上下文中安全地执行阻塞调用,从而维护应用的响应性。
理解Quarkus响应式上下文与Keycloak管理客户端
Quarkus以其对响应式编程模型的强大支持而闻名,尤其是在使用quarkus-resteasy-reactive等扩展时。在这种模型下,所有I/O操作都应是非阻塞的,以最大化资源利用率和吞吐量。然而,当我们需要与外部服务(如Keycloak)进行交互时,尤其是在使用其管理API时,可能会遇到挑战。
quarkus-keycloak-admin-client-reactive扩展旨在简化Quarkus应用与Keycloak管理API的集成。它允许开发者通过注入Keycloak客户端实例来执行管理操作。理想情况下,这个客户端在响应式上下文中也应该是非阻塞的。
遇到的问题:BlockingNotAllowedException
在响应式Quarkus应用中,如果尝试在非阻塞线程(如Vert.x事件循环线程)上执行阻塞操作,Quarkus会抛出BlockingNotAllowedException。这通常发生在数据库访问、外部HTTP调用或文件I/O等操作未能以响应式方式实现时。
考虑以下在ServerRequestFilter中尝试使用注入的Keycloak客户端查询用户属性的场景:
import io.quarkus.security.identity.SecurityIdentity;
import io.smallrye.mutiny.Uni;
import jakarta.inject.Inject;
import jakarta.ws.rs.container.ContainerRequestContext;
import jakarta.ws.rs.ext.Provider;
import jakarta.ws.rs.core.Response;
import org.jboss.resteasy.reactive.server.ServerRequestFilter;
import org.keycloak.admin.client.Keycloak;
@Provider
public class KeycloakAttributeFilter {
@Inject
Keycloak keycloak; // 注入Keycloak管理客户端
@ServerRequestFilter
public Uni<Response> filter(ContainerRequestContext requestContext) {
// 假设从请求上下文获取到用户名
String username = "user-from-context";
return Uni.createFrom().item(() ->
keycloak.realm("my-realm")
.users()
.search(username)
.stream()
.findFirst()
.orElseThrow(() -> new RuntimeException("User not found"))
.firstAttribute("the-attribute"))
.map(attr -> {
if ("some-value".equals(attr)) {
return null; // 继续处理请求
}
return Response.status(403).build(); // 拒绝请求
});
}
}尽管我们使用了Mutiny的Uni来封装操作,并且扩展名称中带有“reactive”,但在执行keycloak.realm(...).users().search(...)等方法时,仍然会抛出BlockingNotAllowedException。这是因为keycloak-admin-client库本身是基于传统的ResteasyClient构建的,其内部的HTTP调用是阻塞的,而非Quarkus响应式ResteasyReactiveClient。直接在响应式上下文中调用这些阻塞方法会导致线程阻塞,从而触发异常。
尝试使用Uni.runSubscriptionOn(Infrastructure.getDefaultWorkerPool())也无法解决此问题,因为它只是将Uni的订阅操作调度到不同的线程池,但实际的阻塞调用仍然会在这个线程池中执行,这并不是Quarkus响应式模型所期望的。
解决方案:利用Vert.x executeBlocking
解决此问题的核心在于,我们需要将这些阻塞的Keycloak管理客户端调用封装在一个允许阻塞的上下文中执行,同时又不中断整个应用程序的响应式流。Vert.x提供了executeBlocking方法,它正是为此目的而设计的。executeBlocking会将提供的阻塞任务提交给Vert.x的内部工作线程池执行,从而避免阻塞事件循环线程。
要使用executeBlocking,首先需要注入Vert.x实例:
import io.vertx.mutiny.core.Vertx;
import io.smallrye.mutiny.Uni;
import jakarta.inject.Inject;
import jakarta.ws.rs.container.ContainerRequestContext;
import jakarta.ws.rs.ext.Provider;
import jakarta.ws.rs.core.Response;
import org.jboss.resteasy.reactive.server.ServerRequestFilter;
import org.keycloak.admin.client.Keycloak;
@Provider
public class KeycloakAttributeFilter {
@Inject
Keycloak keycloak;
@Inject
Vertx vertx; // 注入Vert.x实例
@ServerRequestFilter
public Uni<Response> filter(ContainerRequestContext requestContext) {
String username = "user-from-context"; // 假设从请求上下文获取
return vertx.getOrCreateContext().executeBlocking(
Uni.createFrom().emitter(emitter -> {
try {
// 在executeBlocking内部执行阻塞的Keycloak调用
var attr = keycloak.realm("my-realm")
.users()
.search(username)
.stream()
.findFirst()
.orElseThrow(() -> new RuntimeException("User not found"))
.firstAttribute("the-attribute");
if ("some-value".equals(attr)) {
emitter.complete(null); // 成功,继续请求处理
} else {
emitter.complete(Response.status(403).build()); // 失败,返回403
}
} catch (Exception e) {
emitter.fail(e); // 捕获异常并传递
}
}));
}
}代码解析:
- @Inject Vertx vertx;: 注入Vert.x的Mutiny版本实例,它提供了executeBlocking方法。
- vertx.getOrCreateContext().executeBlocking(...): 这是核心。它接收一个Uni作为参数,该Uni的任务将在Vert.x的工作线程池中执行。这意味着其中的代码可以安全地执行阻塞操作,而不会影响主事件循环线程。
- Uni.createFrom().emitter(emitter -> { ... });: 在executeBlocking内部,我们使用Uni.createFrom().emitter来手动控制Uni的完成或失败。
- 在emitter的回调中,我们执行了原始的、阻塞的Keycloak客户端调用。
- 根据Keycloak调用的结果,通过emitter.complete(null)(表示成功,过滤器允许请求继续)或emitter.complete(Response.status(403).build())(表示失败,返回403响应)来完成Uni。
- try-catch块用于捕获Keycloak调用可能抛出的异常,并通过emitter.fail(e)将异常传递给响应式流。
通过这种方式,我们成功地将阻塞的Keycloak调用隔离到Vert.x的工作线程中,从而避免了BlockingNotAllowedException,并维护了Quarkus应用程序的整体响应性。
注意事项与最佳实践
- 适度使用executeBlocking: executeBlocking是处理阻塞操作的有效手段,但不应滥用。如果存在真正的响应式客户端或库,应优先使用它们。executeBlocking会占用Vert.x的工作线程,过多的阻塞操作会耗尽工作线程池,影响系统吞吐量。
- 错误处理: 在executeBlocking内部,务必进行适当的错误处理。使用try-catch块捕获可能发生的异常,并通过emitter.fail(e)将它们传播到响应式流中,以便后续的onFailure操作能够处理。
- 上下文传递: 如果Keycloak调用依赖于请求上下文中的某些信息(例如安全主体、请求头等),请确保这些信息在executeBlocking的任务内部仍然可用。通常,可以在调用executeBlocking之前从requestContext中提取所需信息,并将其作为局部变量传递到Lambda表达式中。
- 性能考量: 尽管executeBlocking解决了阻塞问题,但阻塞操作本身仍然需要时间。如果Keycloak管理API调用频繁且耗时,可能会成为性能瓶颈。在这种情况下,考虑对Keycloak响应进行缓存,或者重新评估业务流程以减少对管理API的直接、同步依赖。
总结
在Quarkus响应式应用中集成keycloak-admin-client时,由于其底层是阻塞的,直接调用会导致BlockingNotAllowedException。通过巧妙地利用Vert.x的executeBlocking方法,我们可以将这些阻塞操作安全地转移到专门的工作线程池中执行,从而维护应用程序的响应式特性。这种模式是处理现有阻塞库在响应式环境中集成的有效策略,但开发者应始终关注其对性能和资源利用的影响,并优先选择原生响应式解决方案。
今天关于《Quarkus集成Keycloak卡顿解决方法》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
HTML调用摄像头拍摄全攻略
- 上一篇
- HTML调用摄像头拍摄全攻略
- 下一篇
- 钉钉消息延迟问题解决方法
-
- 文章 · java教程 | 1分钟前 |
- Java转义字符使用与处理全攻略
- 246浏览 收藏
-
- 文章 · java教程 | 3分钟前 |
- JavaStream.map集合转换全解析
- 500浏览 收藏
-
- 文章 · java教程 | 20分钟前 |
- Java读写锁实现步骤全解析
- 385浏览 收藏
-
- 文章 · java教程 | 38分钟前 |
- Mac上运行Java控制台程序详解
- 108浏览 收藏
-
- 文章 · java教程 | 40分钟前 |
- Paths类获取文件路径方法详解
- 191浏览 收藏
-
- 文章 · java教程 | 1小时前 | java 聊天室客户端
- Java聊天室客户端开发教程详解
- 142浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java论坛评论功能实现全解析
- 429浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java接口实现松耦合的方法与技巧
- 460浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java运算符优先级全解析
- 479浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java开发个人记事本技巧全解析
- 352浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java实现简易博客推荐算法实战教程
- 366浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- Java捕获ClassNotFoundException方法
- 106浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3328次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3540次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3571次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4695次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3943次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

