JAX-WS: WS с состоянием завершается неудачно в автономном процессе
У меня есть веб-служба с состоянием, развернутая на Tomcat. Он состоит из фабричного сервиса и основного сервиса API и работает просто отлично. Служба Factory возвращает W3CEndpointReference к основному экземпляру API, и клиент использует сеанс.
Сейчас я пытаюсь запустить тот же сервис, что и отдельное приложение. В этом случае W3CEndpointReference, возвращаемая фабрикой, внезапно начинает указывать на фабричный URI, а не на основной сервисный.
Когда сравниваются ссылки из прогонов с Tomcat и автономными, становится ясно, что автономная ссылка имеет неправильный URI. В частности, он указывает на заводской URI, а не на основной API.
Вот правильная ссылка:
...
<ns3:Address>http://localhost:8080/td2ws/api</ns3:Address>
...
Вот ссылки, когда фабрика вызывается в автономном процессе:
<Address>http://localhost:9009/td2ws/factory</Address>
Мое понимание того, что некоторый код в контексте сервлета знает о соответствии между классом обслуживания (Td2Ws) и его URI, и соответствующим образом корректирует ссылки. Этот фрагмент кода не действует в автономном режиме, хотя. Я даже могу подозревать, что в коде используется sun-jaxws.xml, но я не знаю, как его "включить".
Как сделать так, чтобы веб-сервис с сохранением состояния работал в автономном приложении?
Вот соответствующие части кода:
Заводской сервис (без гражданства):
@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...");
Этот код завершается с ошибкой при входе в систему (), поскольку целевой объект не имеет операции входа в систему (в конце концов, он ошибочно нацелен на фабричный URI):
Exception in thread "main" javax.xml.ws.soap.SOAPFaultException:
Cannot find dispatch method for {http://server.td2ws.sf.net/}Login
1 ответ
Некоторое время назад я этого не делал, но я взглянул на функциональный веб-сервис с отслеживанием состояния, который я создал некоторое время назад. Мои аннотации очень похожи на ваши, за исключением того, что у меня есть это на вершине моего класса:
public class MyService {
@Resource
WebServiceContext wsCtx;
...
}
Заметьте, что нет модификатора, если что-то еще, я думаю, должно быть публичным, но я не уверен. Как контейнер знает, чтобы установить это бьет меня, но это работает. Он должен просто найти любой @Resource типа WebServiceContext и установить его. Вы можете получить доступ к информации о сеансе следующим образом:
HttpServletRequest hRequest = (HttpServletRequest)ctx.getMessageContext().get(MessageContext.SERVLET_REQUEST);
Session s = hRequest.getSession();