census.couchdroid.CouchSession.java Source code

Java tutorial

Introduction

Here is the source code for census.couchdroid.CouchSession.java

Source

/*
Copyright 2007 Fourspaces Consulting, LLC
    
Licensed under the Apache License, Version 2.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.apache.org/licenses/LICENSE-2.0
    
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 census.couchdroid;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.List;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

//import org.apache.commons.lang.exception.ExceptionUtils;
import org.apache.commons.logging.Log;
//import org.apache.commons.logging.LogFactory;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.conn.scheme.SchemeRegistry;
import org.apache.http.conn.scheme.Scheme;
import org.apache.http.conn.scheme.PlainSocketFactory;
import org.apache.http.conn.ssl.SSLSocketFactory;
import org.apache.http.params.HttpParams;
import org.apache.http.params.BasicHttpParams;
import org.apache.http.util.ExceptionUtils;
import org.apache.http.auth.AuthScope;
import org.apache.http.auth.UsernamePasswordCredentials;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpDelete;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpPut;
import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.client.params.AllClientPNames;
import org.apache.http.client.params.ClientPNames;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.impl.conn.tsccm.ThreadSafeClientConnManager;
import org.apache.http.message.BasicHeader;

/**
* The Session is the main connection to the CouchDB instance.  However, you'll only use the Session
* to obtain a reference to a CouchDB Database.  All of the main work happens at the Database level.
* <p>
* It uses the Apache's  HttpClient library for all communication with the server.  This is
* a little more robust than the standard URLConnection.
* <p>
* Ex usage: <br>
* Session session = new Session(host,port);
* Database db = session.getDatabase("dbname");
* 
* @author mbreese
* @author brennanjubb - HTTP-Auth username/pass
*/
public class CouchSession {
    private static final String DEFAULT_CHARSET = "UTF-8";

    private static final String MIME_TYPE_JSON = "application/json";

    //protected Log log = LogFactory.getLog(CouchSession.class);
    protected final String host;
    protected final int port;
    protected final String user;
    protected final String pass;
    protected final boolean secure;
    protected final boolean usesAuth;

    protected CouchResponse lastResponse;

    protected HttpClient httpClient;
    protected HttpParams httpParams;

    /**
     * Constructor for obtaining a Session with an HTTP-AUTH username/password and (optionally) a secure connection
     * This isn't supported by CouchDB - you need a proxy in front to use this
     * @param host - hostname
     * @param port - port to use
     * @param user - username
     * @param pass - password
     * @param secure  - use an SSL connection?
     */
    public CouchSession(String host, int port, String user, String pass, boolean usesAuth, boolean secure) {
        this.host = host;
        this.port = port;
        this.user = user;
        this.pass = pass;
        this.usesAuth = usesAuth;
        this.secure = secure;

        httpParams = new BasicHttpParams();
        SchemeRegistry schemeRegistry = new SchemeRegistry();

        schemeRegistry.register(new Scheme("http", PlainSocketFactory.getSocketFactory(), 80));
        schemeRegistry.register(new Scheme("https", SSLSocketFactory.getSocketFactory(), 443));

        ThreadSafeClientConnManager connManager = new ThreadSafeClientConnManager(httpParams, schemeRegistry);
        DefaultHttpClient defaultClient = new DefaultHttpClient(connManager, httpParams);
        if (user != null) {
            defaultClient.getCredentialsProvider().setCredentials(AuthScope.ANY,
                    new UsernamePasswordCredentials(user, pass));
        }

        this.httpClient = defaultClient;

        setUserAgent("couchdb4j");
        setSocketTimeout((30 * 1000));
        setConnectionTimeout((15 * 1000));

    }

    /**
     * Constructor for obtaining a Session with an HTTP-AUTH username/password
     * This isn't supported by CouchDB - you need a proxy in front to use this
     * @param host
     * @param port
     * @param user - username
     * @param pass - password
     */
    public CouchSession(String host, int port, String user, String pass) {
        this(host, port, user, pass, false, false);
    }

    /**
     * Main constructor for obtaining a Session.
     * @param host
     * @param port
     */
    public CouchSession(String host, int port) {
        this(host, port, null, null, false, false);
    }

    /**
     * Optional constructor that indicates an HTTPS connection should be used.
     * This isn't supported by CouchDB - you need a proxy in front to use this
     * 
     * @param host
     * @param port
     * @param secure
     */
    public CouchSession(String host, int port, boolean secure) {
        this(host, port, null, null, false, secure);
    }

    /**
     * Read-only
     * @return the host name
     */
    public String getHost() {
        return host;
    }

    /**
     * read-only
     * 
     * @return the port
     */
    public int getPort() {
        return port;
    }

    /**
     * Is this a secured connection (set in constructor)
     * @return
     */
    public boolean isSecure() {
        return secure;
    }

    /**
     * Retrieves a list of all database names from the server
     * @return
     */
    public List<String> getDatabaseNames() {
        CouchResponse resp = get("_all_dbs");
        JSONArray ar = resp.getBodyAsJSONArray();

        List<String> dbs = new ArrayList<String>(ar.length());
        for (int i = 0; i < ar.length(); i++) {
            dbs.add(ar.optString(i));
        }
        return dbs;
    }

    /**
     * Loads a database instance from the server
     * @param name
     * @return the database (or null if it doesn't exist)
     */
    public CouchDatabase getDatabase(String name) {
        CouchResponse resp = get(name);
        if (resp.isOk()) {
            return new CouchDatabase(resp.getBodyAsJSONObject(), this);
        } else {
            //log.warn("Error getting database: "+name);
        }
        return null;
    }

    /**
     * Creates a new database (if the name doesn't already exist)
     * @param name
     * @return the new database (or null if there was an error)
     */
    public CouchDatabase createDatabase(String name) {
        String dbname = name.toLowerCase().replaceAll("[^a-z0-9_$()+\\-/]", "_");
        if (!dbname.endsWith("/")) {
            dbname += "/";
        }
        CouchResponse resp = put(dbname);
        if (resp.isOk()) {
            return getDatabase(dbname);
        } else {
            //log.warn("Error creating database: "+name);
            return null;
        }
    }

    /**
     * Deletes a database (by name) from the CouchDB server.
     * @param name
     * @return true = successful, false = an error occurred (likely the database named didn't exist) 
     */
    public boolean deleteDatabase(String name) {
        return delete(name).isOk();
    }

    /**
     * Deletes a database from the CouchDB server
     * @param db
     * @return was successful
     */
    public boolean deleteDatabase(CouchDatabase db) {
        return deleteDatabase(db.getName());
    }

    /**
     * For a given url (such as /_all_dbs/), build the database connection url
     * @param url
     * @return the absolute URL (hostname/port/etc)
     */
    protected String buildUrl(String url) {
        return ((secure) ? "https" : "http") + "://" + host + ":" + port + "/" + url;
    }

    protected String buildUrl(String url, String queryString) {
        URI uri;
        try {
            uri = new URI((secure ? "https" : "http"), "", host, port, "/" + url,
                    (queryString != null ? queryString : ""), "");
        } catch (URISyntaxException e) {
            return null;
        }
        return uri.toString();
    }

    protected String buildUrl(String url, NameValuePair[] params) {

        url = ((secure) ? "https" : "http") + "://" + host + ":" + port + "/" + url;

        if (params.length > 0) {
            url += "?";
        }
        for (NameValuePair param : params) {
            url += param.getName() + "=" + param.getValue();
        }

        return url;

    }

    /**
     * Package level access to send a DELETE request to the given URL 
     * @param url
     * @return
     */
    CouchResponse delete(String url) {
        HttpDelete del = new HttpDelete(buildUrl(url));
        return http(del);
    }

    /**
     * Send a POST with no body / parameters
     * @param url
     * @return
     */
    CouchResponse post(String url) {
        return post(url, null, null);
    }

    /**
     * Send a POST with body
     * @param url
     * @param content
     * @return
     */
    CouchResponse post(String url, String content) {
        return post(url, content, null);
    }

    /**
     * Send a POST with a body and query string
     * @param url
     * @param content
     * @param queryString
     * @return
     */
    CouchResponse post(String url, String content, String queryString) {
        HttpPost post = new HttpPost(buildUrl(url, queryString));
        if (content != null) {
            HttpEntity entity;
            try {
                entity = new StringEntity(content, DEFAULT_CHARSET);
                post.setEntity(entity);
                post.setHeader(new BasicHeader("Content-Type", MIME_TYPE_JSON));
            } catch (UnsupportedEncodingException e) {
                //log.error(ExceptionUtils.getStackTrace(e));
            }
        }

        return http(post);
    }

    /**
     * Send a POST with a body, query string and specified content type
     * @author rwilson 
     * @param url
     * @param ctype
     * @param content
     * @param queryString
     * @return
     */
    CouchResponse post(String url, String ctype, String content, String queryString) {
        HttpPost post = new HttpPost(buildUrl(url, queryString));
        if (content != null) {
            HttpEntity entity;
            try {
                entity = new StringEntity(content, DEFAULT_CHARSET);
                post.setEntity(entity);
                if (ctype != null) {
                    post.setHeader(new BasicHeader("Content-Type", ctype));
                }
            } catch (UnsupportedEncodingException e) {
                //log.error(ExceptionUtils.getStackTrace(e));
            }
        }

        return http(post);
    }

    /**
     * Send a PUT  (for creating databases)
     * @param url
     * @return
     */
    CouchResponse put(String url) {
        return put(url, null);
    }

    /**
     * Send a PUT with a body (for creating documents)
     * @param url
     * @param content
     * @return
     */
    CouchResponse put(String url, String content) {
        HttpPut put = new HttpPut(buildUrl(url));
        if (content != null) {
            HttpEntity entity;
            try {
                entity = new StringEntity(content, DEFAULT_CHARSET);
                put.setEntity(entity);
                put.setHeader(new BasicHeader("Content-Type", MIME_TYPE_JSON));
            } catch (UnsupportedEncodingException e) {
                //log.error(ExceptionUtils.getStackTrace(e));
            }
        }
        return http(put);
    }

    /**
     * Overloaded Put using by attachments
     */
    CouchResponse put(String url, String ctype, String content) {
        HttpPut put = new HttpPut(buildUrl(url));
        if (content != null) {
            HttpEntity entity;
            try {
                entity = new StringEntity(content, DEFAULT_CHARSET);
                put.setEntity(entity);
                put.setHeader(new BasicHeader("Content-Type", ctype));
            } catch (UnsupportedEncodingException e) {
                //log.error(ExceptionUtils.getStackTrace(e));
            }
        }
        return http(put);
    }

    /**
     * Overloaded Put using by attachments and query string
     * @author rwilson
     * @param url
     * @param ctype
     * @param content
     * @param queryString
     * @return
     */
    CouchResponse put(String url, String ctype, String content, String queryString) {
        HttpPut put = new HttpPut(buildUrl(url, queryString));
        if (content != null) {
            HttpEntity entity;
            try {
                entity = new StringEntity(content, DEFAULT_CHARSET);
                put.setEntity(entity);
                if (ctype != null) {
                    put.setHeader(new BasicHeader("Content-Type", ctype));
                }
            } catch (UnsupportedEncodingException e) {
                //log.error(ExceptionUtils.getStackTrace(e));
            }
        }
        return http(put);
    }

    /**
     * Send a GET request
     * @param url
     * @return
     */
    CouchResponse get(String url) {
        HttpGet get = new HttpGet(buildUrl(url));
        return http(get);
    }

    /**
     * Send a GET request with a number of name/value pairs as a query string
     * @param url
     * @param queryParams
     * @return
     */
    CouchResponse get(String url, NameValuePair[] queryParams) {
        HttpGet get = new HttpGet(buildUrl(url, queryParams));
        return http(get);
    }

    /**
     * Send a GET request with a queryString (?foo=bar)
     * @param url
     * @param queryString
     * @return
     */
    CouchResponse get(String url, String queryString) {
        HttpGet get = new HttpGet(buildUrl(url, queryString));
        return http(get);
    }

