Compare commits
	
		
			53 Commits
		
	
	
		
			v2.0.3-jer
			...
			v2.0.6-cxf
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| d23a4249f7 | |||
| 8628d2dce8 | |||
| e4e393a855 | |||
| c7ce4e7c57 | |||
| 44e82b3a6b | |||
| ab920aedd7 | |||
| ab6b0d8da7 | |||
| 458cdad213 | |||
| 8b03d251e8 | |||
| a1acb3f9d9 | |||
| c25988fc8f | |||
| 00f25ff054 | |||
| 003afc208e | |||
| 85df5d3ea4 | |||
| 9a9a958913 | |||
| 37e8b63179 | |||
| 255a9c8e85 | |||
| c9833bb4b1 | |||
| 23b9c4ccb2 | |||
| 6bb1561825 | |||
| 0ce2369452 | |||
| ae82bc4ec7 | |||
| cc8d9e9d13 | |||
| 6cb7afd9f1 | |||
| 93a2eaa34e | |||
| 5df5388bd2 | |||
| 1a1cbb1246 | |||
| 5cda622082 | |||
| e1191abaaa | |||
| 67ea4b72b3 | |||
| a574154c20 | |||
| b1b19d8cb0 | |||
| 160766e0ef | |||
| e167ac43cc | |||
| a28fc13161 | |||
| 52c7e3cf6d | |||
| bd7413bb48 | |||
| 75c518801f | |||
| b3bc04467b | |||
| 88c8657a34 | |||
| 28b2478a08 | |||
| dc63abc272 | |||
| dd5d0f504a | |||
| 34d03a91e5 | |||
| 2584b8d668 | |||
| a756b05f6c | |||
| 853b4e66cf | |||
| f2032df0f3 | |||
| 8d3f4559bd | |||
| 3f476bba5c | |||
| 6292dde23b | |||
| 659f71c72c | |||
| 52f77dddbb | 
							
								
								
									
										70
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
|  | ||||
| # Common ReST Client Library | ||||
|  | ||||
| This project provides a library for Spring and POJO-based REST client instantiation. | ||||
| This project provides a library for Spring and POJO-based REST client instantiation.  It includes special classes with classifiers for two popular JAXRS-based client frameworks: Apache CXF and Jersey. | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| @@ -15,6 +15,7 @@ First, you will need to include the library in your project. | ||||
|         <dependency> | ||||
|             <groupId>com.inteligr8</groupId> | ||||
|             <artifactId>common-rest-client</artifactId> | ||||
|             <classifier>...</classifier> | ||||
|             <version>...</version> | ||||
|         </dependency> | ||||
|         ... | ||||
| @@ -23,4 +24,69 @@ First, you will need to include the library in your project. | ||||
| </project> | ||||
| ``` | ||||
|  | ||||
| See the `cxf` and `jersey` branches for examples and more documentation. | ||||
| Valid `classifier` values are `cxf` or `jersey`. | ||||
|  | ||||
| ### Spring Framework | ||||
|  | ||||
| #### Single Client | ||||
|  | ||||
| If you will only be declaring a single client in your Spring context, this is easy.  You will just need two things.  First, inject the single client into any of your Spring beans.  You may do inject it into more than one.  An example is below: | ||||
|  | ||||
| ```java | ||||
| @Component | ||||
| public class ... { | ||||
|     @Autowired | ||||
|     @Qualifier("client.cxf")  // may be optional | ||||
|     private Client client; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Next, you need to configure that client.  You can do that by providing a single implementation of the `ClientConfiguration` (or `ClientCxfConfiguration`) interface. | ||||
|  | ||||
| ```java | ||||
| @Configuration | ||||
| public class ... implements ClientCxfConfiguration { | ||||
|     ... | ||||
| } | ||||
| ``` | ||||
|  | ||||
| For Jersey implementations, just use `client.jersey` and `ClientJerseyConfiguration`.  If you want to provide one of each, then follow the instructions for multiple clients below. | ||||
|  | ||||
| #### Multiple Clients | ||||
|  | ||||
| If you will or may have multiple clients in your Spring context, there is an extra step.  You will still need to define a `ClientConfiguration` for each.  On top of that, you will need to create specialized implementations of each client.  That special implementation will reference the configuration directly.  An example is below. | ||||
|  | ||||
| ```java | ||||
| @Component("my.client") | ||||
| public class MyClient extends ClientCxfImpl { | ||||
|     @Autowired | ||||
|     public MyClient(MyClientConfiguration config) { | ||||
|         super(config); | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You can then inject your client(s) into your Spring beans.  Like the example below: | ||||
|  | ||||
| ```java | ||||
| @Component | ||||
| public class ... { | ||||
|     @Autowired | ||||
|     private MyClient client; | ||||
|  | ||||
|     @PostConstruct | ||||
|     public void init() { | ||||
|         MyJaxRsApi api = this.client.getApi(MyJaxRsApi.class); | ||||
|     } | ||||
| } | ||||
|  | ||||
| ### POJO | ||||
|  | ||||
| You do not have to use the Spring framework to use these classes.  You can instantiate them directly.  But you wil still need to create a `ClientConfiguration` as mentioned above. | ||||
|  | ||||
| ```java | ||||
| MyClientConfiguration config = new MyClientConfiguration(); | ||||
| ... | ||||
| ClientCxfImpl client = new ClientCxfImpl(config); | ||||
| MyJaxRsApi api = client.getApi(MyJaxRsApi.class); | ||||
| ``` | ||||
|   | ||||
							
								
								
									
										37
									
								
								pom.xml
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								pom.xml
									
									
									
									
									
								
							| @@ -6,11 +6,11 @@ | ||||
| 	 | ||||
| 	<groupId>com.inteligr8</groupId> | ||||
| 	<artifactId>common-rest-client</artifactId> | ||||
| 	<version>2.0-SNAPSHOT</version> | ||||
| 	<version>2.0.6-cxf</version> | ||||
| 	<packaging>jar</packaging> | ||||
|  | ||||
| 	<name>ReST API Client for Java</name> | ||||
| 	<description>A common library for building REST API clients</description> | ||||
| 	<description>A common library for building CXF REST API clients</description> | ||||
| 	<url>https://bitbucket.org/inteligr8/common-rest-client</url> | ||||
|  | ||||
| 	<licenses> | ||||
| @@ -46,6 +46,7 @@ | ||||
| 		<junit.version>5.7.2</junit.version> | ||||
| 		<spring.version>5.3.27</spring.version> | ||||
| 		<jackson.version>2.15.1</jackson.version> | ||||
| 		<cxf.version>3.5.6</cxf.version> | ||||
| 	</properties> | ||||
|  | ||||
| 	<dependencies> | ||||
| @@ -97,10 +98,42 @@ | ||||
| 			<version>4.5.14</version> | ||||
| 			<scope>test</scope> | ||||
| 		</dependency> | ||||
| 		 | ||||
| 		<!-- Apache CXF libraries --> | ||||
| 		<dependency> | ||||
| 			<groupId>org.apache.cxf</groupId> | ||||
| 			<artifactId>cxf-rt-rs-client</artifactId> | ||||
| 			<version>${cxf.version}</version> | ||||
| 		</dependency> | ||||
| 	</dependencies> | ||||
| 	 | ||||
| 	<build> | ||||
| 		<plugins> | ||||
| 			<plugin> | ||||
| 				<groupId>org.codehaus.mojo</groupId> | ||||
| 				<artifactId>build-helper-maven-plugin</artifactId> | ||||
| 				<version>3.4.0</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>add-jaxrs-src</id> | ||||
| 						<goals><goal>add-source</goal></goals> | ||||
| 						<configuration> | ||||
| 							<sources> | ||||
| 								<source>src/main/cxf</source> | ||||
| 							</sources> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>add-test-src</id> | ||||
| 						<goals><goal>add-test-source</goal></goals> | ||||
| 						<configuration> | ||||
| 							<sources> | ||||
| 								<source>src/test/cxf</source> | ||||
| 							</sources> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<artifactId>maven-surefire-plugin</artifactId> | ||||
| 				<version>3.1.0</version> | ||||
|   | ||||
							
								
								
									
										48
									
								
								src/main/cxf/com/inteligr8/rs/ClientCxfConfiguration.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								src/main/cxf/com/inteligr8/rs/ClientCxfConfiguration.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| /* | ||||
|  * This program is free software: you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU Lesser General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or (at your | ||||
|  * option) any later version. | ||||
|  *  | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  *  | ||||
|  * You should have received a copy of the GNU General Public License along | ||||
|  * with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.inteligr8.rs; | ||||
|  | ||||
| import org.apache.cxf.jaxrs.client.WebClient; | ||||
|  | ||||
| /** | ||||
|  * This interface defines additional configurations specific to the Apache CXF | ||||
|  * JAX-RS library and its nuances. | ||||
|  *  | ||||
|  * @author brian@inteligr8.com | ||||
|  */ | ||||
| public interface ClientCxfConfiguration extends ClientConfiguration { | ||||
| 	 | ||||
| 	/** | ||||
| 	 * Apache CXF uses a global bus configuration where interceptors could | ||||
| 	 * wreck havoc on your implementation.  This method allows you to | ||||
| 	 * explicitly by-pass the default bus. | ||||
| 	 *  | ||||
| 	 * See https://cxf.apache.org/docs/bus-configuration.html. | ||||
| 	 *  | ||||
| 	 * @return true to use the default bus; false otherwise. | ||||
| 	 */ | ||||
| 	default boolean isDefaultBusEnabled() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
|     /** | ||||
|      * A Jackson provider, logging filter, and authentication filter are already registered. | ||||
|      *  | ||||
|      * @param client A CXF client to configure. | ||||
|      */ | ||||
| 	default void configureClient(WebClient client) { | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										172
									
								
								src/main/cxf/com/inteligr8/rs/ClientCxfImpl.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								src/main/cxf/com/inteligr8/rs/ClientCxfImpl.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | ||||
| /* | ||||
|  * This program is free software: you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU Lesser General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or (at your | ||||
|  * option) any later version. | ||||
|  *  | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  *  | ||||
|  * You should have received a copy of the GNU General Public License along | ||||
|  * with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.inteligr8.rs; | ||||
|  | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
|  | ||||
| import javax.annotation.PostConstruct; | ||||
| import javax.ws.rs.ext.RuntimeDelegate; | ||||
|  | ||||
| import org.apache.cxf.BusFactory; | ||||
| import org.apache.cxf.jaxrs.client.JAXRSClientFactory; | ||||
| import org.apache.cxf.jaxrs.client.WebClient; | ||||
| import org.apache.cxf.jaxrs.impl.RuntimeDelegateImpl; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.stereotype.Component; | ||||
|  | ||||
| import com.fasterxml.jackson.databind.ObjectMapper; | ||||
| import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; | ||||
| import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; | ||||
|  | ||||
| /** | ||||
|  * A class that provides pre-configured JAX-RS Client & WebTarget & | ||||
|  * CXF WebClient objects. | ||||
|  *  | ||||
|  * @author brian@inteligr8.com | ||||
|  */ | ||||
| @Component("client.cxf") | ||||
| public class ClientCxfImpl extends Client { | ||||
| 	 | ||||
| 	private final Logger logger = LoggerFactory.getLogger(ClientCxfImpl.class); | ||||
|  | ||||
|     private final Object sync = new Object(); | ||||
| 	private ClientCxfConfiguration config; | ||||
|     private WebClient client; | ||||
| 	 | ||||
| 	/** | ||||
| 	 * This constructor is for Spring or POJO use. | ||||
| 	 * @param config The client configuration. | ||||
| 	 */ | ||||
| 	@Autowired | ||||
| 	public ClientCxfImpl(ClientCxfConfiguration config) { | ||||
| 		this.config = config; | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * This method registers the Apache CXF library as the default provider for | ||||
| 	 * the JAX-RS specification. | ||||
| 	 */ | ||||
| 	@PostConstruct | ||||
| 	public void register() { | ||||
| 		if (RuntimeDelegate.getInstance() == null) { | ||||
| 			this.logger.info("Setting JAX-RS runtime delegate to the CXF library"); | ||||
| 			RuntimeDelegate.setInstance(new RuntimeDelegateImpl()); | ||||
| 		} else if (RuntimeDelegate.getInstance() instanceof RuntimeDelegateImpl) { | ||||
| 			this.logger.info("JAX-RS runtime delegate already the CXF library"); | ||||
| 		} else { | ||||
| 			this.logger.warn("Setting JAX-RS runtime delegate to the CXF library; was: " + RuntimeDelegate.getInstance().getClass().getName()); | ||||
| 			RuntimeDelegate.setInstance(new RuntimeDelegateImpl()); | ||||
| 		} | ||||
| 		 | ||||
| 		if (this.logger.isInfoEnabled()) | ||||
| 			this.logger.info("API Base URL: " + this.getConfig().getBaseUrl()); | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @return A CXF client (not JAX-RS). | ||||
| 	 */ | ||||
| 	public WebClient getCxfClient() { | ||||
|         synchronized (this.sync) { | ||||
|             if (this.client == null) | ||||
|                 this.client = this.buildCxfClient(null); | ||||
|         } | ||||
|  | ||||
|         return this.client; | ||||
| 	} | ||||
|      | ||||
|     /** | ||||
|      * @param authFilter A dynamic authorization filter. | ||||
|      * @return A pre-configured CXF client (no URL) with the specified authorization. | ||||
|      */ | ||||
|     public WebClient getCxfClient(AuthorizationFilter authFilter) { | ||||
|         if (authFilter == null) { | ||||
|             return this.getCxfClient(); | ||||
|         } else { | ||||
|             return this.buildCxfClient(authFilter); | ||||
|         } | ||||
|     } | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @param authFilter A post-configuration authorization filter. | ||||
| 	 * @return A CXF client (not JAX-RS). | ||||
| 	 */ | ||||
| 	public WebClient buildCxfClient(AuthorizationFilter authFilter) { | ||||
|         ObjectMapper om = new ObjectMapper(); | ||||
|         om.registerModules(new JavaTimeModule()); | ||||
|         this.getConfig().configureJacksonMapper(om); | ||||
|  | ||||
|         JacksonJaxbJsonProvider jacksonProvider = new JacksonJaxbJsonProvider(om, JacksonJaxbJsonProvider.DEFAULT_ANNOTATIONS); | ||||
|         this.getConfig().configureJacksonProvider(jacksonProvider); | ||||
|          | ||||
| 		List<Object> providersAndFilters = new LinkedList<Object>(); | ||||
| 		providersAndFilters.add(jacksonProvider); | ||||
| 		providersAndFilters.add(new CxfLoggingFilter()); | ||||
| 		providersAndFilters.add(new CxfMultipartProvider()); | ||||
| 		 | ||||
| 		if (authFilter == null) | ||||
| 			authFilter = this.getConfig().createAuthorizationFilter(); | ||||
| 		if (authFilter != null) | ||||
| 			providersAndFilters.add(authFilter); | ||||
| 		 | ||||
| 		this.addProvidersAndFilters(providersAndFilters); | ||||
|  | ||||
| 		// we can't use JAXRSClientFactory with a JAXRS client (duh!) | ||||
| 		// so we need to create a CXF client | ||||
| 		WebClient client = WebClient.create(this.getConfig().getBaseUrl(), providersAndFilters); | ||||
|  | ||||
| 		if (!this.getConfig().isDefaultBusEnabled()) { | ||||
| 			// Some applications (like ACS) add interceptors to the default bus | ||||
| 			// those interceptors may treat all messages as SOAP messages (like ACS), resulting in ClassCastExceptions | ||||
| 			// we need to ignore the default bus | ||||
| 			org.apache.cxf.jaxrs.client.ClientConfiguration config = WebClient.getConfig(client); | ||||
| 			config.setBus(BusFactory.newInstance().createBus()); | ||||
| 		} | ||||
| 		 | ||||
| 		this.config.configureClient(client); | ||||
| 		 | ||||
| 		return client; | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @param providersAndFilters A list of JAX-RS and CXF providers. | ||||
| 	 */ | ||||
| 	public void addProvidersAndFilters(List<Object> providersAndFilters) { | ||||
| 		// for extension purposes | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @return The client configuration. | ||||
| 	 */ | ||||
| 	public ClientCxfConfiguration getConfig() { | ||||
| 		return this.config; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * This method retrieves a JAX-RS implementation of the specified API with | ||||
| 	 * the specified authorization. | ||||
| 	 *  | ||||
| 	 * @param authFilter A dynamic authorization filter. | ||||
| 	 * @param apiClass A JAX-RS annotation API class. | ||||
| 	 * @return An instance of the API class. | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public <T> T getApi(AuthorizationFilter authFilter, Class<T> apiClass) { | ||||
| 		return JAXRSClientFactory.fromClient(this.getCxfClient(authFilter), apiClass); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										52
									
								
								src/main/cxf/com/inteligr8/rs/CxfLoggingFilter.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/main/cxf/com/inteligr8/rs/CxfLoggingFilter.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| /* | ||||
|  * This program is free software: you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU Lesser General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or (at your | ||||
|  * option) any later version. | ||||
|  *  | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  *  | ||||
|  * You should have received a copy of the GNU General Public License along | ||||
|  * with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.inteligr8.rs; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
|  | ||||
| import javax.ws.rs.client.ClientRequestContext; | ||||
| import javax.ws.rs.core.MediaType; | ||||
|  | ||||
| import org.apache.cxf.jaxrs.ext.multipart.Attachment; | ||||
| import org.apache.cxf.jaxrs.ext.multipart.MultipartBody; | ||||
| import org.slf4j.Logger; | ||||
|  | ||||
| /** | ||||
|  * This is a CXF specific handling of the logging of multipart requests, which | ||||
|  * would otherwise be ignored by the base LoggingFilter.  It is meant to be | ||||
|  * used for debugging purposes.  When used, it will write to 'jaxrs.request' and | ||||
|  * 'jaxrs.response' loggers at the 'trace' level. | ||||
|  *  | ||||
|  * @author brian@inteligr8.com | ||||
|  */ | ||||
| public class CxfLoggingFilter extends LoggingFilter { | ||||
| 	 | ||||
| 	@Override | ||||
| 	protected void logUnhandledRequest(ClientRequestContext requestContext, Logger logger) throws IOException { | ||||
| 		if (MediaType.MULTIPART_FORM_DATA_TYPE.equals(requestContext.getMediaType())) { | ||||
| 			if (requestContext.getEntity() instanceof MultipartBody) { | ||||
| 				List<String> attIds = new LinkedList<>(); | ||||
| 				for (Attachment att : ((MultipartBody)requestContext.getEntity()).getAllAttachments()) | ||||
| 					attIds.add(att.getContentId()); | ||||
| 				logger.trace("request: {} {}: {}", requestContext.getMethod(), requestContext.getUri(), attIds); | ||||
| 			} else { | ||||
| 				logger.trace("request: {} {}: failed to output form", requestContext.getMethod(), requestContext.getUri()); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										49
									
								
								src/main/cxf/com/inteligr8/rs/CxfMultipartProvider.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								src/main/cxf/com/inteligr8/rs/CxfMultipartProvider.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| /* | ||||
|  * This program is free software: you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU Lesser General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or (at your | ||||
|  * option) any later version. | ||||
|  *  | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  *  | ||||
|  * You should have received a copy of the GNU General Public License along | ||||
|  * with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.inteligr8.rs; | ||||
|  | ||||
| import java.lang.annotation.Annotation; | ||||
| import java.lang.reflect.Type; | ||||
|  | ||||
| import javax.ws.rs.Consumes; | ||||
| import javax.ws.rs.Produces; | ||||
| import javax.ws.rs.core.MediaType; | ||||
| import javax.ws.rs.ext.Provider; | ||||
|  | ||||
| import org.apache.cxf.jaxrs.ext.multipart.MultipartBody; | ||||
| import org.apache.cxf.jaxrs.provider.MultipartProvider; | ||||
|  | ||||
| /** | ||||
|  * This implements a JAX-RS provider that adds support for the handling of CXF | ||||
|  * MultipartBody. | ||||
|  *  | ||||
|  * @author brian@inteligr8.com | ||||
|  */ | ||||
| @Consumes(MediaType.MULTIPART_FORM_DATA) | ||||
| @Produces(MediaType.MULTIPART_FORM_DATA) | ||||
| @Provider | ||||
| public class CxfMultipartProvider extends MultipartProvider { | ||||
| 	 | ||||
| 	@Override | ||||
| 	public boolean isReadable(Class<?> type, Type genericType, Annotation[] annotations, MediaType mediaType) { | ||||
| 		return MultipartBody.class.isAssignableFrom(type) || super.isReadable(type, genericType, annotations, mediaType); | ||||
| 	} | ||||
| 	 | ||||
| 	@Override | ||||
| 	public boolean isWriteable(Class<?> type, Type genericType, Annotation[] annotations, MediaType mediaType) { | ||||
| 		return MultipartBody.class.isAssignableFrom(type) || super.isWriteable(type, genericType, annotations, mediaType); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -22,7 +22,6 @@ import com.fasterxml.jackson.databind.ObjectMapper; | ||||
| import com.fasterxml.jackson.databind.SerializationFeature; | ||||
| import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; | ||||
| import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; | ||||
| import com.fasterxml.jackson.jaxrs.json.JacksonJsonProvider; | ||||
|  | ||||
| /** | ||||
|  * A class that provides pre-configured JAX-RS Client & WebTarget objects. | ||||
| @@ -70,8 +69,10 @@ public abstract class Client { | ||||
| 	public final javax.ws.rs.client.Client buildClient(AuthorizationFilter authFilter) { | ||||
| 	    ObjectMapper om = new ObjectMapper(); | ||||
| 	    om.registerModules(new JavaTimeModule()); | ||||
| 	    this.getConfig().configureJacksonMapper(om); | ||||
| 	     | ||||
| 		JacksonJsonProvider provider = new JacksonJaxbJsonProvider(om, JacksonJaxbJsonProvider.DEFAULT_ANNOTATIONS); | ||||
| 	    JacksonJaxbJsonProvider provider = new JacksonJaxbJsonProvider(om, JacksonJaxbJsonProvider.DEFAULT_ANNOTATIONS); | ||||
| 		this.getConfig().configureJacksonProvider(provider); | ||||
|          | ||||
| 		if (this.getConfig().isWrapRootValueEnabled()) | ||||
| 			provider.enable(SerializationFeature.WRAP_ROOT_VALUE); | ||||
| @@ -87,16 +88,17 @@ public abstract class Client { | ||||
| 		if (authFilter != null) | ||||
| 			clientBuilder.register(authFilter); | ||||
| 		this.buildClient(clientBuilder); | ||||
| 		this.getConfig().configureClient(clientBuilder); | ||||
| 		 | ||||
| 		return clientBuilder.build(); | ||||
| 	} | ||||
|      | ||||
| 	/** | ||||
| 	 * @param clientBuilder A client builder | ||||
| 	 */ | ||||
| 	public void buildClient(ClientBuilder clientBuilder) { | ||||
| 		// for extension purposes | ||||
| 	} | ||||
|     /** | ||||
|      * @param clientBuilder A client builder | ||||
|      */ | ||||
|     public void buildClient(ClientBuilder clientBuilder) { | ||||
|         // for extension purposes | ||||
|     } | ||||
|  | ||||
| 	/** | ||||
| 	 * @return A pre-configured JAX-RS target (client w/ base URL) with configured authorization. | ||||
|   | ||||
| @@ -16,6 +16,11 @@ package com.inteligr8.rs; | ||||
|  | ||||
| import java.net.URI; | ||||
|  | ||||
| import javax.ws.rs.client.ClientBuilder; | ||||
|  | ||||
| import com.fasterxml.jackson.databind.ObjectMapper; | ||||
| import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; | ||||
|  | ||||
| /** | ||||
|  * This interface defines the configurable parameters of the clients; primarily | ||||
|  * their default authentication and authorization. | ||||
| @@ -129,6 +134,18 @@ public interface ClientConfiguration { | ||||
| 		return false; | ||||
| 	} | ||||
|      | ||||
|     /** | ||||
|      * @param mapper A Jackson object mapper to configure. | ||||
|      */ | ||||
|     default void configureJacksonMapper(ObjectMapper mapper) { | ||||
|     } | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @param provider A Jackson JAX-RS provider to configure. | ||||
| 	 */ | ||||
| 	default void configureJacksonProvider(JacksonJaxbJsonProvider provider) { | ||||
| 	} | ||||
| 	 | ||||
| 	 | ||||
|  | ||||
| 	/** | ||||
| @@ -170,4 +187,12 @@ public interface ClientConfiguration { | ||||
| 		} | ||||
| 	} | ||||
|      | ||||
|     /** | ||||
|      * A Jackson provider, logging filter, and authentication filter are already registered. | ||||
|      *  | ||||
|      * @param clientBuilder A JAX-RS client builder to configure. | ||||
|      */ | ||||
|     default void configureClient(ClientBuilder clientBuilder) { | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
							
								
								
									
										83
									
								
								src/main/java/com/inteligr8/rs/ClientImpl.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								src/main/java/com/inteligr8/rs/ClientImpl.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| /* | ||||
|  * This program is free software: you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU Lesser General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or (at your | ||||
|  * option) any later version. | ||||
|  *  | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  *  | ||||
|  * You should have received a copy of the GNU General Public License along | ||||
|  * with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.inteligr8.rs; | ||||
|  | ||||
| import javax.annotation.PostConstruct; | ||||
| import javax.ws.rs.client.ClientBuilder; | ||||
|  | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.stereotype.Component; | ||||
|  | ||||
| /** | ||||
|  * A class that provides pre-configured JAX-RS Client & WebTarget objects | ||||
|  * for Jersey. | ||||
|  *  | ||||
|  * @author brian@inteligr8.com | ||||
|  */ | ||||
| @Component("client.jaxrs") | ||||
| public class ClientImpl extends Client { | ||||
| 	 | ||||
| 	private final Logger logger = LoggerFactory.getLogger(ClientImpl.class); | ||||
| 	 | ||||
| 	private ClientConfiguration config; | ||||
| 	 | ||||
| 	/** | ||||
| 	 * This constructor is for Spring or POJO use. | ||||
| 	 * @param config The client configuration. | ||||
| 	 */ | ||||
| 	@Autowired | ||||
| 	public ClientImpl(ClientConfiguration config) { | ||||
| 		this.config = config; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * This method registers the Jersey library as the default provider for the | ||||
| 	 * JAX-RS specification. | ||||
| 	 */ | ||||
| 	@PostConstruct | ||||
| 	public void register() { | ||||
| 		this.logger.info("API Base URL: {}", this.getConfig().getBaseUrl()); | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @param clientBuilder A client builder. | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public void buildClient(ClientBuilder clientBuilder) { | ||||
| 	} | ||||
| 	 | ||||
| 	/** | ||||
| 	 * @return The client configuration. | ||||
| 	 */ | ||||
| 	public ClientConfiguration getConfig() { | ||||
| 		return this.config; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * This method retrieves a JAX-RS implementation of the specified API with | ||||
| 	 * the specified authorization. | ||||
| 	 *  | ||||
| 	 * @param authFilter A dynamic authorization filter. | ||||
| 	 * @param apiClass A JAX-RS annotation API class. | ||||
| 	 * @return An instance of the API class. | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public <T> T getApi(AuthorizationFilter authFilter, Class<T> apiClass) { | ||||
| 		throw new UnsupportedOperationException(); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -58,7 +58,7 @@ public class LoggingFilter implements ClientRequestFilter, ClientResponseFilter | ||||
| 				logger.trace("request: {} {}: {}", requestContext.getMethod(), requestContext.getUri(), | ||||
| 						((Form)requestContext.getEntity()).asMap()); | ||||
| 			} else { | ||||
| 				this.loggerRequest.trace("request: {} {}: failed to output form", requestContext.getMethod(), requestContext.getUri()); | ||||
| 				logger.trace("request: {} {}: failed to output form", requestContext.getMethod(), requestContext.getUri()); | ||||
| 			} | ||||
| 		} else { | ||||
| 			this.logUnhandledRequest(requestContext, logger); | ||||
|   | ||||
| @@ -74,11 +74,15 @@ public class OAuthAuthorizationCodeAuthorizationFilter extends OAuthAuthorizatio | ||||
| 	 | ||||
| 	@Override | ||||
| 	protected Form createForm() { | ||||
| 		Form form = new Form().param("grant_type", "authorization_code") | ||||
| 				.param("code", this.code); | ||||
| 		Form form = new Form().param("grant_type", "authorization_code"); | ||||
| 		if (this.redirectUri != null) | ||||
| 			form.param("redirect_uri", this.redirectUri.toString()); | ||||
| 		return form; | ||||
| 	} | ||||
| 	 | ||||
| 	@Override | ||||
| 	protected void extendFormSensitive(Form form) { | ||||
| 	    form.param("code", this.code); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -17,12 +17,18 @@ package com.inteligr8.rs; | ||||
| import java.util.Map; | ||||
|  | ||||
| import javax.ws.rs.WebApplicationException; | ||||
| import javax.ws.rs.client.Client; | ||||
| import javax.ws.rs.client.ClientBuilder; | ||||
| import javax.ws.rs.client.ClientRequestContext; | ||||
| import javax.ws.rs.client.Entity; | ||||
| import javax.ws.rs.client.WebTarget; | ||||
| import javax.ws.rs.core.Form; | ||||
| import javax.ws.rs.core.HttpHeaders; | ||||
| import javax.ws.rs.core.Response; | ||||
| import javax.ws.rs.core.Response.Status.Family; | ||||
|  | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; | ||||
|  | ||||
| @@ -33,6 +39,8 @@ import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; | ||||
|  */ | ||||
| public abstract class OAuthAuthorizationFilter implements AuthorizationFilter { | ||||
| 	 | ||||
|     private final Logger logger = LoggerFactory.getLogger(this.getClass()); | ||||
|      | ||||
| 	private final String tokenUrl; | ||||
| 	private final String clientId; | ||||
| 	private final String clientSecret; | ||||
| @@ -111,29 +119,49 @@ public abstract class OAuthAuthorizationFilter implements AuthorizationFilter { | ||||
| 		} | ||||
|  | ||||
| 		form.param("client_id", this.clientId); | ||||
| 		if (this.clientSecret != null) | ||||
| 			form.param("client_secret", this.clientSecret); | ||||
| 		if (this.scope != null) | ||||
| 			form.param("scope", this.scope); | ||||
| 		this.extendRefreshTokenForm(form); | ||||
| 		 | ||||
| 		this.logger.trace("Sending OAuth request: {}", form); | ||||
| 		 | ||||
| 		if (this.refreshToken != null) { | ||||
| 		    this.extendRefreshFormSensitive(form); | ||||
| 		} else { | ||||
|             this.extendFormSensitive(form); | ||||
| 		} | ||||
|  | ||||
|         if (this.clientSecret != null) | ||||
|             form.param("client_secret", this.clientSecret); | ||||
| 		 | ||||
| 		Entity<Form> entity = Entity.form(form); | ||||
| 		 | ||||
| 		WebTarget target = ClientBuilder.newBuilder() | ||||
| 				.register(new JacksonJaxbJsonProvider()) | ||||
| 				.build() | ||||
| 				.target(this.tokenUrl); | ||||
| 		Client client = ClientBuilder.newBuilder() | ||||
|                 .register(new JacksonJaxbJsonProvider()) | ||||
|                 .build(); | ||||
| 		WebTarget target = client.target(this.tokenUrl); | ||||
|  | ||||
| 		@SuppressWarnings("unchecked") | ||||
| 		Map<String, Object> response = target.request().post(entity, Map.class); | ||||
| 		Response response = target.request().post(entity); | ||||
| 		 | ||||
| 		if (response.containsKey("error")) | ||||
| 			throw new WebApplicationException((String)response.get("error"), 400); | ||||
|         this.logger.debug("Received OAuth response: {}", response.getStatus()); | ||||
|  | ||||
| 		this.accessToken = (String)response.get("access_token"); | ||||
| 		this.expiration = System.currentTimeMillis() + ((Number)response.get("expires_in")).longValue() * 1000L; | ||||
| 		this.refreshToken = (String)response.get("refresh_token"); | ||||
| 		this.extendRefreshTokenResponse(response); | ||||
|         @SuppressWarnings("unchecked") | ||||
|         Map<String, Object> responseMap = response.readEntity(Map.class); | ||||
|  | ||||
|         this.logger.trace("Received OAuth response: {}", responseMap); | ||||
| 		 | ||||
| 		if (response.getStatusInfo().getFamily() != Family.SUCCESSFUL) { | ||||
|             String code = (String) responseMap.get("error"); | ||||
|             if (code != null) { | ||||
|                 String description = (String) responseMap.get("error_description"); | ||||
|     			throw new WebApplicationException(code + ": " + description, response.getStatus()); | ||||
|             } else { | ||||
|                 throw new WebApplicationException(response); | ||||
|             } | ||||
| 		} | ||||
| 		 | ||||
| 		this.accessToken = (String)responseMap.get("access_token"); | ||||
| 		this.expiration = System.currentTimeMillis() + ((Number)responseMap.get("expires_in")).longValue() * 1000L; | ||||
| 		this.refreshToken = (String)responseMap.get("refresh_token"); | ||||
| 	} | ||||
| 	 | ||||
| 	protected Form createRefreshForm() { | ||||
| @@ -143,10 +171,9 @@ public abstract class OAuthAuthorizationFilter implements AuthorizationFilter { | ||||
| 	 | ||||
| 	protected abstract Form createForm(); | ||||
| 	 | ||||
| 	protected void extendRefreshTokenForm(Form form) { | ||||
| 	protected void extendRefreshFormSensitive(Form form) { | ||||
| 	} | ||||
|  | ||||
| 	protected void extendRefreshTokenResponse(Map<String, Object> response) { | ||||
| 	} | ||||
|     protected abstract void extendFormSensitive(Form form); | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -38,4 +38,8 @@ public class OAuthClientCredentialAuthorizationFilter extends OAuthAuthorization | ||||
| 		return new Form().param("grant_type", "client_credentials"); | ||||
| 	} | ||||
| 	 | ||||
| 	@Override | ||||
| 	protected void extendFormSensitive(Form form) { | ||||
| 	} | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -53,8 +53,12 @@ public class OAuthPasswordGrantAuthorizationFilter extends OAuthAuthorizationFil | ||||
| 	@Override | ||||
| 	protected Form createForm() { | ||||
| 		return new Form().param("grant_type", "password") | ||||
| 				.param("username", this.username) | ||||
| 				.param("password", this.password); | ||||
| 				.param("username", this.username); | ||||
| 	} | ||||
| 	 | ||||
| 	@Override | ||||
| 	protected void extendFormSensitive(Form form) { | ||||
| 	    form.param("password", this.password); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user