5

我在 Tomcat 上部署了一个有状态的 Web 服务。它由工厂服务和主要 API 服务组成,运行良好。工厂服务返回一个 W3CEndpointReference 给主 API 实例,客户端使用会话。

现在,我正在尝试运行与独立应用程序相同的服务。在这种情况下,工厂返回的 W3CEndpointReference 突然开始指向工厂 URI,而不是主服务一。

当比较针对 Tomcat 和独立运行的引用时,很明显独立引用具有错误的 URI。具体来说,它指向工厂 URI,而不是主 API 之一。

这是正确的参考:

...
<ns3:Address>http://localhost:8080/td2ws/api</ns3:Address> 
...

以下是在独立进程上调用工厂时的参考:

<Address>http://localhost:9009/td2ws/factory</Address>

我的理解是servlet上下文下的一些代码知道服务类(Td2Ws)和它的URI之间的对应关系,并相应地调整引用。不过,那段代码在独立进程下无效。我什至可以怀疑代码使用了 sun-jaxws.xml,但我不知道如何“打开它”。

如何使独立应用程序中的有状态 Web 服务工作?

以下是代码的相关部分:

工厂服务(无国籍):

@WebService(targetNamespace="http://server.td2ws.sf.net/") 
public class Td2WsFactory {
    @WebMethod(operationName="StartSession", action="urn:StartSession")
    @WebResult(name="Reference")
    public W3CEndpointReference startSession() {
        StatefulWebServiceManager<Td2Ws> manager = Td2Ws.getManager();

        // for standalone execution only
        if( manager == null ) {
            manager = new StatefulInstanceResolver<Td2Ws>(Td2Ws.class);
            Td2Ws.setManager(manager);
        }

        Td2Ws session = new Td2Ws();
        return manager.export(W3CEndpointReference.class,session);
    }
}

有状态的 API:

@Stateful
@WebService(targetNamespace="http://server.td2ws.sf.net/") 
@Addressing
public class Td2Ws {
    private static StatefulWebServiceManager<Td2Ws> manager;

    public static void setManager(StatefulWebServiceManager<Td2Ws> manager ) {
        Td2Ws.manager = manager;
    }

    public static StatefulWebServiceManager<Td2Ws> getManager() {
        return Td2Ws.manager;
    }

    @WebMethod(operationName="Login", action="urn:Login")
    public void login(
            @WebParam(name="Username") String username,
            @WebParam(name="Password") String password,
            @WebParam(name="Domain") String domain,
            @WebParam(name="Project") String project
            ) throws Td2WsException {

客户端代码(针对独立):

public static void main(String[] args) throws Exception {
    Endpoint epf = Endpoint.publish("http://localhost:9009/td2ws/factory", new net.sf.td2ws.server.Td2WsFactory());
    Endpoint eps = Endpoint.publish("http://localhost:9009/td2ws/api", new net.sf.td2ws.server.Td2Ws());

    URL factoryUrl = new URL("http://localhost:9009/td2ws/factory?wsdl");
    QName factoryQname = new QName("http://server.td2ws.sf.net/", "Td2WsFactoryService");

    URL sessionUrl = new URL("http://localhost:9009/td2ws/api?wsdl");
    QName sessionQname = new QName("http://server.td2ws.sf.net/", "Td2WsService");

    Td2WsFactory factory = new Td2WsFactoryService(factoryUrl,factoryQname).getTd2WsFactoryPort();
    System.out.println("factory: "+factory);

    W3CEndpointReference session = factory.startSession();
    System.out.println("session: "+session);

    Td2WsService service = new Td2WsService(sessionUrl,sessionQname);
    System.out.println("service: "+service);

    Td2Ws port = service.getPort(session,Td2Ws.class);
    System.out.println("port: "+port);

    port.login(
            Config.get("td.user"), 
            Config.get("td.pass"),
            Config.get("td.domain"),
            Config.get("td.project"));
    System.out.println("logged in...");

该代码在 login() 上失败,因为目标对象没有登录操作(毕竟它错误地定位到工厂 URI):

Exception in thread "main" javax.xml.ws.soap.SOAPFaultException: 
Cannot find dispatch method for {http://server.td2ws.sf.net/}Login
4

1 回答 1

0

我有一段时间没有这样做了,但是我看了一下我前段时间构建的一个功能性的有状态 Web 服务。我的注释与您的非常相似,除了我在班上排名第一:

public class MyService {

  @Resource
  WebServiceContext wsCtx;

  ...

}

请注意,没有修饰符,如果还有什么我认为需要公开的话,但我并不积极。容器如何知道如何设置这个让我感到震惊,但它确实有效。它必须查找任何类型为 WebServiceContext 的 @Resource 并设置它。您可以像这样访问会话信息:

HttpServletRequest hRequest =  (HttpServletRequest)ctx.getMessageContext().get(MessageContext.SERVLET_REQUEST);

Session s = hRequest.getSession();
于 2013-01-07T21:10:27.917 回答