Created prototype implementation of LB2PC proxy

git-svn-id: https://svn.d4science.research-infrastructures.eu/gcube/private/luca.frosini/infrastructure-tests@165191 82a268e6-3cf1-43bd-a215-b396298e98cf
This commit is contained in:
Luca Frosini 2018-03-16 15:30:35 +00:00
parent 7b5773792a
commit a8bf4f14e3
10 changed files with 637 additions and 0 deletions

View File

@ -0,0 +1,5 @@
package org.gcube.lb2pc;
public enum HttpMethod {
POST, GET, PUT, DELETE, OPTIONS, HEAD, TRACE, CONNECT
}

View File

@ -0,0 +1,38 @@
package org.gcube.lb2pc;
public class HttpRequest {
/**
*
*/
private final Proxy ok;
/**
* @param proxy
*/
HttpRequest(Proxy proxy) {
ok = proxy;
}
public String getHeader(String string) {
return null;
}
public String getRequestURI() {
return null;
}
public HttpMethod getMethod() {
return null;
}
public String getContentType() {
return null;
}
public String getContent() {
// TODO Auto-generated method stub
return null;
}
}

View File

@ -0,0 +1,32 @@
package org.gcube.lb2pc;
import java.util.Collection;
public class HttpResponse {
public String getHeader(String string) {
return null;
}
public Collection<String> getHeaderNames() {
return null;
}
public String getContentType() {
return null;
}
public String getContent() {
return null;
}
public HttpStatusCode getHttpStatusCode() {
return null;
}
public void setHeader(String string, String lockURI) {
// TODO Auto-generated method stub
}
}

View File

@ -0,0 +1,32 @@
package org.gcube.lb2pc;
public enum HttpStatusCode {
// 2×× Success
_200(200, "OK"), _201(201, "Created"), _204(204, "No Content"),
// 4×× Client Error
_400(400, "Bad Request"), _401(401, "Unauthorized"), _403(403, "Forbidden"), _404(404, "Not Found"),
_405(405, "Method Not Allowed"), _406(406, "Not Acceptable"), _408(408, "Request Timeout"),
_423(423, "Locked"),
// 5×× Server Error
_500(500, "Internal Server Error"), _503(503, "Service Unavailable");
private int code;
private String messageCode;
private HttpStatusCode(int code, String messageCode) {
this.code = code;
this.messageCode = messageCode;
}
public int getCode() {
return code;
}
public String getMessageCode() {
return messageCode;
}
}

View File

@ -0,0 +1,100 @@
package org.gcube.lb2pc;
public class Lock {
protected String lockURI;
protected LockType type;
protected String transactionURI;
protected String resourceURI;
protected String contentType;
public Lock(LockType type, String transactionURI, String resourceURI) {
super();
this.type = type;
this.transactionURI = transactionURI;
this.resourceURI = resourceURI;
}
public Lock(String lockURI) {
this.lockURI = lockURI;
}
public LockType getType() {
return type;
}
public void setType(LockType type) {
this.type = type;
}
public String getTransactionURI() {
return transactionURI;
}
public void setTransactionURI(String transactionURI) {
this.transactionURI = transactionURI;
}
public String getResourceURI() {
return resourceURI;
}
public void setResourceURI(String resourceURI) {
this.resourceURI = resourceURI;
}
public String getLockURI() {
return lockURI;
}
public String getContentType() {
return contentType;
}
public void setContentType(String contentType) {
this.contentType = contentType;
}
public void setLockURI(String lockURI) {
this.lockURI = lockURI;
}
public void create() {
}
public void read(String contentType) {
this.contentType = contentType;
HttpResponse lockResponse = Proxy.createHttpRequest(HttpMethod.GET, contentType, lockURI);
if(lockResponse.getHttpStatusCode() != HttpStatusCode._200) {
// Lock Not Found or any error which does not allow to proceed
Proxy.sendErrorResponseToClient(HttpStatusCode._401, "Provided Lock does not exist");
}
unmarshalContent(lockResponse.getContent(), contentType);
}
private void unmarshalContent(String content, String contentType) {
// Set fields
}
public void upgrade() {
if(type == LockType.X) {
return;
}
type = LockType.S;
HttpResponse lockResponse = Proxy.createHttpRequest(HttpMethod.PUT, contentType, lockURI);
}
public void delete() {
HttpResponse lockResponse = Proxy.createHttpRequest(HttpMethod.DELETE, null, lockURI);
}
}

