Added incomplete REST client
This commit is contained in:
parent
ccecc067b2
commit
351d5478c7
5 changed files with 223 additions and 48 deletions
|
|
@ -39,53 +39,56 @@ import java.util.HashMap;
|
||||||
*
|
*
|
||||||
* @author Ziver
|
* @author Ziver
|
||||||
*/
|
*/
|
||||||
public class HttpClient implements AutoCloseable{
|
public class HttpClient implements AutoCloseable {
|
||||||
public enum HttpRequestType{
|
public enum HttpRequestType {
|
||||||
GET, POST
|
GET, POST
|
||||||
}
|
}
|
||||||
|
|
||||||
// Request variables
|
// Request variables
|
||||||
private HttpURL url;
|
private HttpURL url;
|
||||||
private HttpRequestType type;
|
private HttpRequestType type;
|
||||||
private HashMap<String,String> headers;
|
private HashMap<String, String> headers;
|
||||||
private HashMap<String,String> cookies;
|
private HashMap<String, String> cookies;
|
||||||
private String data;
|
private String data;
|
||||||
|
|
||||||
// Response variables
|
// Response variables
|
||||||
private HttpHeaderParser rspHeader;
|
private HttpHeaderParser rspHeader;
|
||||||
private InputStream rspStream;
|
private InputStream rspStream;
|
||||||
|
|
||||||
|
|
||||||
|
public HttpClient(HttpRequestType type) {
|
||||||
public HttpClient(HttpRequestType type){
|
|
||||||
this.type = type;
|
this.type = type;
|
||||||
headers = new HashMap<>();
|
headers = new HashMap<>();
|
||||||
cookies = new HashMap<>();
|
cookies = new HashMap<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setURL( URL url){
|
public void setURL(URL url) {
|
||||||
this.url = new HttpURL( url );
|
setURL(new HttpURL(url));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setURL(HttpURL url) {
|
||||||
|
this.url = url;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a parameter to the request
|
* Adds a parameter to the request
|
||||||
*/
|
*/
|
||||||
public void setParameter( String key, String value ){
|
public void setParameter(String key, String value) {
|
||||||
url.setParameter(key, value);
|
url.setParameter(key, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a cookie to the request
|
* Adds a cookie to the request
|
||||||
*/
|
*/
|
||||||
public void setCookie( String key, String value ){
|
public void setCookie(String key, String value) {
|
||||||
cookies.put(key, value);
|
cookies.put(key, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a header value to the request
|
* Adds a header value to the request
|
||||||
*/
|
*/
|
||||||
public void setHeader( String key, String value ){
|
public void setHeader(String key, String value) {
|
||||||
headers.put(key, value);
|
headers.put(key, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -93,7 +96,7 @@ public class HttpClient implements AutoCloseable{
|
||||||
* Sets the content data that will be included in the request.
|
* Sets the content data that will be included in the request.
|
||||||
* NOTE: this will override the POST data parameter inclusion.
|
* NOTE: this will override the POST data parameter inclusion.
|
||||||
*/
|
*/
|
||||||
public void setData( String data ){
|
public void setData(String data) {
|
||||||
this.data = data;
|
this.data = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -101,32 +104,31 @@ public class HttpClient implements AutoCloseable{
|
||||||
* Will send a HTTP request to the target host.
|
* Will send a HTTP request to the target host.
|
||||||
* NOTE: any previous request connections will be closed
|
* NOTE: any previous request connections will be closed
|
||||||
*/
|
*/
|
||||||
public HttpHeaderParser send() throws IOException{
|
public HttpHeaderParser send() throws IOException {
|
||||||
Socket conn = new Socket( url.getHost(), url.getPort());
|
Socket conn = new Socket(url.getHost(), url.getPort());
|
||||||
|
|
||||||
// Request
|
// Request
|
||||||
HttpPrintStream request = new HttpPrintStream( conn.getOutputStream(), HttpMessageType.REQUEST );
|
HttpPrintStream request = new HttpPrintStream(conn.getOutputStream(), HttpMessageType.REQUEST);
|
||||||
request.setRequestType( type.toString() );
|
request.setRequestType(type.toString());
|
||||||
request.setRequestURL( url.getHttpURL() );
|
request.setRequestURL(url.getHttpURL());
|
||||||
request.setHeaders( headers );
|
request.setHeaders(headers);
|
||||||
request.setCookies( cookies );
|
request.setCookies(cookies);
|
||||||
|
|
||||||
if( type == HttpRequestType.POST ){
|
if (type == HttpRequestType.POST) {
|
||||||
String postData;
|
String postData;
|
||||||
if(data != null)
|
if (data != null)
|
||||||
postData = data;
|
postData = data;
|
||||||
else
|
else
|
||||||
postData = url.getParameterString();
|
postData = url.getParameterString();
|
||||||
request.setHeader("Content-Length", ""+postData.length());
|
request.setHeader("Content-Length", "" + postData.length());
|
||||||
request.println();
|
request.println();
|
||||||
request.print( postData );
|
request.print(postData);
|
||||||
}
|
} else
|
||||||
else
|
|
||||||
request.println();
|
request.println();
|
||||||
request.close();
|
request.close();
|
||||||
|
|
||||||
// Response
|
// Response
|
||||||
if(rspHeader != null || rspStream != null) // Close previous request
|
if (rspHeader != null || rspStream != null) // Close previous request
|
||||||
this.close();
|
this.close();
|
||||||
rspStream = new BufferedInputStream(conn.getInputStream());
|
rspStream = new BufferedInputStream(conn.getInputStream());
|
||||||
rspHeader = new HttpHeaderParser(rspStream);
|
rspHeader = new HttpHeaderParser(rspStream);
|
||||||
|
|
@ -134,16 +136,17 @@ public class HttpClient implements AutoCloseable{
|
||||||
return rspHeader;
|
return rspHeader;
|
||||||
}
|
}
|
||||||
|
|
||||||
public HttpHeaderParser getResponseHeader(){
|
public HttpHeaderParser getResponseHeader() {
|
||||||
return rspHeader;
|
return rspHeader;
|
||||||
}
|
}
|
||||||
public InputStream getResponseInputStream(){
|
|
||||||
|
public InputStream getResponseInputStream() {
|
||||||
return rspStream;
|
return rspStream;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
if(rspStream != null)
|
if (rspStream != null)
|
||||||
rspStream.close();
|
rspStream.close();
|
||||||
rspStream = null;
|
rspStream = null;
|
||||||
rspHeader = null;
|
rspHeader = null;
|
||||||
|
|
|
||||||
59
src/zutil/net/ws/rest/RESTClientFactory.java
Normal file
59
src/zutil/net/ws/rest/RESTClientFactory.java
Normal file
|
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2018 Ziver Koc
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package zutil.net.ws.rest;
|
||||||
|
|
||||||
|
import zutil.log.LogUtil;
|
||||||
|
import zutil.net.ws.WSClientFactory;
|
||||||
|
import zutil.net.ws.WSInterface;
|
||||||
|
import zutil.net.ws.WebServiceDef;
|
||||||
|
import zutil.net.ws.soap.SOAPClientInvocationHandler;
|
||||||
|
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an factory that generates clients for web services
|
||||||
|
*
|
||||||
|
* @author Ziver
|
||||||
|
*/
|
||||||
|
public class RESTClientFactory {
|
||||||
|
private static Logger logger = LogUtil.getLogger();
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates a Client Object for the web service.
|
||||||
|
*
|
||||||
|
* @param <T> is the class of the web service definition
|
||||||
|
* @param url is the target service serviceUrl
|
||||||
|
* @param intf is the class of the web service definition
|
||||||
|
* @return a client Object
|
||||||
|
*/
|
||||||
|
public static <T extends WSInterface> T createClient(URL url, Class<T> intf){
|
||||||
|
T obj = WSClientFactory.createClient( intf,
|
||||||
|
new RESTClientInvocationHandler(url, new WebServiceDef(intf)));
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
112
src/zutil/net/ws/rest/RESTClientInvocationHandler.java
Normal file
112
src/zutil/net/ws/rest/RESTClientInvocationHandler.java
Normal file
|
|
@ -0,0 +1,112 @@
|
||||||
|
/*
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2018 Ziver Koc
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package zutil.net.ws.rest;
|
||||||
|
|
||||||
|
import zutil.io.IOUtil;
|
||||||
|
import zutil.log.LogUtil;
|
||||||
|
import zutil.net.http.HttpClient;
|
||||||
|
import zutil.net.http.HttpHeaderParser;
|
||||||
|
import zutil.net.http.HttpURL;
|
||||||
|
import zutil.net.ws.WSInterface;
|
||||||
|
import zutil.net.ws.WSMethodDef;
|
||||||
|
import zutil.net.ws.WSParameterDef;
|
||||||
|
import zutil.net.ws.WebServiceDef;
|
||||||
|
import zutil.net.ws.soap.SOAPHttpPage;
|
||||||
|
|
||||||
|
import java.lang.reflect.InvocationHandler;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.logging.Level;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an abstract client that will do generic requests to a
|
||||||
|
* REST Web service using JSON response.
|
||||||
|
*/
|
||||||
|
public class RESTClientInvocationHandler implements InvocationHandler {
|
||||||
|
private static Logger logger = LogUtil.getLogger();
|
||||||
|
|
||||||
|
private WebServiceDef wsDef;
|
||||||
|
/**
|
||||||
|
* Web address of the web service
|
||||||
|
*/
|
||||||
|
protected URL serviceUrl;
|
||||||
|
|
||||||
|
public RESTClientInvocationHandler(URL url, WebServiceDef webServiceDef) {
|
||||||
|
this.serviceUrl = url;
|
||||||
|
this.wsDef = webServiceDef;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes a request to the target web service
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
|
||||||
|
// Generate XML
|
||||||
|
HttpURL url = generateRESTRequest(method.getName(), args);
|
||||||
|
|
||||||
|
// Send request
|
||||||
|
HttpClient request = new HttpClient(HttpClient.HttpRequestType.POST);
|
||||||
|
request.setURL(url);
|
||||||
|
HttpHeaderParser response = request.send();
|
||||||
|
String rspJson = IOUtil.readContentAsString(request.getResponseInputStream());
|
||||||
|
request.close();
|
||||||
|
|
||||||
|
// DEBUG
|
||||||
|
if (logger.isLoggable(Level.FINEST)) {
|
||||||
|
System.out.println("********** Request");
|
||||||
|
System.out.println(url);
|
||||||
|
System.out.println("********** Response");
|
||||||
|
System.out.println(rspJson);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parseRESTResponse(rspJson);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private HttpURL generateRESTRequest(String targetMethod, Object[] args) {
|
||||||
|
logger.fine("Sending request for " + targetMethod);
|
||||||
|
HttpURL url = new HttpURL(serviceUrl);
|
||||||
|
|
||||||
|
WSMethodDef methodDef = wsDef.getMethod(targetMethod);
|
||||||
|
url.setPath(serviceUrl.getPath()
|
||||||
|
+ (serviceUrl.getPath().endsWith("/") ? "" : "/")
|
||||||
|
+ methodDef.getName());
|
||||||
|
|
||||||
|
for (int i = 0; i < methodDef.getOutputCount(); i++) {
|
||||||
|
WSParameterDef param = methodDef.getOutput(i);
|
||||||
|
url.setParameter(param.getName(), args[i].toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Object parseRESTResponse(String json) {
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -45,7 +45,7 @@ public class SOAPClientFactory {
|
||||||
* Generates a Client Object for the web service.
|
* Generates a Client Object for the web service.
|
||||||
*
|
*
|
||||||
* @param <T> is the class of the web service definition
|
* @param <T> is the class of the web service definition
|
||||||
* @param url is the target service url
|
* @param url is the target service serviceUrl
|
||||||
* @param intf is the class of the web service definition
|
* @param intf is the class of the web service definition
|
||||||
* @return a client Object
|
* @return a client Object
|
||||||
*/
|
*/
|
||||||
|
|
|
||||||
|
|
@ -44,20 +44,22 @@ import java.util.logging.Level;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is an abstract client that will do generic requests to a
|
* This is an abstract client that will do generic requests to a
|
||||||
* SOAP Web service
|
* SOAP Web service
|
||||||
*
|
*
|
||||||
* @author Ziver
|
* @author Ziver
|
||||||
*/
|
*/
|
||||||
public class SOAPClientInvocationHandler implements InvocationHandler {
|
public class SOAPClientInvocationHandler implements InvocationHandler {
|
||||||
private static Logger logger = LogUtil.getLogger();
|
private static Logger logger = LogUtil.getLogger();
|
||||||
|
|
||||||
private WebServiceDef wsDef;
|
private WebServiceDef wsDef;
|
||||||
/** Web address of the web service */
|
/**
|
||||||
|
* Web address of the web service
|
||||||
|
*/
|
||||||
protected URL url;
|
protected URL url;
|
||||||
|
|
||||||
|
|
||||||
public SOAPClientInvocationHandler(URL url, WebServiceDef wsDef){
|
public SOAPClientInvocationHandler(URL url, WebServiceDef wsDef) {
|
||||||
this.url = url;
|
this.url = url;
|
||||||
this.wsDef = wsDef;
|
this.wsDef = wsDef;
|
||||||
}
|
}
|
||||||
|
|
@ -69,10 +71,9 @@ public class SOAPClientInvocationHandler implements InvocationHandler {
|
||||||
@Override
|
@Override
|
||||||
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
|
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
|
||||||
// Generate XML
|
// Generate XML
|
||||||
Document document = genSOAPRequest((WSInterface)proxy, method.getName(), args);
|
Document document = genSOAPRequest((WSInterface) proxy, method.getName(), args);
|
||||||
String reqXml = document.asXML();
|
String reqXml = document.asXML();
|
||||||
|
|
||||||
|
|
||||||
// Send request
|
// Send request
|
||||||
HttpClient request = new HttpClient(HttpClient.HttpRequestType.POST);
|
HttpClient request = new HttpClient(HttpClient.HttpRequestType.POST);
|
||||||
request.setURL(url);
|
request.setURL(url);
|
||||||
|
|
@ -82,7 +83,7 @@ public class SOAPClientInvocationHandler implements InvocationHandler {
|
||||||
request.close();
|
request.close();
|
||||||
|
|
||||||
// DEBUG
|
// DEBUG
|
||||||
if( logger.isLoggable(Level.FINEST) ){
|
if (logger.isLoggable(Level.FINEST)) {
|
||||||
System.out.println("********** Request");
|
System.out.println("********** Request");
|
||||||
System.out.println(reqXml);
|
System.out.println(reqXml);
|
||||||
System.out.println("********** Response");
|
System.out.println("********** Response");
|
||||||
|
|
@ -93,23 +94,23 @@ public class SOAPClientInvocationHandler implements InvocationHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private Document genSOAPRequest(WSInterface obj, String targetMethod, Object[] args){
|
private Document genSOAPRequest(WSInterface obj, String targetMethod, Object[] args) {
|
||||||
logger.fine("Sending request for "+targetMethod);
|
logger.fine("Sending request for " + targetMethod);
|
||||||
Document document = DocumentHelper.createDocument();
|
Document document = DocumentHelper.createDocument();
|
||||||
Element envelope = document.addElement("soap:Envelope");
|
Element envelope = document.addElement("soap:Envelope");
|
||||||
WSMethodDef methodDef = wsDef.getMethod( targetMethod );
|
WSMethodDef methodDef = wsDef.getMethod(targetMethod);
|
||||||
try {
|
try {
|
||||||
envelope.addNamespace("soap", "http://schemas.xmlsoap.org/soap/envelope/");
|
envelope.addNamespace("soap", "http://schemas.xmlsoap.org/soap/envelope/");
|
||||||
envelope.addNamespace("xsi", "http://www.w3.org/2001/XMLSchema-instance");
|
envelope.addNamespace("xsi", "http://www.w3.org/2001/XMLSchema-instance");
|
||||||
envelope.addNamespace("xsd", "http://www.w3.org/2001/XMLSchema");
|
envelope.addNamespace("xsd", "http://www.w3.org/2001/XMLSchema");
|
||||||
|
|
||||||
Element body = envelope.addElement( "soap:Body" );
|
Element body = envelope.addElement("soap:Body");
|
||||||
Element method = body.addElement("");
|
Element method = body.addElement("");
|
||||||
method.addNamespace("m", methodDef.getNamespace() );
|
method.addNamespace("m", methodDef.getNamespace());
|
||||||
method.setName("m:"+methodDef.getName()+"Request");
|
method.setName("m:" + methodDef.getName() + "Request");
|
||||||
for(int i=0; i<methodDef.getOutputCount() ;i++){
|
for (int i = 0; i < methodDef.getOutputCount(); i++) {
|
||||||
WSParameterDef param = methodDef.getOutput( i );
|
WSParameterDef param = methodDef.getOutput(i);
|
||||||
SOAPHttpPage.generateSOAPXMLForObj(method, args[i] , param.getName());
|
SOAPHttpPage.generateSOAPXMLForObj(method, args[i], param.getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
|
@ -119,7 +120,7 @@ public class SOAPClientInvocationHandler implements InvocationHandler {
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Object parseSOAPResponse(String xml){
|
private Object parseSOAPResponse(String xml) {
|
||||||
try {
|
try {
|
||||||
Element response = SOAPHttpPage.getXMLRoot(xml);
|
Element response = SOAPHttpPage.getXMLRoot(xml);
|
||||||
// TODO:
|
// TODO:
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue