com.smartitengineering.util.rest.client.AbstractClientResource.java Source code

Java tutorial

Introduction

Here is the source code for com.smartitengineering.util.rest.client.AbstractClientResource.java

Source

/*
 * This is a utility project for wide range of applications
 *
 * Copyright (C) 2010  Imran M Yousuf (imyousuf@smartitengineering.com)
 *
 * This library 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 library 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
 * Lesser General Public License for more details.
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  10-1  USA
 */
package com.smartitengineering.util.rest.client;

import com.sun.jersey.api.client.Client;
import com.sun.jersey.api.client.ClientResponse;
import com.sun.jersey.api.client.ClientResponse.Status;
import com.sun.jersey.api.client.UniformInterfaceException;
import com.sun.jersey.api.client.WebResource;
import com.sun.jersey.api.client.WebResource.Builder;
import com.sun.jersey.api.client.config.ClientConfig;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.net.URI;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import javax.ws.rs.core.EntityTag;
import javax.ws.rs.core.HttpHeaders;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.core.UriBuilder;
import org.apache.commons.lang.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * An abstract resource representation on client side for a client to a RESTful Web Service. It is designed to have one
 * configuration application wide and one Apache Http Client application wide by default. It uses the Jersey Cacheable
 * Client which uses HTTPCache4J and thus if the configuration is changed for username/password it will reflect on next
 * request. But it is possible to change the client, config by simply specifying your own {@link ClientFactory}
 * @author imyousuf
 * @since 0.2
 */