View File

@ -0,0 +1,5 @@
package org.gcube.lb2pc;
public enum LockType {
S, X
}

View File

@ -0,0 +1,81 @@
package org.gcube.lb2pc;
import java.net.MalformedURLException;
import java.net.URL;
public class Log {
protected HttpMethod operation;
protected String resource;
protected URL locationURI;
protected String contentType;
protected String shadowResourceURI;
public Log(HttpMethod httpMethod) {
super();
this.operation = httpMethod;
this.resource = null;
}
public Log(HttpMethod httpMethod, String resource, String locationURI) throws MalformedURLException {
super();
this.operation = httpMethod;
this.resource = resource;
this.locationURI = new URL(locationURI);
}
public HttpMethod getOperation() {
return operation;
}
public void setOperation(HttpMethod operation) {
this.operation = operation;
}
public String getResource() {
return resource;
}
public void setResource(String resource) {
this.resource = resource;
}
public String getContentType() {
return contentType;
}
public void setContentType(String contentType) {
this.contentType = contentType;
}
public String getShadowResourceURI() {
return shadowResourceURI;
}
public void setShadowResourceURI(String shadowResourceURI) {
this.shadowResourceURI = shadowResourceURI;
}
public void create(String shadowResourceURI, String contentType) {
this.contentType = contentType;
this.shadowResourceURI = shadowResourceURI;
HttpResponse logResponse = Proxy.createHttpRequest(HttpMethod.POST, contentType, getLogCollectionURI());
if(logResponse.getHttpStatusCode() != HttpStatusCode._201) {
// Unable to create log. This does not allow to proceed
Proxy.sendErrorResponseToClient(HttpStatusCode._401, "Provided Lock does not exist");
}
unmarshalContent(this, logResponse.getContent(), contentType);
}
protected String getLogCollectionURI() {
return shadowResourceURI + "/operations";
}
public static void unmarshalContent(Log operationLog, String content, String contentType) {
// Set fields
}
}

View File

@ -0,0 +1,229 @@
package org.gcube.lb2pc;
import java.net.MalformedURLException;
public class Proxy {
public HttpResponse service(HttpRequest request) throws Exception {
HttpMethod httpMethod = request.getMethod();
if(httpMethod == HttpMethod.OPTIONS) {
sendTransactionServerList();
}
String transactionURI = request.getHeader("X-Transaction-URI");
String requestURI = request.getRequestURI();
String contentType = "application/json";
if(transactionURI!=null) {
// Transactional Client
if(request.getHeader("Accept")!=null) {
contentType = request.getHeader("Accept");
}
String lockURI = request.getHeader("X-Lock-URI");
if(lockURI!=null) {
// Client already own a lock, verifying and upgrading if needed
verifyAndUpgradeLock(httpMethod, contentType, lockURI, transactionURI, requestURI);
String shadowResourceURI = ShadowResource.getShadowResourceURI(transactionURI, requestURI);
logRequest(shadowResourceURI, contentType, httpMethod, null);
HttpResponse actionResponse = forwardAction(request);
actionResponse.setHeader("X-Lock-URI", lockURI);
actionResponse.setHeader("X-Transaction-URI", transactionURI);
return replyToClient(actionResponse);
}else {
// Client does not own a lock
if(httpMethod != HttpMethod.POST && httpMethod != HttpMethod.PUT &&
httpMethod != HttpMethod.GET && httpMethod != HttpMethod.HEAD) {
// The first action on a resource must be a GET or a CREATE
sendErrorResponseToClient(HttpStatusCode._403, "A resource must be read before to be able to modify or delete it.");
}
HttpResponse actionResponse = null;
Lock lock = createLock(httpMethod, transactionURI, requestURI);
lockURI = lock.getLockURI();
String resource = null;
String shadowResourceURI= null;
switch(httpMethod) {
case PUT:
HttpResponse r = createHttpRequest(HttpMethod.HEAD, null, null, requestURI);
if(r.getHttpStatusCode() == HttpStatusCode._204 || (r.getHttpStatusCode().getCode()>=200 && r.getHttpStatusCode().getCode()<300)) {
lock.delete();
sendErrorResponseToClient(HttpStatusCode._403, "A resource must be read before to be able to modify or delete it.");
}else {
// Create with PUT
shadowResourceURI = createShadowResource(transactionURI, requestURI, lockURI, contentType, null);
logRequest(shadowResourceURI, contentType, httpMethod, request.getContent());
actionResponse = forwardAction(request);
}
break;
case POST:
HttpResponse getResponse = getResourceOnEffectiveService(requestURI);
resource = getResponse.getContent();
shadowResourceURI = createShadowResource(transactionURI, requestURI, lockURI, contentType, resource);
logRequest(shadowResourceURI, contentType, httpMethod, request.getContent());
actionResponse = forwardAction(request);
String locationURI = actionResponse.getHeader("Location");
logRequest(shadowResourceURI, contentType, httpMethod, null, locationURI);
break;
default:
HttpResponse response = getResourceOnEffectiveService(requestURI);
resource = response.getContent();
shadowResourceURI = createShadowResource(transactionURI, requestURI, lockURI, contentType, resource);
// We don't need any log
// logRequest(shadowResourceURI, contentType, httpMethod, requestURI, request.getContent());
actionResponse = forwardAction(request);
break;
}
actionResponse.setHeader("X-Lock-URI", lockURI);
actionResponse.setHeader("X-Transaction-URI", transactionURI);
return replyToClient(actionResponse);
}
}else {
// Non-Transactional Client
return createMiniTransaction();
}
}
private void logRequest(String shadowResourceURI, String contentType, HttpMethod httpMethod, String resource)
throws MalformedURLException {
this.logRequest(shadowResourceURI, contentType, httpMethod, resource, null);
}
private void logRequest(String shadowResourceURI, String contentType, HttpMethod httpMethod, String resource, String locationURI) throws MalformedURLException {
Log log = new Log(httpMethod, resource, locationURI);
log.create(shadowResourceURI, contentType);
}
private HttpResponse createMiniTransaction() {
// TODO Auto-generated method stub
return null;
}
private void sendTransactionServerList() {
// TODO Auto-generated method stub
}
private static HttpResponse replyToClient(HttpResponse actionResponse) {
return null;
}
private static HttpResponse forwardAction(HttpRequest request) {
return null;
}
private HttpResponse getResourceOnEffectiveService(String resourceURI) {
return null;
}
private String createShadowResource(String transactionURI, String requestURI, String lockURI, String contentType,
String resource) throws MalformedURLException {
ShadowResource shadowResource = new ShadowResource(transactionURI, requestURI, lockURI, contentType);
shadowResource.create();
return shadowResource.getURI();
}
protected Lock createLock(HttpMethod httpMethod, String transacationURI, String resourceURI) {
LockType type = getRequiredType(httpMethod);
Lock lock = new Lock(type, transacationURI, resourceURI);
lock.create();
return lock;
}
private LockType getRequiredType(HttpMethod httpMethod) {
return null;
}
protected void verifyAndUpgradeLock(HttpMethod httpMethod, String contentType, String lockURI,
String transactionURI, String resourceURI) {
Lock lock = retrieveLock(lockURI, contentType);
if(lock.getTransactionURI().compareTo(transactionURI) != 0) {
// Invalid Lock e.g. the lock does not belong to the transaction or does not exist
sendErrorResponseToClient(HttpStatusCode._401, "Provided Lock does belong to provided Transaction");
}
if(lock.getResourceURI().compareTo(resourceURI) != 0) {
sendErrorResponseToClient(HttpStatusCode._401, "Provided Lock does belong to requested URI");
}
if(lock.type == LockType.S) {
if(httpMethod != HttpMethod.GET) {
upgradeLock(lock, lockURI);
// lock.upgrade();
}
}
}
public static void sendErrorResponseToClient(HttpStatusCode name, String string) {
return;
}
/**
* GET lock from LockURI
* @param lockURI
* @return
*/
private Lock retrieveLock(String lockURI, String contentType) {
Lock lock = new Lock(lockURI);
lock.read(contentType);
return lock;
}
public static HttpResponse createHttpRequest(HttpMethod get, String contentType, String uri) {
return null;
}
public static HttpResponse createHttpRequest(HttpMethod get, String contentType, String content, String uri) {
return null;
}
protected Lock upgradeLock(Lock lock, String lockURI) {
if(lock.getType() == LockType.X) {
return lock;
}
lock.setType(LockType.S);
sendLockUpdate(lock, lockURI);
return lock;
}
private void sendLockUpdate(Lock lock, String lockURI) {
return;
}
}

View File

@ -0,0 +1,110 @@
package org.gcube.lb2pc;
import java.net.MalformedURLException;
import java.net.URL;
public class ShadowResource {
protected String transactionURI;
protected String requestURI;
protected String lockURI;
protected String contentType;
protected String content;
public ShadowResource(String transactionURI, String requestURI, String lockURI, String contentType) {
super();
this.transactionURI = transactionURI;
this.requestURI = requestURI;
this.lockURI = lockURI;
this.contentType = contentType;
}
public ShadowResource(String transactionURI, String requestURI, String lockURI, String contentType,
String content) {
super();
this.transactionURI = transactionURI;
this.requestURI = requestURI;
this.lockURI = lockURI;
this.contentType = contentType;
this.content = content;
}
public String getTransactionURI() {
return transactionURI;
}
public void setTransactionURI(String transactionURI) {
this.transactionURI = transactionURI;
}
public String getRequestURI() {
return requestURI;
}
public void setRequestURI(String requestURI) {
this.requestURI = requestURI;
}
public String getLockURI() {
return lockURI;
}
public void setLockURI(String lockURI) {
this.lockURI = lockURI;
}
public String getContentType() {
return contentType;
}
public void setContentType(String contentType) {
this.contentType = contentType;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
public void create() throws MalformedURLException {
String shadowResourceURI = getURI();
HttpResponse response = Proxy.createHttpRequest(HttpMethod.PUT, contentType, marshall(), shadowResourceURI);
if(response.getHttpStatusCode() != HttpStatusCode._201) {
Proxy.sendErrorResponseToClient(HttpStatusCode._403, "Unable to fullfill request");
}
}
protected String marshall() {
return null;
}
protected String getURI() throws MalformedURLException {
return getShadowResourceURI(transactionURI, requestURI);
}
protected static String getPath(String uri) throws MalformedURLException {
URL url = new URL(uri);
return url.getPath();
}
public static String getShadowResourceURI(String transactionURI, String resourceURI) throws MalformedURLException {
String path = getPath(resourceURI);
if(path.contains("/operations/")) {
path.replace("/operations/", "/_operations/");
}
if(path.endsWith("/operations")) {
path.replace("/operations", "/_operations");
}
if(transactionURI.endsWith("/")) {
transactionURI = transactionURI.substring(0, transactionURI.length() - 2);
}
return transactionURI + path;
}
}

View File

@ -0,0 +1,5 @@
package org.gcube.lb2pc;
public class Utility {
}