WSDL First ile Apache CXF
Geliştirdiğiniz CXF-POJO uygulaması, istemci ve sunucu arasında çok sıkı bir bağlantıya neden olur. Hizmet arayüzüne doğrudan erişim sağlamak, ciddi güvenlik tehditleri de oluşturabilir. Bu nedenle, WSDL (Web Servisleri Tanımlama Dili) kullanılarak genellikle istemci ve sunucu arasında ayrıştırma istenir.
Web servis arayüzünü XML tabanlı bir WSDL belgesine yazıyoruz. Bu WSDL'yi daha sonra istemci ve sunucu uygulamalarımız tarafından uygulanan ve kullanılan Apache CXF arayüzlerine eşlemek için bir araç kullanacağız. Ayrıştırmayı sağlamak için WSDL ile başlamak tercih edilen bir yoldur. Bunun için önce yeni bir dil öğrenmelisiniz - WSDL. WSDL yazmak dikkatli bir yaklaşıma ihtiyaç duyar ve üzerinde çalışmaya başlamadan önce bu konuda biraz anlayış kazanmanız daha iyi olur.
Bu derste, bir WSDL belgesinde bir web hizmeti arayüzü tanımlayarak başlayacağız. WSDL ile başlayarak hem sunucu hem de istemci uygulamaları oluşturmak için CXF'yi nasıl kullanacağımızı öğreneceğiz. Uygulamayı CXF kullanımına odaklanmak için basit tutacağız. Sunucu uygulaması oluşturulduktan sonra, yerleşik bir CXF sınıfı kullanarak onu istenen bir URL'de yayınlayacağız.
Öncelikle kullanacağımız WSDL'yi tanımlayalım.
HelloWorld için WSDL
Uygulayacağımız web hizmetinin adında tek bir web yöntemi olacak greetings kabul eden stringparametresi kullanıcı adını tutar ve kullanıcı adına bir selamlama mesajı ekledikten sonra arayana bir dizi mesajı verir. Tam wsdl aşağıda gösterilmiştir -
//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>
Sözdizimsel olarak doğru bir wsdl yazmak, geliştiriciler için her zaman bir zorluk olmuştur; wsdl oluşturmak için birçok araç ve çevrimiçi düzenleyici mevcuttur. Bu düzenleyiciler, bir mesajda iletmek istediğiniz parametrelerle birlikte uygulamak istediğiniz mesajların adlarını ve istemci uygulamanızın almasını istediğiniz geri dönüş mesajı türünü sorarlar. Wsdl sözdizimini biliyorsanız, tüm belgeyi elinize alabilir veya kendinizinkini oluşturmak için düzenleyicilerden birini kullanabilirsiniz.
Yukarıdaki wsdl'de, adında tek bir mesaj tanımladık greetings. Mesaj, aranan servise gönderilir.HelloWorldService koşuyor http://localhost:9090/HelloServerPort.
Bununla birlikte şimdi sunucu geliştirmeye geçeceğiz. Sunucuyu geliştirmeden önce web servisimize Apache CXF arayüzü oluşturmamız gerekiyor. Bu verilen wsdl'den yapılacak. Bunu yapmak için adlı bir araç kullanırsınızwsdl2java.
Wsdl2java Eklentisi
Projeyi oluşturmak için maven kullanacağımızdan, aşağıdaki eklentiyi pom.xml dosya.
<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>
Unutmayın ki konumu wsdl olarak dosyala src/main/resources/Hello.wsdl. Projeniz için uygun bir dizin yapısı oluşturduğunuzdan ve daha önce gösterilenleri eklediğinizden emin olmalısınız.hello.wsdl dosyayı belirtilen klasöre.
wsdl2javaeklenti bu wsdl'yi derleyecek ve önceden tanımlanmış bir klasörde Apache CXF sınıfları oluşturacaktır. Tam proje yapısı, hazır referansınız için burada gösterilmektedir.
Şimdi, kullanarak bir sunucu oluşturmaya hazırsınız. wsdl2javaoluşturulan sınıflar. Wsdl2java'nın oluşturduğu sınıflar aşağıdaki şekilde gösterilmektedir -
Oluşturulan Servis Arayüzü
Oluşturulan sınıflar listesinde, bunlardan birinin Apache CXF arayüzü olduğunu fark etmiş olmalısınız - bu HelloWorldPortType.java. Bu dosyayı kod düzenleyicinizde inceleyin. Dosya içeriği, hazır referansınız için burada gösterilmektedir -
//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
);
}
Arayüzün adı verilen bir yöntem içerdiğine dikkat edin greetings. Bu, wsdl'mizdeki bir mesaj tipiydi. wsdl2javaaracı, bu yöntemi oluşturulan arayüze ekledi. Şimdi, wsdl'nize yazdığınız mesaj ne olursa olsun, arayüzde buna karşılık gelen bir yöntemin üretileceğini anlayabilirsiniz.
Şimdi, göreviniz wsdl'nizde tanımladığınız çeşitli mesajlara karşılık gelen tüm bu yöntemleri uygulamak olacaktır. Önceki Apache CXF-First örneğinde, web hizmetimiz için bir Apache CXF arayüzü ile başladığımızı unutmayın. Bu durumda, Apache CXF arayüzü wsdl'den oluşturulur.
Servis Arayüzünü Uygulama
Servis arayüzünün uygulanması önemsizdir. Tam uygulama aşağıdaki listede gösterilmektedir -
//HelloWorldImpl.java
package com.tutorialspoint.helloworld;
public class HelloWorldImpl implements HelloWorldPortType {
@Override
public String greetings(String name) {
return ("hi " + name);
}
}
Kod, adı verilen tek arayüz yöntemini uygular greetings. Yöntem bir parametre alırstring yazın, başına bir "merhaba" mesajı ekler ve ortaya çıkan dizeyi arayana geri döndürür.
Ardından sunucu uygulamasını yazacağız.
Geliştirme Sunucusu
Sunucu uygulaması geliştirmek bir kez daha önemsiz. Burada sağlanan CXF'yi kullanacağızEndpointhizmetimizi yayınlamak için sınıf. Bu, aşağıdaki iki kod satırında yapılır -
HelloWorldPortType implementor = new HelloWorldImpl();
Endpoint.publish("http://localhost:9090/HelloServerPort",
implementor,
new LoggingFeature());
İlk olarak, hizmet uygulama sınıfımızın bir nesnesini oluşturuyoruz - HelloWorldImpl. Ardından, bu referansı ikinci parametre olarakpublishyöntem. İlk parametre, hizmetin yayınlandığı adrestir - istemciler hizmete erişmek için bu URL'yi kullanır. Sunucu uygulamasının tüm kaynağı burada verilmiştir -
//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);
}
}
Bu sunucu sınıfını oluşturmak için, klasörünüze bir yapı profili eklemeniz gerekir. pom.xml. Bu aşağıda gösterilmiştir -
<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>
Tam nitelikli adın Serversınıf konfigürasyonda belirtilmiştir. Ayrıca, bağımlılık etiketi, sunucu uygulamamızı dağıtmak için gömülü jetty web sunucusunu kullanacağımızı belirtir.
Sunucuyu Dağıtma
Son olarak, sunucu uygulamasını dağıtmak için, uygulamanızı bir web uygulaması olarak kurmak için pom.xml'de bir değişiklik daha yapmanız gerekecektir. Eklemeniz gereken kodpom.xml aşağıda verilmiştir -
<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>
Uygulamayı dağıtmadan önce, projenize iki dosya daha eklemeniz gerekir. Bunlar aşağıdaki ekran görüntüsünde gösterilmektedir -
Bu dosyalar, eşlemeyi tanımlayan CXF standart dosyalarıdır. CXFServlet. İçindeki kodweb.xml dosya hızlı başvurunuz için burada gösterilmektedir -
//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>
İçinde cxf-servlet.xmlhizmetinizin uç noktası için özellikleri bildirirsiniz. Bu, aşağıdaki kod pasajında gösterilmektedir -
<beans ...>
<jaxws:endpoint xmlns:helloworld = "http://tutorialspoint.com/"
id="helloHTTP"
address = "http://localhost:9090/HelloServerPort"
serviceName = "helloworld:HelloServiceService"
endpointName = "helloworld:HelloServicePort">
</jaxws:endpoint>
</beans>
Burada hizmet uç noktanızın kimliğini, hizmetin kullanılabileceği adresi, hizmet adını ve uç nokta adını tanımlarız. Artık hizmetinizin bir CXF sunucu uygulaması tarafından nasıl yönlendirildiğini ve işlendiğini anlıyorsunuz.
Son pom.xml
pom.xmlbirkaç tane daha bağımlılık içerir. Tüm bağımlılıkları açıklamak yerine, aşağıya pom.xml'nin son sürümünü ekledik -
<?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>
Ayrıca, ilerideki bölümlerde yakında öğreneceğimiz müşteri oluşturmak için bir profil içerdiğini unutmayın.
HelloWorld Hizmetini Çalıştırmak
Artık web uygulamasını çalıştırmaya hazırsınız. Komut penceresinde, aşağıdaki komutu kullanarak derleme komut dosyasını çalıştırın.
mvn clean install
Bu, wsdl'nizden uygun Apache CXF sınıflarını oluşturacak, Apache CXF sınıflarınızı derleyecek, sunucuyu gömülü jetty sunucusuna yerleştirecek ve uygulamanızı çalıştıracaktır.
Konsolda aşağıdaki mesajı göreceksiniz -
INFO: Setting the server's publish address to be
http://localhost:9090/HelloServerPort
Server ready...
Daha önce olduğu gibi, tarayıcınızda sunucu URL'sini açarak sunucuyu test edebilirsiniz.
Herhangi bir işlem belirtmediğimiz için, uygulamamız tarafından tarayıcıya sadece bir hata mesajı dönmektedir. Şimdi eklemeyi deneyin?wsdl URL'nize ve aşağıdaki çıktıyı göreceksiniz -
Yani sunucu uygulamamız beklendiği gibi çalışıyor. SOAP İstemcisini aşağıdaki gibi kullanabilirsiniz:Postman hizmetinizi daha ayrıntılı test etmek için daha önce açıklanmıştır.
Bu öğreticinin bir sonraki kısmı, hizmetimizi kullanan bir müşteri yazmaktır.
Müşteri Geliştirme
İstemciyi bir CXF uygulamasında yazmak, bir sunucu yazmak kadar önemlidir. Esasen yalnızca üç satırdan oluşan istemci için eksiksiz kod, geri kalan satırlar yalnızca hizmet bilgilerini kullanıcıya yazdırır.
//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")));
}
}
Burada, hizmetimizin bir örneğini oluşturuyoruz HelloWorldService, arayarak limanını alın getHelloWorldPort yöntemi ve sonra bizim greetingsona mesaj. İstemciyi çalıştırın ve aşağıdaki çıktıyı göreceksiniz -
service: {http://helloworld.tutorialspoint.com/}HelloWorldService
wsdl location: file:/Users/drsarang/Desktop/tutorialpoint/cxf-
wsdl/src/main/resources/Hello.wsdl
hi drsarang
Şimdiye kadar CXF'yi Apache CXF-First ve WSDL-First mimarileriyle nasıl kullanacağınızı öğrendiniz. Apache CXF-First yaklaşımında, bir POJO kullandınız.ServerFactoryBeanbir sunucu oluşturmak için CXF kitaplıklarından sınıf. Kullandığınız bir müşteri oluşturmak içinClientProxyFactoryBeanCXF kütüphanesinden sınıf. WSDL-First yaklaşımında, kullandınızEndpointhizmeti istenen URL ve belirli bir uygulayıcıda yayınlamak için sınıf. Artık bu teknikleri farklı protokolleri ve taşımaları entegre etmek için genişletebilirsiniz.