    /**
     * Method that actually performs the GET/PUT/POST/DELETE calls.
     * Executes the given HttpMethod on the HttpClient object (one HttpClient per Session).
     * <p>
     * This returns a CouchResponse, which can be used to get the status of the call (isOk), 
     * and any headers / body that was sent back.
     * 
     * @param req
     * @return the CouchResponse (status / error / json document)
     */
    protected CouchResponse http(HttpRequestBase req) {
        HttpResponse httpResponse = null;
        HttpEntity entity = null;

        try {
            if (usesAuth) {
                req.getParams().setBooleanParameter(ClientPNames.HANDLE_AUTHENTICATION, true);
            }
            httpResponse = httpClient.execute(req);

            entity = httpResponse.getEntity();
            lastResponse = new CouchResponse(req, httpResponse);

        } catch (IOException e) {
            android.util.Log.e("CouchSession", "Got exception: " + e.getMessage());
        } finally {
            if (entity != null) {
                try {
                    entity.consumeContent();
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
        return lastResponse;
    }

    /**
     * Returns the last response for this given session
     * - useful for debugging purposes
     * @return
     */
    public CouchResponse getLastResponse() {
        return lastResponse;
    }

    public void setUserAgent(String ua) {
        httpParams.setParameter(AllClientPNames.USER_AGENT, ua);
    }

    public void setConnectionTimeout(int milliseconds) {
        httpParams.setIntParameter(AllClientPNames.CONNECTION_TIMEOUT, milliseconds);
    }

    public void setSocketTimeout(int milliseconds) {
        httpParams.setIntParameter(AllClientPNames.SO_TIMEOUT, milliseconds);
    }

    protected String encodeParameter(String paramValue) {
        try {
            return URLEncoder.encode(paramValue, DEFAULT_CHARSET);
        } catch (UnsupportedEncodingException e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * This method will retrieve a list of replication tasks that are currently running under the couch server this 
     * session is attached to.
     * 
     * @return List of replication tasks running on the server.
     */
    public List<CouchReplicationTask> getReplicationTasks() {
        final List<CouchReplicationTask> replicationTasks = new ArrayList<CouchReplicationTask>();
        CouchResponse resp = get("_active_tasks");
        JSONArray ar = resp.getBodyAsJSONArray();

        for (int i = 0; i < ar.length(); i++) {
            JSONObject task = null;
            try {
                task = ar.getJSONObject(i);
            } catch (JSONException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            try {
                if (CouchReplicationTask.TASK_TYPE.equals(task.getString(CouchTask.TASK_TYPE_KEY))) {
                    final CouchReplicationTask replicationTask = new CouchReplicationTask(
                            task.getString(CouchTask.TASK_TASK_KEY), task.getString(CouchTask.TASK_STATUS_KEY),
                            task.getString(CouchTask.TASK_PID_KEY));

                    if (replicationTask.loadDetailsFromTask() == true) {
                        replicationTasks.add(replicationTask);
                    } else {
                        //log.error("Unable to load replication task details from server response.");
                    }
                } else {
                    //log.trace("Ignoring non-replication task.");
                }
            } catch (JSONException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }

        //log.trace("Found " + replicationTasks.size() + " replication tasks");

        return replicationTasks;
    }

    /**
     * This method will attempt to start the replication task on the couch server instance this session is attached to.
     * 
     * @param task Task to start on the server
     * @return True if the task was accepted by the couch server instance; False otherwise
     */
    public boolean postReplicationTask(final CouchReplicationTask task) {
        final String postUrl = buildUrl("_replicate");

        try {

            //log.trace("Post URL: " + postUrl);

            final JSONObject replicateReq = task.getCreateRequest();

            //log.trace(replicateReq.toString());

            CouchResponse resp = post("_replicate", replicateReq.toString());

            return (resp.getErrorId() == null);
        } catch (Exception e) {
            //log.info("Exception while attempting to post replication task." + e);

        }
        return false;
    }
}