- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
TL;DR : 我尝试创建一个测试,该测试对 2 个不同的 SOAP 操作(saveUser 和 getUser)执行 2 个 CXF 调用,但第二个失败,因为设置了 http 请求中的 SOAPAction作为“saveUser”而不是“getUser”。我找到了 2 个解决方案来解决这个问题,但我对它们并不完全满意。
这是我的端点配置:
final JaxWsProxyFactoryBean jaxWsProxyFactory = new JaxWsProxyFactoryBean();
jaxWsProxyFactory.setServiceClass(Internal.class);
jaxWsProxyFactory.setAddress(backendUrl + "INTERNAL");
jaxWsProxyFactory.setFeatures(getFeatures());
final Internal portType = (Internal) jaxWsProxyFactory.create();
((BindingProvider) portType).getRequestContext().put("thread.local.request.context", "true");
((BindingProvider) portType).getRequestContext().put("schema-validation-enabled", "false");
这是我的用法:
internalBackendG5.saveUser(getRequest);
internalBackendG5.getUser(saveRequest);
这两个操作被定义为(感谢 wsdl2java):
(为了更简洁,我重写了一些包,下面的代码可能包含错别字)
@WebMethod(action = "http://www.test.org/internal/saveUser")
@WebResult(name = "saveUserResponse", targetNamespace = "urn:test.com:xsd:internal:userpreferences", partName = "parameters")
public com.test.internal.type.SaveUserResponseType saveUser(
@WebParam(partName = "parameters", name = "saveUser", targetNamespace = "urn:test.com:xsd:internal:userpreferences")
com.test.internal.type.SaveUserRequestType parameters
);
和
@WebMethod(action = "http://www.test.org/internal/getUser")
@WebResult(name = "getUserResponse", targetNamespace = "urn:test.com:xsd:internal:userpreferences", partName = "parameters")
public com.test.internal.type.GetUserResponseType getUser(
@WebParam(partName = "parameters", name = "getUser", targetNamespace = "urn:test.com:xsd:internal:userpreferences")
com.test.internal.type.GetUserRequestType parameters
);
当我运行我的测试时,我在第二次调用时收到一个错误,该错误作为 Soap 异常从后端抛出:
javax.xml.ws.soap.SOAPFaultException: Unexpected element {urn:test.com:xsd:internal:userpreferences}getUser found. Expected {urn:test.com:xsd:internal:userpreferences}saveUser.
当我查看服务器收到的 Soap 请求时,我得到:
对于第一个请求(saveUser):
Headers: {[...] SOAPAction=["http://www.test.org/internal/saveUser"], user-agent=[Apache CXF 2.7.12]}
Payload: <soap:Envelope><soap:Body><saveUser>[...]</ns5:saveUser></soap:Body></soap:Envelope>
对于第二个请求(getUser):
Headers: {[...] SOAPAction=["http://www.test.org/internal/saveUser"], user-agent=[Apache CXF 2.7.12]}
Payload: <soap:Envelope><soap:Body><getUser>[...]</getUser></soap:Body></soap:Envelope>
当我只启动两个请求中的一个时,它们都有效。
经过一些研究,我发现 Apache Cxf 拦截器 SoapPreProtocolOutInterceptor 是罪魁祸首。
当请求中已存在 SoapAction 字符串时,它不会覆盖它。我的第二个调用(出于某些我无法解释的原因)重用了在处理第一个调用时计算的 Cxf 请求上下文!
我做了什么来克服这个问题:- 编写一个在 SoapPreProtocolOutInterceptor 之前调用的拦截器,强制删除 header 中的 SoapAction,以强制此拦截器重新计算操作。- 或者我可以重置两个请求之间的请求上下文:
((BindingProvider) portType).getRequestContext().put(Header.HEADER_LIST, emptyList);
所以我的问题是:是否存在 CXF 错误?(不应在 2 个请求之间共享 requestContext)或我错过了一些 Cxf 配置?
PS:我使用了这个依赖:
group:'org.apache.cxf', name:'cxf-rt-transports-http-jetty', version:'2.7.5'
group: 'org.apache.cxf', name: 'cxf-rt-features-clustering', version:'2.7.12'
group:'org.apache.cxf', name:'cxf-rt-frontend-jaxws', version:'2.7.12'
group: 'org.apache.cxf', name: 'cxf-api', version: '2.7.12'
group: 'org.apache.cxf', name: 'cxf-rt-bindings-soap', version: '2.7.12'
提前致谢!
最佳答案
我遇到了同样的问题,经过一些调试,我想我已经找到了原因。
在调用任何类似这样的操作之前,我在 PROTOCOL_HEADERS 映射上设置了我自己的自定义 http header :
Map<String, List<String>> headers = new HashMap<>();
headers.put("header-name", Arrays.asList("value"));
proxy.getRequestContext().put(Message.PROTOCOL_HEADERS, headers);
从您的示例中,我看不到您正在设置 PROTOCOL_HEADERS,但这是我的根本原因,因为这是包含 SOAPAction header 的映射。也许您的代码的其他部分可能正在篡改它。
首先,我们知道客户端代理请求上下文不是线程安全的
Official JAX-WS answer: No. According to the JAX-WS spec, the client proxies are NOT thread safe. To write portable code, you should treat them as non-thread safe and synchronize access or use a pool of instances or similar.
CXF answer: CXF proxies are thread safe for MANY use cases. The exceptions are:
Use of ((BindingProvider)proxy).getRequestContext() - per JAX-WS spec, the request context is PER INSTANCE. Thus, anything set there will affect requests on other threads.
这对我使用此上下文提出了一些危险信号,但这似乎不是问题的原因,我只是向它添加了一个简单的协议(protocol) header ,这不是特定于用户的。
上下文在 ClientImpl 类中声明,并用作每个服务调用的基础上下文:
protected Map<String, Object> currentRequestContext = new ConcurrentHashMap<String, Object>(8, 0.75f, 4);
每次您请求上下文时,都会执行以下方法:
public Map<String, Object> getRequestContext() {
if (isThreadLocalRequestContext()) {
if (!requestContext.containsKey(Thread.currentThread())) {
requestContext.put(Thread.currentThread(), new EchoContext(currentRequestContext));
}
return requestContext.get(Thread.currentThread());
}
return currentRequestContext;
}
假设我们没有设置线程本地属性,所以我们总是返回当前请求上下文,它现在有我们的 PROTOCOL_HEADERS 映射,其中仍然没有 SOAPAction header 。
然后是 SoapPreProtocolOutInterceptor 类,(我只提取了 setSoapAction(...) 方法的相关部分:
Map<String, List<String>> reqHeaders
= CastUtils.cast((Map<?, ?>)message.get(Message.PROTOCOL_HEADERS));
if (reqHeaders == null) {
reqHeaders = new TreeMap<String, List<String>>(String.CASE_INSENSITIVE_ORDER);
}
if (reqHeaders.size() == 0) {
message.put(Message.PROTOCOL_HEADERS, reqHeaders);
}
if (!reqHeaders.containsKey(SoapBindingConstants.SOAP_ACTION)) {
reqHeaders.put(SoapBindingConstants.SOAP_ACTION, Collections.singletonList(action));
}
这里我们将在最后一个 if 语句中结束,当我们在这个客户端上执行我们的first 调用时,因此在SAME 协议(protocol) header 上设置 SOAP_ACTION header map,它是当前请求上下文的一部分,在 ClientImpl 类中声明。
此客户端的任何后续请求都将以已经填充的映射为基础,其中包含协议(protocol) header 和上下文中的 SOAP_ACTION header ,从而导致观察到的行为。我们永远不会输入为我们的服务调用上下文创建带有 PROTOCOL_HEADERS 的新映射的 if 语句。
解决方案:
我认为最干净的解决方案是只使用拦截器在协议(protocol) header 映射中设置值,请注意我们根本不在基本请求上下文上操作:
public class HttpHeaderExampleInterceptor extends AbstractPhaseInterceptor
{
private static final String HEADER_NAME = "header_name";
private String value;
public HttpHeaderExampleInterceptor(String value)
{
super(Phase.MARSHAL);
this.value = value;
}
/**
* @see org.apache.cxf.binding.soap.interceptor.SoapPreProtocolOutInterceptor#setSoapAction(SoapMessage)
*/
@Override
public void handleMessage(Message message) throws Fault
{
Map<String, List<String>> protocolHeaders = CastUtils.cast((Map<?, ?>) message.get(Message.PROTOCOL_HEADERS));
if (protocolHeaders == null)
{
protocolHeaders = new TreeMap<String, List<String>>(String.CASE_INSENSITIVE_ORDER);
message.put(Message.PROTOCOL_HEADERS, protocolHeaders);
}
protocolHeaders.put(HEADER_NAME, Collections.singletonList(value));
}
}
希望这会有所帮助,如果有人在我的调试中发现错误,请分享,因为这个问题确实很有趣。
关于java - CXF 请求 HTTP 上下文不会在 2 个请求之间刷新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39015192/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!