Apache CXF - Guide rapide

Dans l'environnement actuel, vous pouvez créer une application de service Web à l'aide de plusieurs options. Vous pouvez utiliser un ou plusieurs des protocoles de communication standard et largement acceptés. Par exemple, SOAP , XML / HTTP, RESTful HTTP et CORBA (Common Object Request Broker Architecture, qui était très populaire autrefois mais qui n'est pas si fréquemment utilisée aujourd'hui.

Vous avez également le choix entre différents transports tels que HTTP, JMS , JBI et le choix d'API frontales comme JAX-RS et JAX-WS . Ayant autant d'options pour le développement de services Web, il est nécessaire de disposer d'un cadre de services open source pour coller toutes les options mentionnées ci-dessus ensemble et c'est ce que fait Apache CXF .

Dans ce didacticiel, vous apprendrez à utiliser CXF pour créer à la fois un service Web et un client qui consomme le service, en utilisant une ou plusieurs des options que nous avons répertoriées ci-dessus. Ce didacticiel vous guidera tout au long du développement du code pour le serveur et le client. Comme chaque application ne peut utiliser qu'une seule des options de chaque catégorie, à savoir le frontend, le transport et le protocole, compte tenu de toutes les permutations et combinaisons de ces trois, le nombre d'applications sera exorbitant.

Ce didacticiel décrit en détail le développement des projets suivants -

  • CXF avec des objets Apache CXF simples (POJO)

  • CXF avec JAX-WS

  • CXF avec WSDL

  • CXF avec JAX-RS

  • CXF avec JMS

Pour faire simple, nous avons utilisé maven avec son interface de ligne de commande. Vous pouvez utiliser votre IDE préféré pour créer un projet maven.

Dans le chapitre suivant, commençons par le premier.

Dans ce chapitre, vous apprendrez à développer une application Web simple qui envoie un message d'accueil à l'utilisateur. Un projet de service Web utilise le modèle WSDL . Le CXF vous permet de masquer ce modèle WSDL en fournissant une interface simple pour mapper les API Apache CXF au WSDL sous-jacent.

Dans ce projet le plus simple, l'interface du service Web sera directement exposée au client et le client utilisera des API Apache CXF natives pour appeler le service Web.

Tout d'abord, nous allons créer un service Web. Chaque service a une interface qui est exposée au client. Nous pouvons écrire cette interface comme une simple interface Apache CXF ou comme un document WSDL. Dans cette approche Apache CXF-First, nous exposerons notre service via une interface Apache CXF.

Développement d'un service Web

Le service que nous allons créer sur le Web aura une seule méthode Web appelée greetings. La méthode prend unstringargument de type dans lequel nous enverrons le nom de l'utilisateur. Le service renverra un message d'accueil à l'appelant avec le nom d'utilisateur reçu inclus dans le message.

Interface de service Web

Pour exposer l'interface de notre service Web, nous allons créer une interface Apache CXF comme suit -

//HelloWorld.java
package com.tutorialspoint.cxf.pojo;
public interface HelloWorld {
   String greetings(String text);
}

L'interface n'a qu'une seule méthode appelée greetings. Le serveur implémentera cette interface. Dans notre application triviale, cette interface est directement exposée au client. En règle générale, dans une application de service Web, vous utilisez WSDL pour décrire l'interface de service Web. Dans cette application simple, nous fournirons cette interface directe au développeur client. Le client appellerait alors legreetingsmessage sur l'objet serveur. Alors commençons par créer le service Web.

Implémentation de service Web

le HelloWorld l'interface est implémentée dans le HelloWorldImpl Classe Apache CXF comme indiqué ci-dessous -

//HelloWorldImpl.java
package com.tutorialspoint.cxf.pojo;
public class HelloWorldImpl implements HelloWorld {
   @Override
   public String greetings(String text) {
      return "Hi " + text;
   }
}

le greetings la méthode reçoit un paramètre de string type, l'ajoute à un message d'accueil et renvoie la chaîne résultante à l'appelant.

Ensuite, nous écrivons l'application serveur pour héberger le HelloWorld un service.

Création du serveur

L'application serveur se compose de deux parties -

  • La première partie crée une usine pour notre service Web, et

  • La deuxième partie écrit un main méthode pour l'instancier.

Le serveur utilise ServerFactoryBean classe fournie par les bibliothèques CXF pour exposer notre HelloWorldinterface avec les clients distants. Ainsi, nous instancions d'abord leServerFactoryBean class, puis définissez ses différentes propriétés -

ServerFactoryBean factory = new ServerFactoryBean();

Nous définissons la classe de service à appeler en appelant le setServiceClass méthode sur le factory objet -

factory.setServiceClass(HelloWorld.class);

Nous définissons l'URL pour appeler notre service en appelant l'usine setAddressméthode. Notez que le service sera publié à cette URL.

factory.setAddress("http://localhost:5000/Hello");

Dans ce cas, le service est déployé sur le serveur intégré et écoutera le port 5000. Vous pouvez opter pour n'importe quel numéro de port de votre choix.

Avant de créer l'usine, vous devez informer l'usine de notre classe d'implémentation de service. Cela se fait en appelant lesetServiceBean méthode sur le factory objet comme montré ici -

factory.setServiceBean(new HelloWorldImpl());

Le bean de service est défini sur l'instance de notre classe d'implémentation de service. Enfin, nous créons l'usine en appelant soncreate méthode -

factory.create();

Maintenant que nous avons développé l'usine pour exécuter notre service Web, nous écrirons ensuite un main méthode pour l'instancier et le maintenir pendant un certain temps.

Maintenant, écrivez un main méthode pour instancier la HelloServer classe comme suit -

public static void main(String[] args) throws Exception {
   new HelloServer();
   System.out.println("Listening on port 5000 ...");
}

Une fois instancié, le HelloServerclass continuera à fonctionner indéfiniment. Pour les déploiements de production, vous garderez certainement votre serveur en marche pour toujours. Dans la situation actuelle, nous terminerions le serveur après un temps prédéterminé comme suit -

Thread.sleep(5 * 60 * 1000);
System.out.println("Server exiting ...");
System.exit(0);

L'ensemble du code pour le HelloServer la classe est donnée ci-dessous -

//HelloServer.java
//HelloServer.java
package com.tutorialspoint.cxf.pojo;
import org.apache.cxf.frontend.ServerFactoryBean;
public class HelloServer {
   protected HelloServer() throws Exception {
      ServerFactoryBean factory = new ServerFactoryBean();
      factory.setServiceClass(HelloWorld.class);
      factory.setAddress("http://localhost:5000/Hello");
      factory.setServiceBean(new HelloWorldImpl());
      factory.create();
   }
   public static void main(String[] args) throws Exception {
      new HelloServer();
      System.out.println("Listening on port 5000 ...");
      Thread.sleep(5 * 60 * 1000);
      System.out.println("Server exiting ...");
      System.exit(0);
   }
}

L'application serveur que nous avons créée utilise ServerFactoryBeanclasse des bibliothèques CXF. Nous devons maintenant inclure ces bibliothèques dans notre projet pour réussir à compiler leHelloServerclasse. Nous utiliseronsMaven pour configurer les dépendances du projet.

Configurer le projet Maven

Pour créer un projet Maven, tapez la commande suivante dans votre fenêtre de ligne de commande. Notez que nous avons testé cela sur une machine Mac. Pour les installations Windows et Linux, les instructions peuvent différer à certains endroits.

mvn archetype:generate

Lorsqu'on vous demande les propriétés, entrez les valeurs suivantes -

Define value for property 'groupId': : com.tutorialspoint
Define value for property 'artifactId': : cxf-pojo
Define value for property 'version': 1.0-SNAPSHOT: : 1.0
Define value for property 'package': com.tutorialspoint: : com.tutorialspoint.cxf.pojo

À la fin de la commande maven, vous trouverez la structure de dossier appropriée créée dans votre dossier actuel avec le fichier pom.xml.

La structure de répertoire générée est affichée ici -

Vous allez ajouter les dépendances CXF dans le pom.xmlet copiez également les fichiers Apache CXF créés ci-dessus dans le dossier approprié de la structure créée par maven. Pour votre référence, nous avons donné ci-dessous le fichier pom.xml du projet que nous avons créé sur notre machine.

<?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-pojo</artifactId>
   <version>1.0</version>
   <packaging>jar</packaging>
   
   <profiles>
      <profile>
         <id>server</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.cxf.pojo.HelloServer
                           </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.cxf.pojo.HelloClient
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
      </profile>
   </profiles>

   <dependencies>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-features-logging</artifactId>
         <version>3.3.0</version>
         <type>jar</type>
      </dependency>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-frontend-simple</artifactId>
         <version>3.3.0</version>
         <type>jar</type>
      </dependency>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-transports-http</artifactId>
         <version>3.3.0</version>
      </dependency>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-frontend-jaxws</artifactId>
         <version>3.3.0</version>
      </dependency>
      <!-- Jetty is needed if you're using the CXFServlet -->
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-transports-http-jetty</artifactId>
         <version>3.3.0</version>
      </dependency>
   </dependencies>
   <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>
</project>

Le pom.xml ci-dessus peut inclure des dépendances supplémentaires qui ne sont pas pertinentes pour ce projet, mais qui sont nécessaires pour notre prochain projet dans ce didacticiel. Quoi qu'il en soit, il n'y a aucun mal en tant que tel à inclure des dépendances supplémentaires.

Structure des dossiers du projet

La structure du dossier du projet sur ma machine après avoir placé les fichiers serveur et client Apache CXF est indiquée ci-dessous pour votre référence rapide -

Serveur en cours d'exécution

Pour créer le projet, utilisez la commande suivante dans votre fenêtre de ligne de commande -

mvn clean install

Vous pouvez démarrer le serveur à l'aide de la commande suivante -

mvn -Pserver

Cela démarrera le serveur et vous verrez l'invite suivante sur la console -

INFO: Creating Service {http://pojo.cxf.tutorialspoint.com/}HelloWorld from class com.tutorialspoint.cxf.pojo.HelloWorld
INFO: Setting the server's publish address to be http://localhost:5000/Hello
Listening on port 5000 ...

Maintenant, dans la fenêtre de votre navigateur, spécifiez l'URL de notre service publié. Vous verrez la sortie suivante -

Cela confirme que notre service s'exécute sur le port spécifié sur un hôte local. Puisque nous n'avons pas spécifié legreetings message dans notre appel, un message d'erreur SOAP est renvoyé au navigateur.

Vous pouvez tester davantage votre service Web à l'aide d'un client SOAP de votre choix. Ici, nous avons utilisé Postman pour tester notre serveur.

La sortie est comme indiqué ici -

Observe ceci SOAP Requesta été codé à la main. Après avoir publié la demande, le serveur a envoyé unSOAP Response message, qui est visible dans la partie inférieure de la capture d'écran.

À partir de là, vous pouvez comprendre que CXF maintient l'utilisation des protocoles SOAP pour les requêtes et les réponses tout en vous offrant une vue unifiée de diverses technologies Web qui existent dans le monde d'aujourd'hui. Cela simplifie grandement le développement d'applications Web.

Notre prochaine tâche est de créer un client qui consommera le service Web que vous avez créé.

Création du client

Dans l'application serveur HelloWorldest l'interface qui expose notre service Web. Le service Web lui-même fournit simplement un message de bienvenue au client. En général, l'interface du service Web est exposée au monde extérieur à l'aide de WSDL (Web Services Description Language). Dans cette application triviale, nous exposerons notre service Web au client en exposant directement l'interface de service et c'est leHelloWorld.class.

À cette fin, CXF fournit une classe d'usine appelée ClientProxyFactoryBean cela nous permet de nous attacher à l'interface souhaitée à l'instance d'usine créée.

Tout d'abord, nous créons une instance de bean d'usine comme suit -

ClientProxyFactoryBean factory = new ClientProxyFactoryBean();

Nous appelons le setAddresssur l'instance du bean usine pour définir l'URL par laquelle notre service Web peut être appelé. Dans notre cas, nous utiliserons l'URL utilisée lors de la création du serveur lors de notre étape précédente -

factory.setAddress("http://localhost:5000/Hello");

Ensuite, nous appelons le create méthode sur le factory instance pour attacher notre interface de service HelloWorld.class à lui.

HelloWorld helloServer = factory.create(HelloWorld.class);

Enfin, nous appelons le greetings méthode pour appeler le service Web distant.

System.out.println(helloServer.greetings(System.getProperty("user.name")));

Cela imprimerait un message d'accueil sur votre console.

La source complète de l'application cliente est indiquée ci-dessous -

//HelloClient.java
package com.tutorialspoint.cxf.pojo;
import org.apache.cxf.frontend.ClientProxyFactoryBean;
public class HelloClient {
   public static void main(String[] args) throws Exception {
      ClientProxyFactoryBean factory = new ClientProxyFactoryBean();
      factory.setAddress("http://localhost:5000/Hello");
      HelloWorld helloServer = factory.create(HelloWorld.class);
      System.out.println(helloServer.greetings(System.getProperty("user.name")));
   }
}

Client en cours d'exécution

Assurez-vous que le serveur est toujours en cours d'exécution sur votre machine. Au cas où, s'il a expiré, redémarrez le serveur avec la commande suivante -

mvn -Pserver

Vous verrez le message suivant sur la console -

Listening on port 5000 ...

Maintenant, avant que le serveur n'expire, ce que nous avons défini sur 5 minutes, ouvrez une autre fenêtre de ligne de commande et démarrez le client avec la commande suivante -

mvn -Pclient

Vous verrez un message similaire au suivant sur la ligne de commande -

Hi tutorialspoint

Notez que tutorialspointest notre nom d'utilisateur. Vous recevrez une salutation avec votre propre nom.

Dans le chapitre suivant, nous apprendrons à utiliser CXF dans un projet JAX-WS (Apache CXF API for XML Web Services).

Dans cette application JAX-WS, nous utiliserons l'approche Apache CXF-first comme la précédente application POJO. Nous allons donc d'abord créer une interface pour notre service Web.

Déclaration de l'interface de service

Comme dans le cas précédent, nous allons créer un service trivial qui n'a qu'une seule méthode d'interface appelée Greetings. Le code de l'interface de service est indiqué ci-dessous -

//HelloWorld.java
package com.tutorialspoint.cxf.jaxws.helloworld;
import javax.jws.WebService;

@WebService
public interface HelloWorld {
   String greetings(String text);
}

Nous annotons l'interface avec un @WebServicemarque. Ensuite, nous allons implémenter cette interface.

Implémentation de l'interface Web

La mise en œuvre de l'interface Web est illustrée ici -

//HelloWorldImpl.java
package com.tutorialspoint.cxf.jaxws.helloworld;
public class HelloWorldImpl implements HelloWorld {
   @Override
   public String greetings(String name) {
      return ("hi " + name);
   }
}

La méthode des salutations est annotée avec @Overridemarque. La méthode renvoie un message "salut" à l'appelant.

Ensuite, nous écrirons le code pour développer le serveur.

Développement du serveur

Contrairement à l'application POJO, nous allons maintenant découpler l'interface en utilisant la classe Endpoint fournie par CXF pour publier notre service. Cela se fait dans les deux lignes de code suivantes -

HelloWorld implementor = new HelloWorldImpl();
Endpoint.publish(
   "http://localhost:9090/HelloServerPort",
   implementor,
   new LoggingFeature()
);

Le premier paramètre de la méthode de publication spécifie l'URL à laquelle notre service sera mis à disposition des clients. Le deuxième paramètre spécifie la classe d'implémentation de notre service. Le code complet du serveur est affiché ci-dessous -

//Server.java
package com.tutorialspoint.cxf.jaxws.helloworld;
import javax.xml.ws.Endpoint;
import org.apache.cxf.ext.logging.LoggingFeature;
public class Server {
   public static void main(String[] args) throws Exception {
      HelloWorld implementor = new HelloWorldImpl();
      Endpoint.publish("http://localhost:9090/HelloServerPort",
      implementor,
      new LoggingFeature());
      System.out.println("Server ready...");
      Thread.sleep(5 * 60 * 1000);
      System.out.println("Server exiting ...");
      System.exit(0);
   }
}

Pour déployer notre serveur, vous devrez apporter quelques modifications supplémentaires à votre projet comme indiqué ci-dessous.

Déploiement du serveur

Enfin, pour déployer l'application serveur, vous devrez effectuer une autre modification dans pom.xml pour configurer votre application en tant qu'application Web. Le code que vous devez ajouter à votrepom.xml est donné ci-dessous -

<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.cxf.jaxws.helloworld.Server
                        </mainClass>
                     </configuration>
                  </execution>
               </executions>
            </plugin>
         </plugins>
      </build>
   </profile>
</profiles>

Avant de déployer l'application, vous devez ajouter deux fichiers supplémentaires à votre projet. Ceux-ci sont indiqués dans la capture d'écran ci-dessous -

Ces fichiers sont des fichiers standard CXF qui définissent le mappage pour CXFServlet. Le code dans leweb.xml Le fichier est affiché ici pour votre référence rapide -

//Web.xml
<?xml version = "1.0" encoding = "UTF-8"??>
<web-app xmlns = "http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.5"
xsi:schemaLocation = "http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
   <display-name>cxf</display-name>
   <servlet>
      <description>Apache CXF Endpoint</description>
      <display-name>cxf</display-name>
      <servlet-name>cxf</servlet-name>
      <servlet-class>
         org.apache.cxf.transport.servlet.CXFServlet
      </servlet-class>
      <load-on-startup>
         1
      </load-on-startup>
   </servlet>
   <servlet-mapping>
      <servlet-name>
         cxf
      </servlet-name>
      <url-pattern>
         /services/*
      </url-pattern>
   </servlet-mapping>
   <session-config>
      <session-timeout>60</session-timeout>
   </session-config>
</web-app>

dans le cxf-servlet.xml,vous déclarez les propriétés du point de terminaison de votre service. Ceci est montré dans l'extrait de code ci-dessous -

<beans ...>
   <jaxws:endpoint xmlns:helloworld = "http://tutorialspoint.com/"
      id = "helloHTTP"
      address = "http://localhost:9090/HelloServerPort"
      serviceName = "helloworld:HelloServiceService"
      endpointName = "helloworld:HelloServicePort">
   </jaxws:endpoint>
</beans>

Ici, nous définissons l'id de notre point de terminaison de service, l'adresse sur laquelle le service sera disponible, le nom du service et le nom du point de terminaison. Vous avez maintenant appris comment votre service est acheminé et traité par un servlet CXF.

Le pom.xml final

le pom.xmlinclut quelques dépendances supplémentaires. Plutôt que de décrire toutes les dépendances, nous avons inclus la version finale de pom.xml ci-dessous -

<?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-jaxws</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.cxf.jaxws.helloworld.Server
                           </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.cxf.jaxws.helloworld.Client
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
      </profile>
   </profiles>
   <dependencies>
      <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-http</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>

Notez qu'il comprend également un profil de création de client que nous apprendrons dans les sections ultérieures de ce didacticiel.

Exécution du service HelloWorld

Vous êtes maintenant prêt à exécuter l'application Web. Dans la fenêtre de commande, exécutez le script de construction à l'aide de la commande suivante.

mvn clean install
mvn -Pserver

Vous verrez le message suivant sur la console -

INFO: Setting the server's publish address to be http://localhost:9090/HelloServerPort
Server ready…

Comme précédemment, vous pouvez tester le serveur en ouvrant l'URL du serveur dans votre navigateur.

Comme nous n'avons précisé aucune opération, seul un message d'erreur est renvoyé au navigateur par notre application.

Maintenant, essayez d'ajouter le ?wsdl à votre URL et vous verrez la sortie suivante -

Notre application serveur fonctionne donc comme prévu. Vous pouvez utiliser le client SOAP tel quePostman décrit précédemment pour tester davantage votre service.

Dans la section suivante, nous allons apprendre à écrire un client qui utilise notre service.

Client en développement

L'écriture du client dans une application CXF est aussi simple que l'écriture d'un serveur. Voici le code complet pour le client -

//Client.java
package com.tutorialspoint.cxf.jaxws.helloworld;
import javax.xml.namespace.QName;
import javax.xml.ws.Service;
import javax.xml.ws.soap.SOAPBinding;
public final class Client {
   private static final QName SERVICE_NAME
   = new QName("http://helloworld.jaxws.cxf.tutorialspoint.com/",
   "HelloWorld");
   private static final QName PORT_NAME
   = new QName("http://helloworld.jaxws.cxf.tutorialspoint.com/",
   "HelloWorldPort");
   private Client() {
   }
   public static void main(String[] args) throws Exception {
      Service service = Service.create(SERVICE_NAME);
      System.out.println("service created");
      String endpointAddress = "http://localhost:9090/HelloServerPort";
      service.addPort(PORT_NAME, SOAPBinding.SOAP11HTTP_BINDING,
      endpointAddress);
      HelloWorld hw = service.getPort(HelloWorld.class);
      System.out.println(hw.greetings("World"));
   }
}

Ici, nous utilisons le CXF fourni Serviceclass pour se lier au service connu. Nous appelons lecreate méthode sur le Serviceclass pour obtenir une instance du service. Nous définissons le port connu en appelant leaddPort méthode sur le service exemple.

Maintenant, nous sommes prêts à consommer le service, ce que nous faisons en obtenant d'abord l'interface de service en appelant le getPort méthode sur le serviceexemple. Enfin, nous appelons notregreetings méthode pour imprimer le message d'accueil sur la console.

Maintenant que vous avez appris les bases de CXF en utilisant l'approche Apache CXF-First, vous allez maintenant apprendre à utiliser CXF avec l'approche WSDL-First dans notre prochain chapitre.

L'application CXF-POJO que vous avez développée se traduit par un couplage très étroit entre le client et le serveur. Donner un accès direct à l'interface de service peut également poser de graves menaces pour la sécurité. Ainsi, un découplage entre le client et le serveur est généralement souhaité, ce qui est réalisé en utilisant WSDL (Web Services Description Language).

Nous écrivons l'interface du service Web dans un document WSDL basé sur XML. Nous utiliserons un outil pour mapper ce WSDL aux interfaces Apache CXF qui sont ensuite implémentées et utilisées par nos applications client et serveur. Pour assurer le découplage, commencer par un WSDL est un moyen préféré. Pour cela, vous devez d'abord apprendre un nouveau langage - WSDL. L'écriture de WSDL nécessite une approche prudente et il serait préférable que vous puissiez mieux comprendre ce sujet avant de commencer à travailler dessus.

Dans cette leçon, nous commencerons par définir une interface de service Web dans un document WSDL. Nous allons apprendre à utiliser CXF pour créer à la fois des applications serveur et client en commençant par WSDL. Nous allons garder l'application simple pour maintenir l'accent sur l'utilisation de CXF. Une fois l'application serveur créée, nous la publierons sur l'URL souhaitée à l'aide d'une classe CXF intégrée.

Tout d'abord, décrivons le WSDL que nous allons utiliser.

WSDL pour HelloWorld

Le webservice que nous allons implémenter aura une seule méthode web appelée greetings qui accepte un stringparamètre contenant le nom d'utilisateur et renvoie un message sous forme de chaîne à l'appelant après avoir ajouté un message d'accueil au nom d'utilisateur. Le wsdl complet est montré ci-dessous -

//Hello.wsdl
<?xml version = "1.0" encoding = "UTF-8"?>
<wsdl:definitions xmlns:soap = "http://schemas.xmlsoap.org/wsdl/soap/"
   xmlns:tns = "http://helloworld.tutorialspoint.com/"
   xmlns:wsdl = "http://schemas.xmlsoap.org/wsdl/"
   xmlns:xsd = "http://www.w3.org/2001/XMLSchema"
   name = "HelloWorld"
   targetNamespace = "http://helloworld.tutorialspoint.com/">
   <wsdl:types>
      <xsd:schema attributeFormDefault = "unqualified"
         elementFormDefault = "qualified"
         targetNamespace = "http://helloworld.tutorialspoint.com/">
         <xsd:element name = "greetings" type = "tns:greetings"/>
         <xsd:complexType name = "greetings">
            <xsd:sequence>
               <xsd:element minOccurs = "0" name = "arg0" type = "xsd:string"/>
            </xsd:sequence>
         </xsd:complexType>
         <xsd:element name = "greetingsResponse"
         type = "tns:greetingsResponse"/>
         <xsd:complexType name = "greetingsResponse">
            <xsd:sequence>
               <xsd:element minOccurs = "0" name = "return" type = "xsd:string"/>
            </xsd:sequence>
         </xsd:complexType>
      </xsd:schema>
   </wsdl:types>
   <wsdl:message name = "greetings">
      <wsdl:part element = "tns:greetings" name = "parameters"> </wsdl:part>
   </wsdl:message>
   <wsdl:message name = "greetingsResponse">
      <wsdl:part element = "tns:greetingsResponse" name = "parameters"> </wsdl:part>
   </wsdl:message>
   <wsdl:portType name = "HelloWorldPortType">
      <wsdl:operation name = "greetings">
         <wsdl:input message = "tns:greetings" name = "greetings">  </wsdl:input>
         <wsdl:output message = "tns:greetingsResponse" name = "greetingsResponse">
         </wsdl:output>
      </wsdl:operation>
   </wsdl:portType>
   <wsdl:binding name = "HelloWorldSoapBinding" type = "tns:HelloWorldPortType">
      <soap:binding style = "document"
      transport = "http://schemas.xmlsoap.org/soap/http"/>
      <wsdl:operation name = "greetings">
         <soap:operation soapAction = "" style = "document"/>
         <wsdl:input name = "greetings"></wsdl:input>
         <wsdl:output name = "greetingsResponse">
            <soap:body use = "literal"/>
         </wsdl:output>
         </wsdl:operation>
   </wsdl:binding>
   <wsdl:service name = "HelloWorldService">
      <wsdl:port binding = "tns:HelloWorldSoapBinding" name = "HelloWorldPort">
         <soap:address location = "http://localhost:9090/HelloServerPort"/>
      </wsdl:port>
   </wsdl:service>
</wsdl:definitions>

Notez que l'écriture d'un wsdl syntaxiquement correct a toujours été un défi pour les développeurs; il existe de nombreux outils et des éditeurs en ligne sont disponibles pour créer un wsdl. Ces éditeurs demandent les noms des messages que vous souhaitez implémenter ainsi que les paramètres que vous souhaitez passer dans un message et le type de message de retour que vous souhaitez que votre application cliente reçoive. Si vous connaissez la syntaxe wsdl, vous pouvez coder manuellement l'intégralité du document ou utiliser l'un des éditeurs pour créer le vôtre.

Dans le wsdl ci-dessus, nous avons défini un seul message appelé greetings. Le message est remis au service appeléHelloWorldService qui court à http://localhost:9090/HelloServerPort.

Avec cela, nous allons maintenant procéder au développement du serveur. Avant de développer le serveur, nous devons générer une interface Apache CXF avec notre service Web. Cela doit être fait à partir du wsdl donné. Pour ce faire, vous utilisez un outil appeléwsdl2java.

Le plugin wsdl2java

Comme nous allons utiliser maven pour construire le projet, vous devrez ajouter le plugin suivant au pom.xml fichier.

<plugins>
   <plugin>
      <groupId>org.apache.cxf</groupId>
      <artifactId>cxf-codegen-plugin</artifactId>
      <version>3.3.0</version>
      <executions>
         <execution>
            <id>generate-sources</id>
            <phase>generate-sources</phase>
            <configuration>
               <wsdlOptions>
                  <wsdlOption>
                     <wsdl>src/main/resources/hello.wsdl</wsdl>
                     <faultSerialVersionUID> 1 </faultSerialVersionUID>
                  </wsdlOption>
               </wsdlOptions>
            </configuration>
            <goals>
               <goal>wsdl2java</goal>
            </goals>
         </execution>
      </executions>
   </plugin>
</plugins>

Notez que nous spécifions l'emplacement du wsdl classer comme src/main/resources/Hello.wsdl. Vous devrez vous assurer que vous créez une structure de répertoires appropriée pour votre projet et ajoutez lehello.wsdl fichier dans le dossier spécifié.

le wsdl2javaplugin compilera ce wsdl et créera des classes Apache CXF dans un dossier prédéfini. La structure complète du projet est présentée ici pour votre référence.

Vous êtes maintenant prêt à créer un serveur en utilisant le wsdl2javaclasses générées. Les classes créées par wsdl2java sont illustrées dans la figure ci-dessous -

Interface de service générée

Dans la liste des classes générées, vous devez avoir remarqué que l'une d'elles est une interface Apache CXF - c'est HelloWorldPortType.java. Examinez ce fichier dans votre éditeur de code. Le contenu du fichier est affiché ici pour votre référence prête -

//HelloWorldPortType.java
package com.tutorialspoint.helloworld;
import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebResult;
import javax.jws.WebService;
import javax.xml.bind.annotation.XmlSeeAlso;
import javax.xml.ws.RequestWrapper;
import javax.xml.ws.ResponseWrapper;
/**
* This class was generated by Apache CXF 3.3.0
* 2019-02-11T12:05:55.220+05:30
* Generated source version: 3.3.0
*
*/

@WebService(targetNamespace = "http://helloworld.tutorialspoint.com/",
   name = "HelloWorldPortType")
@XmlSeeAlso({ObjectFactory.class})
public interface HelloWorldPortType {
   @WebMethod
   @RequestWrapper(localName = "greetings", targetNamespace =
      "http://helloworld.tutorialspoint.com/", className =
      "com.tutorialspoint.helloworld.Greetings")
      @ResponseWrapper(localName = "greetingsResponse", targetNamespace =
         "http://helloworld.tutorialspoint.com/", className =
         "com.tutorialspoint.helloworld.GreetingsResponse")
   @WebResult(name = "return", targetNamespace =
      "http://helloworld.tutorialspoint.com/")
   public java.lang.String greetings(
      @WebParam(name = "arg0", targetNamespace =
      "http://helloworld.tutorialspoint.com/")
      java.lang.String arg0
   );
}

Notez que l'interface contient une méthode appelée greetings. C'était un type de message dans notre wsdl. lewsdl2javatool a ajouté cette méthode à l'interface générée. Maintenant, vous pouvez comprendre que quels que soient les messages que vous écrivez dans votre wsdl, une méthode correspondante serait générée dans l'interface.

Maintenant, votre tâche serait d'implémenter toutes ces méthodes correspondant aux différents messages que vous avez définis dans votre wsdl. Notez que dans l'exemple précédent d'Apache CXF-First, nous avons commencé avec une interface Apache CXF pour notre service Web. Dans ce cas, l'interface Apache CXF est créée à partir de wsdl.

Implémentation de l'interface de service

L'implémentation de l'interface de service est triviale. La mise en œuvre complète est indiquée dans la liste ci-dessous -

//HelloWorldImpl.java
package com.tutorialspoint.helloworld;
public class HelloWorldImpl implements HelloWorldPortType {
   @Override
   public String greetings(String name) {
      return ("hi " + name);
   }
}

Le code implémente la méthode d'interface unique appelée greetings. La méthode prend un paramètre destring type, y ajoute un message "salut" et renvoie la chaîne résultante à l'appelant.

Ensuite, nous écrirons l'application serveur.

Développement du serveur

Le développement d'une application serveur est une fois de plus trivial. Ici, nous utiliserons le CXF fourniEndpointclasse pour publier notre service. Cela se fait dans les deux lignes de code suivantes -

HelloWorldPortType implementor = new HelloWorldImpl();
   Endpoint.publish("http://localhost:9090/HelloServerPort",
      implementor,
      new LoggingFeature());

Tout d'abord, nous créons un objet de notre classe d'implémenteur de service - HelloWorldImpl. Ensuite, nous passons cette référence en tant que deuxième paramètre aupublishméthode. Le premier paramètre est l'adresse à laquelle le service est publié - les clients utiliseraient cette URL pour accéder au service. La source complète de l'application serveur est donnée ici -

//Server.java
package com.tutorialspoint.helloworld;
import javax.xml.ws.Endpoint;
import org.apache.cxf.ext.logging.LoggingFeature;
public class Server {
   public static void main(String[] args) throws Exception {
      HelloWorldPortType implementor = new HelloWorldImpl();
      Endpoint.publish("http://localhost:9090/HelloServerPort",
         implementor,
         new LoggingFeature());
      System.out.println("Server ready...");
      Thread.sleep(5 * 60 * 1000);
      System.out.println("Server exiting");
      System.exit(0);
   }
}

Pour créer cette classe de serveur, vous devrez ajouter un profil de construction dans votre pom.xml. Ceci est montré ci-dessous -

<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.helloworld.Server
                     </mainClass>
                  </configuration>
               </execution>
            </executions>
         </plugin>
      </plugins>
   </build>
   <dependencies>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-transports-http-jetty</artifactId>
         <version>3.3.0</version>
      </dependency>
   </dependencies>
</profile>

Notez que le nom complet du Serverla classe est spécifiée dans la configuration. En outre, la balise de dépendance spécifie que nous utiliserons le serveur Web de jetée intégré pour déployer notre application serveur.

Déploiement du serveur

Enfin, pour déployer l'application serveur, vous devrez effectuer une autre modification dans pom.xml pour configurer votre application en tant qu'application Web. Le code que vous devez ajouter à votrepom.xml est donné ci-dessous -

<defaultGoal>install</defaultGoal>
<pluginManagement>
   <plugins>
      <plugin>
         <artifactId>maven-war-plugin</artifactId>
         <version>3.2.2</version>
         <configuration>
            <webXml>src/main/webapp/WEB-INF/web.xml</webXml>
            <webResources>
               <resource>
                  <directory>src/main/resources</directory>
                  <targetPath>WEB-INF</targetPath>
                  <includes>
                     <include>*.wsdl</include>
                  </includes>
               </resource>
            </webResources>
         </configuration>
      </plugin>
   </plugins>
</pluginManagement>

Avant de déployer l'application, vous devez ajouter deux fichiers supplémentaires à votre projet. Ceux-ci sont indiqués dans la capture d'écran ci-dessous -

Ces fichiers sont des fichiers standard CXF qui définissent le mappage pour CXFServlet. Le code dans leweb.xml Le fichier est affiché ici pour votre référence rapide -

//cxf-servlet.xml
<web-app xmlns = "http://java.sun.com/xml/ns/javaee"
   xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" version="2.5"
   xsi:schemaLocation = "http://java.sun.com/xml/ns/javaee
   http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
   <display-name>cxf</display-name>
   <servlet>
      <description>Apache CXF Endpoint</description>
      <display-name>cxf</display-name>
      <servlet-name>cxf</servlet-name>
      <servlet-class>
         org.apache.cxf.transport.servlet.CXFServlet
      </servlet-class>
      <load-on-startup>1</load-on-startup>
   </servlet>
   <servlet-mapping>
      <servlet-name>cxf</servlet-name>
      <url-pattern>/services/*</url-pattern>
   </servlet-mapping>
   <session-config>
      <session-timeout>60</session-timeout>
   </session-config>
</web-app>

dans le cxf-servlet.xmlvous déclarez les propriétés du point de terminaison de votre service. Ceci est montré dans l'extrait de code ci-dessous -

<beans ...>
   <jaxws:endpoint xmlns:helloworld = "http://tutorialspoint.com/"
      id="helloHTTP"
      address = "http://localhost:9090/HelloServerPort"
      serviceName = "helloworld:HelloServiceService"
      endpointName = "helloworld:HelloServicePort">
   </jaxws:endpoint>
</beans>

Ici, nous définissons l'id de notre point de terminaison de service, l'adresse sur laquelle le service sera disponible, le nom du service et le nom du point de terminaison. Vous comprenez maintenant comment votre service est acheminé et traité par un servlet CXF.

Le pom.xml final

le pom.xmlinclut quelques dépendances supplémentaires. Plutôt que de décrire toutes les dépendances, nous avons inclus la version finale de pom.xml ci-dessous -

<?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-wsdl</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>
   <build>
      <defaultGoal>install</defaultGoal>
      <pluginManagement>
         <plugins>
            <plugin>
               <artifactId>maven-war-plugin</artifactId>
               <version>3.2.2</version>
               <configuration>
                  <webXml>src/main/webapp/WEB-INF/web.xml</webXml>
                  <webResources>
                     <resource>
                        <directory>src/main/resources</directory>
                        <targetPath>WEB-INF</targetPath>
                        <includes>
                           <include>*.wsdl</include>
                        </includes>
                     </resource>
                  </webResources>
               </configuration>
            </plugin>
         </plugins>
      </pluginManagement>
      <plugins>
         <plugin>
            <groupId>org.apache.cxf</groupId>
            <artifactId>cxf-codegen-plugin</artifactId>
            <version>3.3.0</version>
            <executions>
               <execution>
                  <id>generate-sources</id>
                  <phase>generate-sources</phase>
                  <configuration>
                     <wsdlOptions>
                        <wsdlOption>
                           <wsdl>src/main/resources/Hello.wsdl</wsdl>
                           <faultSerialVersionUID>1</faultSerialVersionUID>
                        </wsdlOption>
                     </wsdlOptions>
                  </configuration>
                  <goals>
                     <goal>wsdl2java</goal>
                  </goals>
               </execution>
            </executions>
         </plugin>
      </plugins>
   </build>
   <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.helloworld.Server
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
         <dependencies>
            <dependency>
               <groupId>org.apache.cxf</groupId>
               <artifactId>cxf-rt-transports-http-jetty</artifactId>
               <version>3.3.0</version>
            </dependency>
         </dependencies>
      </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.helloworld.Client
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
      </profile>
   </profiles>
   <dependencies>
      <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-http</artifactId>
         <version>3.3.0</version>
      </dependency>
      
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-management</artifactId>
         <version>3.3.0</version>
      </dependency>
      
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-features-metrics</artifactId>
         <version>3.3.0</version>
      </dependency>
      
      <dependency>
         <groupId>org.apache.cxf.xjc-utils</groupId>
         <artifactId>cxf-xjc-runtime</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.codehaus.mojo</groupId>
         <artifactId>exec-maven-plugin</artifactId>
         <version>1.6.0</version>
      </dependency>
      
      <dependency>
         <groupId>org.slf4j</groupId>
         <artifactId>slf4j-api</artifactId>
         <version>1.8.0-beta2</version>
      </dependency>
      
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-transports-http-jetty</artifactId>
         <version>3.3.0</version>
      </dependency>
   </dependencies>
</project>

Notez qu'il comprend également un profil pour la création de clients que nous apprendrons bientôt dans les sections suivantes.

Exécution du service HelloWorld

Vous êtes maintenant prêt à exécuter l'application Web. Dans la fenêtre de commande, exécutez le script de construction à l'aide de la commande suivante.

mvn clean install

Cela générera les classes Apache CXF appropriées à partir de votre wsdl, compilera vos classes Apache CXF, déploiera le serveur sur le serveur de jetée intégré et exécutera votre application.

Vous verrez le message suivant sur la console -

INFO: Setting the server's publish address to be 
http://localhost:9090/HelloServerPort
Server ready...

Comme précédemment, vous pouvez tester le serveur en ouvrant l'URL du serveur dans votre navigateur.

Comme nous n'avons précisé aucune opération, seul un message d'erreur est renvoyé au navigateur par notre application. Maintenant, essayez d'ajouter le?wsdl à votre URL et vous verrez la sortie suivante -

Notre application serveur fonctionne donc comme prévu. Vous pouvez utiliser le client SOAP tel quePostman décrit précédemment pour tester davantage votre service.

La prochaine partie de ce tutoriel consiste à écrire un client qui utilise notre service.

Client en développement

L'écriture du client dans une application CXF est aussi importante que l'écriture d'un serveur. Voici le code complet du client qui se compose essentiellement de seulement trois lignes, le reste des lignes imprime simplement les informations de service à l'utilisateur.

//Client.java
package com.tutorialspoint.helloworld;
public class Client {
   public static void main(String[] args) throws Exception {
      //Create the service client with its default wsdlurl
      HelloWorldService helloServiceService = new HelloWorldService();
      System.out.println("service: " +
         helloServiceService.getServiceName());
      System.out.println("wsdl location: " +
         helloServiceService.getWSDLDocumentLocation());
      HelloWorldPortType helloService =
         helloServiceService.getHelloWorldPort();
      System.out.println(helloService.greetings
      (System.getProperty("user.name")));
   }
}

Ici, nous créons simplement une instance de notre service HelloWorldService, récupérez son port en appelant getHelloWorldPort méthode, puis passez notre greetingsmessage à lui. Exécutez le client et vous verrez la sortie suivante -

service: {http://helloworld.tutorialspoint.com/}HelloWorldService
wsdl location: file:/Users/drsarang/Desktop/tutorialpoint/cxf-
wsdl/src/main/resources/Hello.wsdl
hi drsarang

Jusqu'à présent, vous avez appris à utiliser CXF avec les architectures Apache CXF-First et WSDL-First. Dans l'approche Apache CXF-First, vous avez utilisé un POJO avecServerFactoryBeanclasse des bibliothèques CXF pour créer un serveur. Pour créer un client que vous avez utiliséClientProxyFactoryBeanclasse de la bibliothèque CXF. Dans l'approche WSDL-First, vous avez utiliséEndpointclasse pour publier le service à l'URL souhaitée et un implémenteur spécifié. Vous pouvez désormais étendre ces techniques pour intégrer différents protocoles et transports.

Avant de poursuivre dans ce chapitre, nous supposons que vous savez comment écrire un service Web RESTful en Java. Je vais vous montrer comment utiliser CXF en plus de ce JAX-RS (API Java pour les services Web RESTful). Nous allons créer un service Web qui maintient une liste des derniers films. Lorsque l'utilisateur demande un film, il spécifie l'ID du film dans sa demande, le serveur localisera le film et le retournera au client. Dans notre cas trivial, nous retournerons simplement le nom du film au client et non le fichier MP4 binaire réel. Commençons donc par créer une application JAX-RS.

Déclaration d'un élément de film

Nous déclarerons un élément racine XML appelé Movie pour stocker l'id et le nom d'un film donné. L'élément est déclaré dans un fichier appelé Movie.java. Le contenu du fichier est affiché ici -

//Movie.java
package com.tutorialspoint.cxf.jaxrs.movie;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement(name = "Movie")
public class Movie {
   private long id;
   private String name;
   public long getId() {
      return id;
   }
   public void setId(long id) {
      this.id = id;
   }
   public String getName() {
      return name;
   }
   public void setName(String name) {
      this.name = name;
   }
}

Notez l'utilisation de XmlRootElement balise pour déclarer l'élément XML pour le Moviemarque. Ensuite, nous allons créer un service qui contient la liste des films dans sa base de données.

Création de la base de données Movie Service

Pour stocker la liste des films, nous utilisons Java fourni Mapqui stocke les paires clé-valeur. Si la liste est longue, vous utiliserez un stockage de base de données externe qui sera également plus facile à gérer. Dans notre cas trivial, nous ne stockerons que cinq films dans notre base de données. Le code de la classe MovieService est donné ci-dessous -

//MovieService.java
package com.tutorialspoint.cxf.jaxrs.movie;
import java.util.HashMap;
import java.util.Map;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
@Path("/movieservice/")
@Produces("text/xml")
public class MovieService {
   long currentId = 123;
   Map<Long, Movie> movies = new HashMap<>();
   public MovieService() {
      init();
   }
   @GET
   @Path("/movie/{id}/")
   public Movie getMovie(@PathParam("id") String id) {
      long idNumber = Long.parseLong(id);
      return movies.get(idNumber);
   }
   final void init() {
      Movie c1 = new Movie();
      c1.setName("Aquaman");
      c1.setId(1001);
      movies.put(c1.getId(), c1);
      
      Movie c2 = new Movie();
      c2.setName("Mission Imposssible");
      c2.setId(1002);
      movies.put(c2.getId(), c2);
      
      Movie c3 = new Movie();
      c3.setName("Black Panther");
      c3.setId(1003);
      movies.put(c3.getId(), c3);
      
      Movie c4 = new Movie();
      c4.setName("A Star is Born");
      c4.setId(1004);
      movies.put(c4.getId(), c4);
      
      Movie c5 = new Movie();
      c5.setName("The Meg");
      c5.setId(1005);
      movies.put(c5.getId(), c5);
   }
}

Notez que nous utilisons les deux annotations suivantes pour spécifier le chemin d'URL de notre service de film et son type de retour -

@Path("/movieservice/")
@Produces("text/xml")

Nous utilisons les annotations @GET et @Path pour spécifier l'URL de la requête GET comme suit -

@GET
@Path("/movie/{id}/")

La base de données de films elle-même est initialisée dans la méthode init, où nous ajoutons cinq éléments de film à la base de données.

Notre prochaine tâche est d'écrire une application serveur.

Développement du serveur

Pour créer un serveur, nous utilisons CXF fourni JAXRSServerFactoryBean classe.

JAXRSServerFactoryBean factory = new JAXRSServerFactoryBean();

Nous définissons ses classes de ressources en appelant le setResourceClasses méthode.

factory.setResourceClasses(Movie.class);
factory.setResourceClasses(MovieService.class);

Nous définissons le fournisseur de services en appelant le setResourceProvider méthode.

factory.setResourceProvider(MovieService.class,
new SingletonResourceProvider(new MovieService()));

Nous définissons le souhaité publish adresse en appelant le aetAddress méthode -

factory.setAddress("http://localhost:9000/");

Enfin, nous publions le serveur en appelant la méthode create sur le factory exemple.

factory.create();

Le code complet de l'application serveur est donné ci-dessous -

//Server.java
package com.tutorialspoint.cxf.jaxrs.movie;
import org.apache.cxf.jaxrs.JAXRSServerFactoryBean;
import org.apache.cxf.jaxrs.lifecycle.SingletonResourceProvider;
public class Server {
   public static void main(String[] args) throws Exception {
      JAXRSServerFactoryBean factory = new JAXRSServerFactoryBean();
      factory.setResourceClasses(Movie.class);
      factory.setResourceClasses(MovieService.class);  
      factory.setResourceProvider(MovieService.class,
         new SingletonResourceProvider(new MovieService()));
      factory.setAddress("http://localhost:9000/");
      factory.create();
      
      System.out.println("Server ready...");
      Thread.sleep(5 * 60 * 1000);
      
      System.out.println("Server exiting ...");
      System.exit(0);
   }
}

Le pom.xml final

Ici, nous avons inclus la version finale de pom.xml ci-dessous -

<?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-jaxrs</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.cxf.jaxrs.movie.Server
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
         <dependencies>
            <dependency>
               <groupId>org.apache.cxf</groupId>
               <artifactId>cxf-rt-transports-http-jetty</artifactId>
               <version>3.3.0</version>
            </dependency>
         </dependencies>
      </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.cxf.jaxrs.movie.Client
                           </mainClass>
                        </configuration>
                     </execution>
                  </executions>
               </plugin>
            </plugins>
         </build>
      </profile>
   </profiles>
   <dependencies>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-transports-http</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>
      <dependency>
         <groupId>org.apache.cxf</groupId>
         <artifactId>cxf-rt-frontend-jaxrs</artifactId>
         <version>3.3.0</version>
         </dependency>
      <dependency>
         <groupId>jakarta.ws.rs</groupId>
         <artifactId>jakarta.ws.rs-api</artifactId>
         <version>2.1.5</version>
      </dependency>
      <dependency>
         <groupId>org.apache.httpcomponents</groupId>
         <artifactId>httpclient</artifactId>
         <version>4.5.7</version>
      </dependency>
   </dependencies>
</project>

Client en développement

L'écriture du client RS est triviale. Nous créons simplement un objet URL et ouvrons son flux. Nous utilisons la classe IOUtils fournie par CXF pour copier le contenu du flux d'entrée dans un flux local.

URL url = new URL("http://localhost:9000/movieservice/movie/1002");
try (InputStream instream = url.openStream();
CachedOutputStream outstream = new CachedOutputStream()) {
   IOUtils.copy(instream, outstream);
}

Le code complet de l'application client est donné ci-dessous -

//Client.java
package com.tutorialspoint.cxf.jaxrs.movie;
import java.io.InputStream;
import java.net.URL;
import org.apache.cxf.helpers.IOUtils;
import org.apache.cxf.io.CachedOutputStream;
public class Client {
   public static void main(String[] args) throws Exception {
      URL url = new URL("http://localhost:9000/movieservice/movie/1002");
      try (InputStream instream = url.openStream();
      CachedOutputStream outstream = new CachedOutputStream()) {
         IOUtils.copy(instream, outstream);
         String str = outstream.getOut().toString();
         System.out.println(str);
      }
   }
}

Test de l'application JAX-RS

Exécutez le serveur en utilisant la commande suivante dans la fenêtre de ligne de commande -

mvn -Pserver

Maintenant, vous verrez le message suivant sur la console -

INFO: Setting the server's publish address to be http://localhost:9000

Maintenant, ouvrez votre navigateur et tapez l'URL suivante -

http://localhost:9000/movieservice/movie/1002

Vous verrez ce qui suit dans la fenêtre du navigateur.

Vous pouvez appeler le service en utilisant une application client Java que nous avons développée en exécutant la commande suivante dans une fenêtre de ligne de commande distincte.

mvn -Pclient

Vous verrez la sortie suivante -

<?xml version="1.0" encoding = "UTF-8" standalone="yes"?>
<Movie><id>1002</id><name>Mission Imposssible</name></Movie>

Les exemples CXF fournissent plusieurs exemples d'utilisation de CXF avec JAX-RS. Les lecteurs intéressés sont encouragés à étudier ces échantillons.

Comme mentionné précédemment, vous pouvez utiliser CXF avec le transport JMS. Dans ce cas, le client enverra un message JMS à un serveur de messagerie connu. Notre application serveur écoute en permanence le serveur de messagerie pour les messages entrants. Lorsque le message arrive, il traite le message, exécute la demande du client et envoie la réponse comme un autre message au client.

Comme précédemment, nous allons d'abord créer un exemple d'application serveur qui fournit une méthode Web unique appelée sayHi.

Création d'une interface de service

L'interface de service pour notre HelloWorld le service est montré ici -

//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);
}

Service de mise en œuvre

L'implémentation de l'interface de service est définie comme suit -

//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;
   }
}

L'implémentation renvoie simplement un message Hello à l'utilisateur. Comme vous le voyez, l'interface et son implémentation sont similaires à tous les projets précédents de ce tutoriel que vous avez étudiés jusqu'à présent.

Maintenant, vient le point le plus important qui est de créer une application serveur qui configure une file d'attente de messages et continue d'écouter les messages entrants.

Création du serveur

Dans l'application serveur, nous créons d'abord un JMS point final comme suit -

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";

Notez que nous avons configuré une file d'attente sur un port spécifié qui vit pendant une durée spécifiée. Nous créons maintenant un service de messagerie en instanciantorg.apache.activemq.broker.BrokerServiceclasse. Ceci est une classe de serveur pourActiveMQ serveur de messagerie.

BrokerService broker = new BrokerService();

Vous pouvez utiliser tout autre serveur de messagerie de votre choix autre que ActiveMQ. Nous connectons maintenant ce serveur à un URI souhaité.

broker.addConnector("tcp://localhost:61616");

Nous mettons en place le répertoire pour le stockage des données des messages entrants -

broker.setDataDirectory("target/activemq-data");

Enfin, nous démarrons le serveur en utilisant la méthode start -

broker.start();

Ensuite, nous créons une instance de notre bean service HelloWorld en utilisant la classe de bean de fabrique de serveur utilisée dans notre application POJO précédente -

Object implementor = new HelloWorldImpl();
JaxWsServerFactoryBean factory = new JaxWsServerFactoryBean();
factory.setServiceClass(HelloWorld.class);

Ensuite, nous configurons le point de terminaison JMS sur l'usine afin que l'usine continue à écouter les messages entrants -

factory.setTransportId
(JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress(JMS_ENDPOINT_URI);

Enfin, nous configurons la classe de mise en œuvre dans l'usine et commençons à l'exécuter -

factory.setServiceBean(implementor);
factory.create();

À ce stade, votre serveur est opérationnel. Notez que puisque nous avons utilisé la classe de bean usine comme dans l'application POJO, le besoin de CXFServlet et du fichier web.xml n'est pas requis.

Le code d'application serveur complet est affiché ici -

//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);
   }
}

Ajouter des dépendances

L'application serveur que nous avons créée utilise le serveur de messagerie ActiveMQ. Ainsi, vous devrez ajouter quelques dépendances supplémentaires à votre projet. Le fichier pom.xml complet est affiché ici pour que vous puissiez comprendre les dépendances supplémentaires nécessaires.

<?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>

Serveur en cours d'exécution

Pour commencer à exécuter le serveur, comme dans les cas précédents, tapez la commande suivante dans votre fenêtre de commande -

mvn -Pserver

Cela démarrera le serveur de messages ActiveMQ, configurera la file d'attente de messagerie et créera un bean usine qui continue d'écouter cette file d'attente.

Notre prochaine tâche est de créer une application client.

Création du client

Dans l'application cliente, nous configurons d'abord le point de terminaison JMS identique à celui utilisé dans l'application serveur -

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";

Nous créons une usine comme dans l'application POJO.

JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();

Nous définissons l'URI du point de terminaison et la classe d'implémentation comme suit -

factory.setTransportId (JMSSpecConstants.SOAP_JMS_SPECIFICATION_TRANSPORTID);
factory.setAddress (JMS_ENDPOINT_URI);
HelloWorld client = factory.create(HelloWorld.class);

Enfin, nous appelons la méthode de service et imprimons sa sortie résultante -

String reply = client.sayHi("TutorialsPoint");
System.out.println(reply);

Le code client complet est donné ci-dessous -

// 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);
   }
}

Apache CXF - Conclusion

CXF offre une approche unifiée pour combiner plusieurs protocoles et transports Web existants dans le monde actuel pour la création d'applications Web. Vous avez appris à démarrer avec une interface Java traditionnelle pour créer une application Web qui utilise CXF. Ensuite, vous avez appris à créer une application Web et son client en commençant par WSDL.

Le WSDL fournit une représentation XML de votre interface de service. Vous avez utilisé l'outil wsdl2java pour créer des interfaces Java à partir de WSDL et finalement écrit le serveur et le client à l'aide des interfaces créées. Le didacticiel vous a également brièvement présenté l'utilisation de CXF dans votre application de service Web RESTful. Enfin, nous avons également discuté de la manière dont CXF peut être utilisé avec JMS. Vous pouvez maintenant vous référer aux échantillons CXF pour une étude plus approfondie.

Note - L'ensemble du code source du projet peut être téléchargé à partir d'ici.