使用 Camel 的 REST 服务调用需要首先调用身份验证 api
Posted
技术标签:
【中文标题】使用 Camel 的 REST 服务调用需要首先调用身份验证 api【英文标题】:REST service call with Camel which requires authentication api called first 【发布时间】:2017-12-14 00:49:02 【问题描述】:Camel 必须调用 REST 服务进行某些集成,但是,REST 服务有一个身份验证 api (POST api),需要首先调用它来获取令牌,然后必须调用嵌入令牌的其他后续 api 调用在 HTTP 请求的标头中。
Spring Restemplate 或 apache camel 是否有一些 api 来支持相同的功能?
【问题讨论】:
我在这个用例中使用了 Apache Camel。因此,您实际上需要调用两个服务。我创建了一个路由,它将在授权标头中返回缓存的或新的访问令牌(并缓存它)(我无法再访问 repo)并在主路由中使用丰富调用。 Camel 有 FluentProducerTemplate / ProducerTemplate API,您可以使用它类似于 Spring xxxTemplate 但适用于所有 Camel 组件/端点。 @gusto2 遵循了您的方法。它的工作..添加了工作解决方案作为其他人使用的答案(如果需要)。 【参考方案1】:遵循@gusto2 方法,它几乎可以正常工作。
所以,我创建了两条路由 --> 第一个是基于定时器的,如下所示,它生成令牌,定期刷新它(因为路由是基于定时器的)并将令牌存储在局部变量中以供某些人重用其他路线。
@Component
public class RestTokenProducerRoute extends RouteBuilder
private String refreshedToken;
@Override
public void configure() throws Exception
restConfiguration().producerComponent("http4");
from("timer://test?period=1200000") //called every 20 mins
.process(
exchange -> exchange.getIn().setBody(
new UserKeyRequest("apiuser", "password")))
.marshal(userKeyRequestJacksonFormat) //convert it to JSON
.setHeader(Exchange.HTTP_METHOD, constant("POST"))
.setHeader(Exchange.CONTENT_TYPE, constant("application/json"))
.to("http4://localhost:8085/Service/Token")
.unmarshal(userKeyResponseJacksonFormat)
.process(new Processor()
public void process(Exchange exchange) throws Exception
UserKeyResponse response= exchange.getIn().getBody(
UserKeyResponse.class); //get the response object
System.out.println(response + "========>>>>>>" +
response.getResult());
setRefreshedToken(response.getResult()); //store the token in some object
).log("$body");
public String getRefreshedToken()
return refreshedToken;
public void setRefreshedToken(String refreshedToken)
this.refreshedToken = refreshedToken;
第二个路由可以调用后续的api,这些api将使用第一个路由生成的令牌,它会是这样的。必须添加错误处理场景,其中令牌可能无效或过期。但我想这将是另一个需要解决的问题。
@Component
public class RestTokenUserOnboardRoute extends RouteBuilder
private JacksonDataFormat OtherDomainUserRequestJacksonFormat = new JacksonDataFormat(
OtherDomainUserRequest.class);
private JacksonDataFormat OtherDomainUserResponseJacksonFormat = new JacksonDataFormat(
OtherDomainUserResponse.class);
@Override
public void configure() throws Exception
restConfiguration().producerComponent("http4");
//This route is subscribed to a Salesforce topic, which gets invoked when there is any new messages in the topic.
from("salesforce:CamelTestTopic?sObjectName=MyUser__c&sObjectClass="+MyUser__c.class.getName()))
.convertBodyTo(OtherDomainUserRequest.class)
.marshal(OtherDomainUserRequestJacksonFormat).log("$body")
.setHeader(Exchange.HTTP_METHOD, constant("POST"))
.setHeader(Exchange.CONTENT_TYPE, constant("application/json"))
.log("The token being passed is ==> $bean:tokenObj?method=getRefreshedToken")
.setHeader("Authorization", simple("$bean:tokenObj?method=getRefreshedToken"))
.to("http4://localhost:8085/Service/DomainUser")
.unmarshal(OtherDomainUserResponseJacksonFormat)
.process(new Processor()
public void process(Exchange exchange) throws Exception
OtherDomainUserResponse response = exchange.getIn().getBody(
OtherDomainUserResponse.class);
System.out.println(response + "==================>>>>>> " + response.getStatusCode());
).log("$body");
所以,这里的令牌是从tokenObj
bean 中消耗的(RestTokenProducerRoute
的实例,它定义了方法getRefreshedToken()
。它返回存储的令牌。
不用说,您已经在 camelcontext 注册表中设置了 bean 以及其他设置(如组件、路由等),如下所示。在我的情况下,如下所示。
@Autowired
public RestTokenUserOnboardRoute userOnboardRoute;
@Autowired
public RestTokenProducerRoute serviceTokenProducerRoute;
@Autowired
private RestTokenProducerRoute tokenObj;
@Override
protected CamelContext createCamelContext() throws Exception
SimpleRegistry registry = new SimpleRegistry();
registry.put("tokenObj", tokenObj); //the tokenObj bean,which can be used anywhere in the camelcontext
SpringCamelContext camelContext = new SpringCamelContext();
camelContext.setRegistry(registry); //add the registry
camelContext.setApplicationContext(getApplicationContext());
camelContext.addComponent("salesforce", salesforceComponent());
camelContext.getTypeConverterRegistry().addTypeConverter(DomainUserRequest.class, MyUser__c.class, new MyTypeConverter());
camelContext.addRoutes(route()); //Some other route
camelContext.addRoutes(serviceTokenProducerRoute); //Token producer Route
camelContext.addRoutes(userOnboardRoute); //Subsequent API call route
camelContext.start();
return camelContext;
这解决了我动态设置令牌的问题,其中令牌是由于执行其他路由而产生的。
【讨论】:
我看到代码正在为每个服务器请求请求一个新令牌。一开始可能没问题,但最佳实践(以及我未来的建议)是缓存(camel.apache.org/cache.html)并重用令牌(或者 - 可能使用永久应用程序级令牌),这样你就不会放置不必要的负载您的服务到另一个外部服务。不管怎样-感谢分享:) @gusto2 对于每个服务器请求,它不请求新令牌。按照 TokenProducer 路由中的配置,令牌每 20 分钟刷新一次。因此,每 20 分钟路由将被调用并将令牌缓存/存储在内存 bean (tokenObj) 中。并且其他服务器请求将简单地从 tokenObj 访问令牌,而不知道它何时刷新。所以,它肯定不会为每个服务器请求请求一个新的令牌。以上是关于使用 Camel 的 REST 服务调用需要首先调用身份验证 api的主要内容,如果未能解决你的问题,请参考以下文章
Apache Camel - 调用 http 或 rest 调用(通过 Shiro Security 过滤)
使用SpringOAuthResttemplate的Camel Rest api使用者
Apache Camel JMS - 异常未通过请求/回复返回给调用者