如何在将来取消时在Callable中终止CXF Web服务调用

时间:2009-07-13 17:03:30

标签: java cxf future interrupt callable

修改

这个问题到目前为止经历了几次迭代,所以请随时查看修订版本以查看有关历史和所尝试事项的背景信息。


我正在使用CompletionService和ExecutorService以及Callable,通过CXF生成的代码同时调用几个不同的Web服务上的许多函数。这些服务都为一组信息提供不同的信息。我用于我的项目。但是,服务可能无法长时间响应而不会抛出异常,从而延长等待组合信息的时间。

为了解决这个问题,我正在同时运行所有服务调用,并在几分钟后想要终止任何尚未完成的调用,并且最好记录哪些调用尚未完成或者通过抛出详细的例外。

这里有一些高度简化的代码来说明我已经在做什么:

private Callable<List<Feature>> getXXXFeatures(final WiwsPortType port, 
final String accessionCode) {
    return new Callable<List<Feature>>() {
        @Override
        public List<Feature> call() throws Exception {
            List<Feature> features = new ArrayList<Feature>();
            //getXXXFeatures are methods of the WS Proxy
            //that can take anywhere from second to never to return
            for (RawFeature raw : port.getXXXFeatures(accessionCode)) {
                Feature ft = convertFeature(raw);
                features.add(ft);
            }
            if (Thread.currentThread().isInterrupted())
                log.error("XXX was interrupted");
            return features;
        }
    };
}

同时启动WS调用的代码:

WiwsPortType port = new Wiws().getWiws();
List<Future<List<Feature>>> ftList = new ArrayList<Future<List<Feature>>>();
//Counting wrapper around CompletionService, 
    //so I could implement ccs.hasRemaining()
CountingCompletionService<List<Feature>> ccs = 
        new CountingCompletionService<List<Feature>>(threadpool);
ftList.add(ccs.submit(getXXXFeatures(port, accessionCode)));
ftList.add(ccs.submit(getYYYFeatures(port accessionCode)));
ftList.add(ccs.submit(getZZZFeatures(port, accessionCode)));

List<Feature> allFeatures = new ArrayList<Feature>();
while (ccs.hasRemaining()) {
            //Low for testing, eventually a little more lenient
    Future<List<Feature>> polled = ccs.poll(5, TimeUnit.SECONDS);
    if (polled != null)
        allFeatures.addAll(polled.get());
    else {
        //Still jobs remaining, but unresponsive: Cancel them all
        int jobsCanceled = 0;
        for (Future<List<Feature>> job : ftList)
            if (job.cancel(true))
                jobsCanceled++;
        log.error("Canceled {} feature jobs because they took too long",
                        jobsCanceled);
        break;
    }
}

我对此代码的问题是,在等待port.getXXXFeatures(...)返回时,Callables实际上没有被取消,但不知何故继续运行。正如您在if (Thread.currentThread().isInterrupted()) log.error("XXX was interrupted");语句中看到的,在port.getFeatures返回后设置了中断标志 ,这仅在Webservice调用正常完成后才可用,而不是在我调用时被中断取消。

任何人都可以告诉我我做错了什么以及如何在给定时间段后停止正在运行的CXF Webservice呼叫,并在我的应用程序中注册此信息?

最好的问候,蒂姆

2 个答案:

答案 0 :(得分:2)

编辑3 新答案。

我看到了这些选项:

  • 将您的问题作为功能请求发布在Apache CXF上
  • 自行修复ACXF并公开一些功能。
  • 在Apache CXF中查找异步WS调用支持的选项
  • 考虑切换到不同的WS提供程序(JAX-WS?)
  • 如果服务支持,您的WS使用RESTful API自行调用(例如,带参数的纯HTTP请求)
  • 仅限über专家:使用真正的线程/线程组并使用非正统方法终止线程。

答案 1 :(得分:1)

CXF文档有一些关于在HTTPURLConnection上设置读取超时的说明: http://cwiki.apache.org/CXF20DOC/client-http-transport-including-ssl-support.html

这可能会满足您的需求。如果服务器没有及时响应,则引发异常并且可调用将获得异常。 (除了有一个可能会挂起的bug。我不记得是否已经修复了2.2.2或者它现在只是在SNAPSHOTS中。)