/**
*
* Copyright
* 2009-2015 Jayway Products AB
* 2016-2017 Föreningen Sambruk
*
* Licensed under AGPL, Version 3.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.gnu.org/licenses/agpl.txt
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package se.streamsource.dci.restlet.client;
import java.io.IOException;
import org.qi4j.api.injection.scope.Uses;
import org.qi4j.api.util.Iterables;
import org.restlet.Request;
import org.restlet.Response;
import org.restlet.data.MediaType;
import org.restlet.data.Method;
import org.restlet.data.Reference;
import org.restlet.data.Status;
import org.restlet.representation.EmptyRepresentation;
import org.restlet.representation.ObjectRepresentation;
import org.restlet.resource.ResourceException;
import se.streamsource.dci.value.ResourceValue;
import se.streamsource.dci.value.link.LinkValue;
import se.streamsource.dci.value.link.Links;
/**
* Base class for client-side Command/Query resources
*/
public class CommandQueryClient
{
@Uses
private CommandQueryClientFactory cqcFactory;
@Uses
private Reference reference;
private ResourceValue resourceValue;
public Reference getReference()
{
return reference;
}
public ResourceValue getResource()
{
return resourceValue;
}
// Queries
public synchronized ResourceValue query() throws ResourceException
{
return resourceValue = query( "", ResourceValue.class);
}
public synchronized <T> T query( String operation, Class<T> queryResult ) throws ResourceException
{
return query( operation, queryResult, null);
}
public synchronized <T> T query(String operation, Class<T> queryResult, Object queryRequest) throws ResourceException
{
Response response = invokeQuery( operation, queryRequest );
if (response.getStatus().isSuccess())
{
cqcFactory.updateCache( response );
return cqcFactory.readResponse( response, queryResult );
} else
{
// This will throw an exception
handleError( response );
return null;
}
}
public synchronized <T> T queryLink(LinkValue link, Class<T> queryResult)
{
return query( link.href().get(), queryResult );
}
public boolean hasQueryWithRelation(String relation)
{
return Iterables.matchesAny( Links.withRel( relation ), resourceValue.queries().get() );
}
public synchronized <T> T queryByRelation(String relation, Class<T> resultClass)
{
LinkValue link = Iterables.first( Iterables.filter( Links.withRel( relation ), resourceValue.queries().get()));
if (link == null) {
throw new ResourceException( Status.CLIENT_ERROR_NOT_FOUND );
}
return queryLink(link, resultClass);
}
public boolean hasCommandWithRelation(String relation)
{
return Iterables.matchesAny( Links.withRel( relation ), resourceValue.commands().get() );
}
// Commands
public synchronized void command(String relation)
throws ResourceException
{
LinkValue link = Iterables.first( Iterables.filter( Links.withRel( relation ), resourceValue.commands().get()));
if (link == null)
throw new ResourceException( Status.CLIENT_ERROR_NOT_FOUND );
// Check if we should do POST or PUT
if (Links.withClass( "idempotent" ).satisfiedBy( link ))
putCommand( link.href().get() );
else
postLink( link );
}
public synchronized void postLink( LinkValue link ) throws ResourceException
{
postCommand( link.href().get(), new EmptyRepresentation() );
}
public synchronized void postLink( LinkValue link, Object requestObject ) throws ResourceException
{
postCommand( link.href().get(), requestObject );
}
public synchronized void postCommand( String operation ) throws ResourceException
{
postCommand( operation, new EmptyRepresentation() );
}
public synchronized void postCommand( String operation, Object requestObject )
throws ResourceException
{
postCommand( operation, requestObject, cqcFactory.getHandler() );
}
public synchronized void postCommand( String operation, Object requestObject, ResponseHandler responseHandler )
throws ResourceException
{
Reference ref = new Reference( reference.toUri().toString() + operation );
Request request = new Request( Method.POST, ref );
cqcFactory.writeRequest(request, requestObject);
cqcFactory.updateCommandRequest( request );
Response response = new Response( request );
cqcFactory.getClient().handle( request, response );
try
{
if (response.getStatus().isSuccess())
{
cqcFactory.updateCache( response );
responseHandler.handleResponse( response );
} else
{
handleError( response );
}
} finally
{
try
{
response.getEntity().exhaust();
} catch (Throwable e)
{
// Ignore
}
}
}
public synchronized void create() throws ResourceException
{
putCommand( null );
}
public synchronized void putCommand( String operation ) throws ResourceException
{
putCommand( operation, null, cqcFactory.getHandler() );
}
public synchronized void putCommand( String operation, Object requestObject )
throws ResourceException
{
postCommand( operation, requestObject, cqcFactory.getHandler() );
}
public synchronized void putCommand( String operation, Object requestObject, ResponseHandler responseHandler) throws ResourceException
{
Reference ref = new Reference( reference.toUri().toString() );
if (operation != null)
{
ref = ref.addSegment( operation );
}
Request request = new Request( Method.PUT, ref );
cqcFactory.writeRequest(request, requestObject);
cqcFactory.updateCommandRequest(request);
int tries = 3;
while (true)
{
try
{
Response response = new Response( request );
cqcFactory.getClient().handle( request, response );
try
{
if (response.getStatus().isSuccess())
{
cqcFactory.updateCache( response );
responseHandler.handleResponse( response );
} else
{
handleError( response );
}
} finally
{
try
{
response.getEntity().exhaust();
} catch (Throwable e)
{
// Ignore
}
}
break;
} catch (ResourceException e)
{
if (e.getStatus().equals( Status.CONNECTOR_ERROR_COMMUNICATION ) ||
e.getStatus().equals( Status.CONNECTOR_ERROR_CONNECTION ))
{
if (tries == 0)
throw e; // Give up
else
{
// Try again
tries--;
continue;
}
} else
{
// Abort
throw e;
}
}
}
}
// Delete
public synchronized void delete() throws ResourceException
{
delete(cqcFactory.getHandler());
}
public synchronized void delete(ResponseHandler responseHandler) throws ResourceException
{
Request request = new Request( Method.DELETE, new Reference( reference.toUri() ).toString() );
cqcFactory.updateCommandRequest( request );
int tries = 3;
while (true)
{
Response response = new Response( request );
try
{
cqcFactory.getClient().handle( request, response );
if (!response.getStatus().isSuccess())
{
handleError( response );
} else
{
// Reset modification date
cqcFactory.updateCache( response );
responseHandler.handleResponse( response );
}
break;
} catch (ResourceException e)
{
if (e.getStatus().equals( Status.CONNECTOR_ERROR_COMMUNICATION ) ||
e.getStatus().equals( Status.CONNECTOR_ERROR_CONNECTION ))
{
if (tries == 0)
throw e; // Give up
else
{
// Try again
tries--;
continue;
}
} else
{
// Abort
throw e;
}
} finally
{
try
{
response.getEntity().exhaust();
} catch (Throwable e)
{
// Ignore
}
}
}
}
// Browse to other resources
public synchronized CommandQueryClient getSubClient( String pathSegment )
{
Reference subReference = reference.clone().addSegment( pathSegment ).addSegment( "" );
return cqcFactory.newClient(subReference);
}
public synchronized CommandQueryClient getClient( String relativePath )
{
if (relativePath.startsWith("http://"))
return cqcFactory.newClient(new Reference(relativePath));
Reference reference = this.reference.clone();
if (relativePath.startsWith( "/" ))
reference.setPath( relativePath );
else
{
reference.setPath( reference.getPath() + relativePath );
reference = reference.normalize();
}
return cqcFactory.newClient( reference );
}
public synchronized CommandQueryClient getClient( LinkValue link )
{
if (link == null)
throw new NullPointerException("No link specified");
return getClient( link.href().get() );
}
// Internal
private Object handleError( Response response )
throws ResourceException
{
if (response.getStatus().equals( Status.SERVER_ERROR_INTERNAL ))
{
if (MediaType.APPLICATION_JAVA_OBJECT.equals(response.getEntity().getMediaType()))
{
try
{
Object exception = new ObjectRepresentation( response.getEntity() ).getObject();
throw new ResourceException( (Throwable) exception );
} catch (IOException e)
{
throw new ResourceException( e );
} catch (ClassNotFoundException e)
{
throw new ResourceException( e );
}
}
throw new ResourceException( Status.SERVER_ERROR_INTERNAL, response.getEntityAsText() );
} else
{
if (response.getEntity() != null)
{
String text = response.getEntityAsText();
throw new ResourceException( response.getStatus().getCode(), response.getStatus().getName(), text, response.getRequest().getResourceRef().toUri().toString() );
} else
{
throw new ResourceException( response.getStatus().getCode(), response.getStatus().getName(), response.getStatus().getDescription(), response.getRequest().getResourceRef().toUri().toString() );
}
}
}
private Response invokeQuery( String operation, Object queryRequest )
throws ResourceException
{
Reference ref = new Reference( reference.toUri().toString() + operation );
Request request = new Request( Method.GET, ref );
if (queryRequest != null)
cqcFactory.writeRequest(request, queryRequest);
cqcFactory.updateQueryRequest( request );
Response response = new Response( request );
cqcFactory.getClient().handle( request, response );
return response;
}
@Override
public String toString()
{
return reference.toString();
}
}