public abstract class AbstractClientResource<T, P extends Resource>
        implements Resource<T>, WritableResource<T>, PaginatedResource<P>, ConfigProcessor {

    protected static final URI BASE_URI;
    protected static final ConnectionConfig CONNECTION_CONFIG;

    static {
        CONNECTION_CONFIG = ConfigFactory.getInstance().getConnectionConfig();
        if (CONNECTION_CONFIG != null) {
            BASE_URI = UriBuilder.fromUri(CONNECTION_CONFIG.getContextPath()).path(CONNECTION_CONFIG.getBasicUri())
                    .host(CONNECTION_CONFIG.getHost()).port(CONNECTION_CONFIG.getPort()).scheme("http").build();
        } else {
            BASE_URI = null;
        }
    }
    protected Logger logger = LoggerFactory.getLogger(getClass());
    private Resource referrer;
    private URI thisResourceUri;
    private URI absoluteThisResourceUri;
    private String representationType;
    private Class<? extends T> entityClass;
    private T lastReadStateOfEntity;
    private MultivaluedMap<String, ResourceLink> relatedResourceUris;
    private ClientUtil clientUtil;
    private ClientFactory clientFactory;
    private int getInvocationCount;
    private boolean followRedirectionEnabled;
    private boolean invokeGet;
    private final Map<String, Resource> nestedResources;
    private final Map<String, Map<String, Object>> cachedHeaders;

    protected AbstractClientResource(Resource referrer, ResourceLink resouceLink)
            throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, resouceLink, null);
    }

    protected AbstractClientResource(Resource referrer, ResourceLink resouceLink, Class<? extends T> entityClass)
            throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, resouceLink, entityClass, ClientUtilFactory.getInstance().getClientUtil(entityClass));
    }

    protected AbstractClientResource(Resource referrer, ResourceLink resouceLink, Class<? extends T> entityClass,
            ClientUtil clientUtil) throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, resouceLink, entityClass, clientUtil, true);
    }

    protected AbstractClientResource(Resource referrer, ResourceLink resouceLink, Class<? extends T> entityClass,
            ClientUtil clientUtil, boolean invokeGet) throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, resouceLink, entityClass, clientUtil, invokeGet, null);
    }

    protected AbstractClientResource(Resource referrer, ResourceLink resouceLink, Class<? extends T> entityClass,
            ClientUtil clientUtil, boolean invokeGet, ClientFactory clientFactory)
            throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, resouceLink.getUri(), resouceLink.getMimeType(), entityClass, clientUtil, invokeGet,
                clientFactory, true);
    }

    protected AbstractClientResource(Resource referrer, URI thisResourceUri, String representationType,
            Class<? extends T> entityClass) throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, thisResourceUri, representationType, entityClass,
                ClientUtilFactory.getInstance().getClientUtil(entityClass));
    }

    protected AbstractClientResource(Resource referrer, URI thisResourceUri, String representationType,
            Class<? extends T> entityClass, ClientUtil clientUtil)
            throws IllegalArgumentException, UniformInterfaceException {
        this(referrer, thisResourceUri, representationType, entityClass, clientUtil, true, null, true);
    }

    /**
     * Construct a generic HTTP client resource's super class with necessary information for it to work properly
     * @param referrer The resource from which one arrived to this resource
     * @param thisResourceUri The URI, could be absolute or relative, of this resource.
     * @param representationType The MIME Type to expect for this resource, a.k.a., Accept HTTP header
     * @param entityClass The Entity class to ask Jersey to de-serialize the GET entity to.
     * @param clientUtil The client util instance to parse linked resources for this representation entity.
     * @param invokeGet If true GET will be issued during construction synchronously. It also means that on GET nested 
     *                  resources will also be GET so synchronize them. By default its true.
     * @throws IllegalArgumentException If thisResourceUri or representationType or entityClass is null
     * @throws UniformInterfaceException If status is anything but < 300 or 304.
     */
    protected AbstractClientResource(Resource referrer, URI thisResourceUri, String representationType,
            Class<? extends T> entityClass, ClientUtil clientUtil, boolean invokeGet, ClientFactory clientFactory,
            boolean followRedirection) throws IllegalArgumentException, UniformInterfaceException {
        if (thisResourceUri == null) {
            throw new IllegalArgumentException("URI to current resource can not be null");
        }
        if (StringUtils.isBlank(representationType)) {
            throw new IllegalArgumentException("Accept header value can not be null!");
        }
        if (entityClass == null) {
            entityClass = initializeEntityClassFromGenerics();
            if (entityClass == null) {
                throw new IllegalArgumentException("Entity class can not be null!");
            }
        }
        if (clientUtil == null && entityClass != null) {
            clientUtil = ClientUtilFactory.getInstance().getClientUtil(entityClass);
        }
        if (clientFactory == null) {
            if (referrer == null) {
                clientFactory = ApplicationWideClientFactoryImpl.getClientFactory(CONNECTION_CONFIG, this);
            } else {
                clientFactory = referrer.getClientFactory();
            }
        }
        this.clientFactory = clientFactory;
        this.referrer = referrer;
        this.thisResourceUri = thisResourceUri;
        this.representationType = representationType;
        this.entityClass = entityClass;
        this.relatedResourceUris = new ConcurrentMultivalueMap<String, ResourceLink>();
        this.clientUtil = clientUtil;
        this.absoluteThisResourceUri = generateAbsoluteUri();
        this.followRedirectionEnabled = followRedirection;
        this.nestedResources = new HashMap<String, Resource>();
        this.cachedHeaders = new HashMap<String, Map<String, Object>>();
        if (invokeGet) {
            get();
        }
    }

    protected final Class<? extends T> initializeEntityClassFromGenerics() {
        Class<? extends T> extractedEntityClass = null;
        try {
            Type paramType = ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
            if (paramType instanceof ParameterizedType) {
                paramType = ((ParameterizedType) paramType).getRawType();
            }
            Class<T> pesistenceRegistryClass = paramType instanceof Class ? (Class<T>) paramType : null;
            if (logger.isDebugEnabled()) {
                logger.debug("Entity class predicted to: " + pesistenceRegistryClass.toString());
            }
            extractedEntityClass = pesistenceRegistryClass;
        } catch (Exception ex) {
            logger.warn("Could not predict entity class ", ex);
        }
        return extractedEntityClass;
    }

    protected boolean isFollowRedirectionEnabled() {
        return followRedirectionEnabled;
    }

    protected void setFollowRedirectionEnabled(boolean followRedirectionEnabled) {
        this.followRedirectionEnabled = followRedirectionEnabled;
    }

    protected boolean isInvokeGet() {
        return invokeGet;
    }

    protected void setInvokeGet(boolean invokeGet) {
        this.invokeGet = invokeGet;
    }

    protected void addNestedResource(String key, Resource resource) {
        nestedResources.put(key, resource);
    }

    protected void removeNestedResource(String key) {
        nestedResources.remove(key);
    }

    protected <K> Resource<K> getNestedResource(String key) {
        return nestedResources.get(key);
    }

    protected Map<String, Resource> getNestedResources() {
        return Collections.unmodifiableMap(nestedResources);
    }

    protected final URI generateAbsoluteUri() {
        final URI thisUri = this.thisResourceUri;
        final URI referrerUri = getReferrerUri();
        return getHttpClient().getAbsoluteUri(thisUri, referrerUri);
    }

    protected final URI getReferrerUri() {
        return this.referrer == null ? null : this.referrer.getUri();
    }

    protected void getIfFirstTimeRequest() {
        if (getGetInvocationCount() <= 0) {
            get();
        }
    }

    protected int getGetInvocationCount() {
        return getInvocationCount;
    }

    protected ClientUtil getClientUtil() {
        return clientUtil;
    }

    protected MultivaluedMap<String, ResourceLink> getRelatedResourceUris() {
        return relatedResourceUris;
    }

    @Override
    public Class<? extends T> getEntityClass() {
        return entityClass;
    }

    @Override
    public String getResourceRepresentationType() {
        return representationType;
    }

    @Override
    public final T get() {
        return get(getUri());
    }

    protected T get(URI uri) {
        ClientResponse response = ClientUtil.readEntity(uri, getHttpClient(), getResourceRepresentationType(),
                ClientResponse.class);
        response.bufferEntity();
        if (logger.isDebugEnabled()) {
            logger.debug("Request Accept header: " + getResourceRepresentationType());
            logger.debug("Response header: " + response.getType());
        }
        final int status = response.getStatus();
        if (followRedirectionEnabled && status == ClientResponse.Status.MOVED_PERMANENTLY.getStatusCode()) {
            final URI location = response.getLocation();
            if (location != null) {
                this.thisResourceUri = location;
                this.absoluteThisResourceUri = generateAbsoluteUri();
                return get();
            }
        }
        if (followRedirectionEnabled && (status == ClientResponse.Status.FOUND.getStatusCode()
                || status == ClientResponse.Status.SEE_OTHER.getStatusCode())) {
            final URI location = response.getLocation();
            if (location != null) {
                URI absolutionLocation = getHttpClient().getAbsoluteUri(location, getReferrerUri());
                return get(absolutionLocation);
            }
        }
        if (status < 300 || (status == ClientResponse.Status.NOT_MODIFIED.getStatusCode())) {
            if (response.hasEntity() && response.getStatus() != ClientResponse.Status.NO_CONTENT.getStatusCode()) {
                lastReadStateOfEntity = response.getEntity(getEntityClass());
                if (getClientUtil() != null) {
                    try {
                        getClientUtil().parseLinks(lastReadStateOfEntity, getRelatedResourceUris());
                    } catch (Exception ex) {
                        logger.warn(ex.getMessage(), ex);
                    }
                }
                if (invokeGet) {
                    invokeGETOnNestedResources();
                }
            } else {
                lastReadStateOfEntity = null;
            }
            getInvocationCount++;
            Map<String, Object> headers = new HashMap<String, Object>();
            EntityTag tag = response.getEntityTag();
            if (tag != null) {
                headers.put(HttpHeaders.ETAG, tag);
            }
            Date date = response.getLastModified();
            if (date != null) {
                headers.put(HttpHeaders.LAST_MODIFIED, date);
            }
            cachedHeaders.put(uri.toString(), headers);
            return lastReadStateOfEntity;
        }
        throw new UniformInterfaceException(response, true);
    }

    protected void invokeGETOnNestedResources() {
        final Collection<Resource> values = nestedResources.values();
        for (Resource resource : values) {
            resource.get();
        }
    }

    @Override
    public T getLastReadStateOfEntity() {
        getIfFirstTimeRequest();
        return lastReadStateOfEntity;
    }

    @Override
    public URI getUri() {
        return getAbsoluteThisResourceUri();
    }

    @Override
    public ClientResponse delete(Status... status) {
        WebResource webResource = getHttpClient().getWebResource(getUri());
        Builder builder = addNecessaryHeaders(getUri(), webResource);
        final ClientResponse response = builder.delete(ClientResponse.class);
        response.bufferEntity();
        checkStatus(response, status);
        return response;
    }

    @Override
    public <P> ClientResponse put(String contentType, P param, Status... status) {
        WebResource webResource = getHttpClient().getWebResource(getUri());
        Builder builder = addNecessaryHeaders(getUri(), webResource);
        builder.type(contentType);
        final ClientResponse response = builder.put(ClientResponse.class, param);
        response.bufferEntity();
        checkStatus(response, status);
        return response;
    }

    @Override
    public <P> ClientResponse post(String contentType, P param, Status... status) {
        WebResource webResource = getHttpClient().getWebResource(getUri());
        Builder builder = addNecessaryHeaders(getUri(), webResource);
        builder.type(contentType);
        final ClientResponse response = builder.post(ClientResponse.class, param);
        response.bufferEntity();
        checkStatus(response, status);
        return response;
    }

    @Override
    public <V> Resource<V> getReferrer() {
        return referrer;
    }

    @Deprecated
    public URI getBaseUri() {
        return BASE_URI;
    }

    protected URI getThisResourceUri() {
        return thisResourceUri;
    }

    protected URI getAbsoluteThisResourceUri() {
        return absoluteThisResourceUri;
    }

    @Deprecated
    protected UriBuilder getBaseUriBuilder() {
        return UriBuilder.fromUri(BASE_URI.toString());
    }

    protected UriBuilder getCurrentUriBuilder() {
        return UriBuilder.fromUri(getAbsoluteThisResourceUri());
    }

    protected final Client getClient() {
        return clientFactory.getClient();
    }

    protected final ClientConfig getClientConfig() {
        return clientFactory.getClientConfig();
    }

    protected final HttpClient getHttpClient() {
        return clientFactory.getHttpClient();
    }

    protected abstract void processClientConfig(ClientConfig clientConfig);

    @Override
    public final void process(ClientConfig clientConfig) {
        processClientConfig(clientConfig);
    }

    @Override
    public ResourceLink nextUri() {
        return getNextUri();
    }

    @Override
    public ResourceLink previousUri() {
        return getPreviousUri();
    }

    @Override
    public P next() {
        return getPageableResource(nextUri());
    }

    @Override
    public P previous() {
        return getPageableResource(previousUri());
    }

    @Override
    public ClientFactory getClientFactory() {
        return clientFactory;
    }

    protected abstract ResourceLink getNextUri();

    protected abstract ResourceLink getPreviousUri();

    protected P getPageableResource(ResourceLink link) {
        if (link == null) {
            return null;
        }
        return instantiatePageableResource(link);
    }

    protected abstract P instantiatePageableResource(ResourceLink link);

    protected void checkStatus(ClientResponse response, Status... status) {
        if (status == null || status.length == 0) {
            return;
        }
        if (Arrays.<Status>asList(status).contains(response.getClientResponseStatus())) {
            return;
        }
        throw new UniformInterfaceException(response);
    }

    protected Builder addNecessaryHeaders(URI uri, WebResource resource) {
        Map<String, Object> headers = cachedHeaders.get(uri.toString());
        Builder builder = resource.getRequestBuilder();
        if (headers != null) {
            final Object etag = headers.get(HttpHeaders.ETAG);
            if (etag != null) {
                builder.header(HttpHeaders.IF_MATCH, etag);
            }
            final Object lastModified = headers.get(HttpHeaders.LAST_MODIFIED);
            if (lastModified != null) {
                builder.header(HttpHeaders.IF_UNMODIFIED_SINCE, lastModified);
            }
        }
        return builder;
    }
}