Apache CXF z JMS
Jak wspomniano wcześniej, można używać CXF z transportem JMS. W takim przypadku klient wyśle wiadomość JMS do znanego serwera przesyłania komunikatów. Nasza aplikacja serwerowa nieustannie nasłuchuje serwera komunikacyjnego w poszukiwaniu wiadomości przychodzących. Gdy nadejdzie wiadomość, przetwarza ją, wykonuje żądanie klienta i wysyła odpowiedź jako kolejną wiadomość do klienta.
Jak wcześniej, najpierw utworzymy przykładową aplikację serwerową, która udostępnia pojedynczą metodę internetową o nazwie sayHi.
Tworzenie interfejsu usługi
Interfejs serwisowy dla naszego HelloWorld usługa jest pokazana tutaj -
//HelloWorld.java
package com.tutorialspoint.service;
import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebService;
@WebService
public interface HelloWorld {
@WebMethod
String sayHi(@WebParam(name = "name") String name);
}
Usługa wdrożeniowa
Implementacja interfejsu usługi jest zdefiniowana w następujący sposób -
//HelloWorldImpl.java
package com.tutorialspoint.service.impl;
import javax.jws.WebService;
import com.tutorialspoint.service.HelloWorld;
@WebService
public class HelloWorldImpl implements HelloWorld {
@Override
public String sayHi(String name) {
return "Hello " + name;
}
}
Implementacja po prostu zwraca użytkownikowi wiadomość Hello. Jak widać, interfejs i jego implementacja są podobne do wszystkich wcześniejszych projektów w tym samouczku, które do tej pory znasz.
Teraz najważniejsza kwestia polega na utworzeniu aplikacji serwerowej, która ustawia kolejkę wiadomości i nasłuchuje wiadomości przychodzących.
Tworzenie serwera
W aplikacji serwerowej najpierw tworzymy plik JMS punkt końcowy w następujący sposób -
private static final String JMS_ENDPOINT_URI =
"jms:queue:test.cxf.jmstransport.queue?timeToLive=1000"
+ "&jndiConnectionFactoryName=ConnectionFactory"
+ "&jndiInitialContextFactory"
+ "= org.apache.activemq.jndi.ActiveMQInitialContextFactory"
+ "&jndiURL = tcp://localhost:61616";
Zauważ, że ustawiliśmy kolejkę na określonym porcie, który żyje przez określony czas. Teraz tworzymy usługę przesyłania wiadomości, tworząc instancjęorg.apache.activemq.broker.BrokerServiceklasa. To jest klasa serwera dlaActiveMQ serwer wiadomości.
BrokerService broker = new BrokerService();
Możesz użyć dowolnego innego serwera przesyłania wiadomości, który wybierzesz inny niż ActiveMQ. Teraz łączymy ten serwer z żądanym identyfikatorem URI.
broker.addConnector("tcp://localhost:61616");
Ustawiliśmy katalog do przechowywania danych wiadomości przychodzących -
broker.setDataDirectory("target/activemq-data");
Na koniec uruchamiamy serwer metodą start -
broker.start();
Następnie tworzymy wystąpienie naszego komponentu bean usług HelloWorld przy użyciu klasy komponentu bean serwera fabrycznego używanej we wcześniejszej aplikacji POJO -
Object implementor = new HelloWorldImpl();
JaxWsServerFactoryBean factory = new JaxWsServerFactoryBean();
factory.setServiceClass(HelloWorld.class);
Następnie skonfigurowaliśmy punkt końcowy JMS w fabryce, aby fabryka nadal nasłuchiwała przychodzących wiadomości -
factory.setTransportId
(JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress(JMS_ENDPOINT_URI);
Na koniec ustawiliśmy klasę implementującą w fabryce i zaczynamy ją uruchamiać -
factory.setServiceBean(implementor);
factory.create();
W tym momencie Twój serwer jest uruchomiony. Zauważ, że ponieważ użyliśmy fabrycznej klasy bean, tak jak w aplikacji POJO, nie jest wymagana potrzeba CXFServlet i pliku web.xml.
Pełny kod aplikacji serwera jest pokazany tutaj -
//ServerJMS.java
package com.tutorialspoint.server;
import java.util.Collections;
import org.apache.cxf.ext.logging.LoggingFeature;
import org.apache.cxf.jaxws.JaxWsServerFactoryBean;
import org.apache.cxf.transport.jms.spec.JMSSpecConstants;
import com.tutorialspoint.service.HelloWorld;
import com.tutorialspoint.service.impl.HelloWorldImpl;
import org.apache.activemq.broker.BrokerService;
public final class ServerJMS {
private static final String JMS_ENDPOINT_URI =
"jms:queue:test.cxf.jmstransport.queue?timeToLive=1000"
+ "&jndiConnectionFactoryName=ConnectionFactory"
+ "&jndiInitialContextFactory"
+ "= org.apache.activemq.jndi.ActiveMQInitialContextFactory"
+ "&jndiURL = tcp://localhost:61616";
public static void main(String[] args) throws Exception {
BrokerService broker = new BrokerService();
broker.addConnector("tcp://localhost:61616");
broker.setDataDirectory("target/activemq-data");
broker.start();
Object implementor = new HelloWorldImpl();
JaxWsServerFactoryBean factory = new JaxWsServerFactoryBean();
factory.setServiceClass(HelloWorld.class);
factory.setTransportId
(JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress(JMS_ENDPOINT_URI);
factory.setServiceBean(implementor);
factory.setFeatures(Collections.singletonList(new LoggingFeature()));
factory.create();
System.out.println("Server ready...");
Thread.sleep(5 * 60 * 1000);
System.out.println("Server exiting");
System.exit(0);
}
}
Dodawanie zależności
Stworzona przez nas aplikacja serwerowa korzysta z serwera komunikacyjnego ActiveMQ. Dlatego będziesz musiał dodać kilka dodatkowych zależności do swojego projektu. Pełny plik pom.xml jest pokazany tutaj, abyś mógł zrozumieć dodatkowe potrzebne zależności.
<?xml version = "1.0" encoding = "UTF-8"?>
<project xmlns = "http://maven.apache.org/POM/4.0.0"
xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation = "http://maven.apache.org/POM/4.0.0
http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.tutorialspoint</groupId>
<artifactId>cxf-jms</artifactId>
<version>1.0</version>
<packaging>jar</packaging>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>1.8</maven.compiler.source>
<maven.compiler.target>1.8</maven.compiler.target>
</properties>
<profiles>
<profile>
<id>server</id>
<build>
<defaultGoal>test</defaultGoal>
<plugins>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.6.0</version>
<executions>
<execution>
<phase>test</phase>
<goals>
<goal>java</goal>
</goals>
<configuration>
<mainClass>
com.tutorialspoint.server.ServerJMS
</mainClass>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</profile>
<profile>
<id>client</id>
<build>
<defaultGoal>test</defaultGoal>
<plugins>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<executions>
<execution>
<phase>test</phase>
<goals>
<goal>java</goal>
</goals>
<configuration>
<mainClass>
com.tutorialspoint.client.ClientJMS
</mainClass>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</profile>
</profiles>
<dependencies>
<dependency>
<groupId>org.apache.activemq</groupId>
<artifactId>activemq-broker</artifactId>
<version>5.15.8</version>
</dependency>
<dependency>
<groupId>org.apache.activemq</groupId>
<artifactId>activemq-kahadb-store</artifactId>
<version>5.15.8</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-frontend-jaxws</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-jms</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-features-logging</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-http-jetty</artifactId>
<version>3.3.0</version>
</dependency>
</dependencies>
</project>
Running Server
Aby rozpocząć uruchamianie serwera, tak jak we wcześniejszych przypadkach, wpisz następujące polecenie w oknie poleceń -
mvn -Pserver
Spowoduje to uruchomienie serwera komunikatów ActiveMQ, skonfigurowanie kolejki przesyłania komunikatów i utworzenie fabrycznego komponentu bean, który będzie nasłuchiwał tej kolejki.
Naszym kolejnym zadaniem jest stworzenie aplikacji klienckiej.
Tworzenie klienta
W aplikacji klienckiej w pierwszej kolejności ustawiamy punkt końcowy JMS taki sam jak używany w aplikacji serwerowej -
private static final String JMS_ENDPOINT_URI =
"jms:queue:test.cxf.jmstransport.queue?timeToLive=1000"
+ "&jndiConnectionFactoryName=ConnectionFactory"
+ "&jndiInitialContextFactory"
+ " = org.apache.activemq.jndi.ActiveMQInitialContextFactory"
+ "&jndiURL = tcp://localhost:61616";
Fabrykę tworzymy jak w aplikacji POJO.
JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();
Ustawiamy identyfikator URI punktu końcowego i klasę implementującą w następujący sposób -
factory.setTransportId (JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress (JMS_ENDPOINT_URI);
HelloWorld client = factory.create(HelloWorld.class);
Na koniec wywołujemy metodę service i wypisujemy jej wynik -
String reply = client.sayHi("TutorialsPoint");
System.out.println(reply);
Pełny kod klienta znajduje się poniżej -
// ClientJMS.java
package com.tutorialspoint.client;
import com.tutorialspoint.service.HelloWorld;
import org.apache.cxf.jaxws.JaxWsProxyFactoryBean;
import org.apache.cxf.transport.jms.spec.JMSSpecConstants;
public final class ClientJMS {
private static final String JMS_ENDPOINT_URI =
"jms:queue:test.cxf.jmstransport.queue?timeToLive=1000"
+ "&jndiConnectionFactoryName=ConnectionFactory"
+ "&jndiInitialContextFactory"
+ " = org.apache.activemq.jndi.ActiveMQInitialContextFactory"
+ "&jndiURL = tcp://localhost:61616";
public static void main(String[] args) throws Exception {
JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();
factory.setTransportId(JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress(JMS_ENDPOINT_URI);
HelloWorld client = factory.create(HelloWorld.class);
String reply = client.sayHi("TutorialsPoint");
System.out.println(reply);
System.exit(0);
}
}