/* 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 com.fourspaces.couchdb; import java.io.IOException; import com.fourspaces.couchdb.util.JSONUtils; import static com.fourspaces.couchdb.util.JSONUtils.urlEncodePath; import net.sf.json.*; import net.sf.json.util.JSONStringer; import org.apache.commons.lang.StringUtils; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; /** * This represents a particular database on the CouchDB server * <p/> * Using this object, you can get/create/update/delete documents. * You can also call views (named and adhoc) to query the underlying database. * * @author mbreese */ public class Database { Log log = LogFactory.getLog(Database.class); private final String name; private int documentCount; private int updateSeq; private Session session; private static final String VIEW = "/_view/"; private static final String DESIGN = "_design/"; /** * C-tor only used by the Session object. You'd never call this directly. * * @param json * @param session */ Database(JSONObject json, Session session) { name = json.getString("db_name"); documentCount = json.getInt("doc_count"); updateSeq = json.getInt("update_seq"); this.session = session; } /** * The name of the database * * @return */ public String getName() { return name; } /** * The number of documents in the database <b>at the time that it was retrieved from the session</b> * This number probably isn't accurate after the initial load... so if you want an accurate * assessment, call Session.getDatabase() again to reload a new database object. * * @return */ public int getDocumentCount() { return documentCount; } /** * The update seq from the initial database load. The update sequence is the 'revision id' of an entire database. Useful for getting all documents in a database since a certain revision * * @return * @see getAllDocuments() */ public int getUpdateSeq() { return updateSeq; } /** * Runs the standard "_all_docs" view on this database * * @return ViewResults - the results of the view... this can be iterated over to get each document. */ public ViewResults getAllDocuments() { return view(new View("_all_docs"), false); } /** * Gets all design documents * * @return ViewResults - all design docs */ public ViewResults getAllDesignDocuments() { View v = new View("_all_docs"); v.startKey = "%22_design%2F%22"; v.endKey = "%22_design0%22"; v.includeDocs = Boolean.TRUE; return view(v, false); } /** * Runs the standard "_all_docs" view on this database, with count * * @return ViewResults - the results of the view... this can be iterated over to get each document. */ public ViewResults getAllDocumentsWithCount(int count) { View v = new View("_all_docs"); v.setCount(count); return view(v, false); } /** * Runs "_all_docs_by_update_seq?startkey=revision" view on this database * * @return ViewResults - the results of the view... this can be iterated over to get each document. */ public ViewResults getAllDocuments(int revision) { return view(new View("_all_docs_by_seq?startkey=" + revision), false); } /** * Runs a named view on the database * This will run a view and apply any filtering that is requested (reverse, startkey, etc). * * @param view * @return */ public ViewResults view(View view) { return view(view, true); } /** * Runs a view, appending "_view" to the request if isPermanentView is true. * * * * @param view * @param isPermanentView * @return */ private ViewResults view(final View view, final boolean isPermanentView) { String url = null; if (isPermanentView) { String[] elements = view.getFullName().split("/"); url = this.name + "/" + ((elements.length < 2) ? elements[0] : DESIGN + elements[0] + VIEW + elements[1]); } else { url = this.name + "/" + view.getFullName(); } CouchResponse resp = session.get(url, view.getQueryString()); if (resp.isOk()) { ViewResults results = new ViewResults(view, resp.getBodyAsJSONObject()); results.setDatabase(this); return results; } return null; } /** * Runs a named view <i>Not currently working in CouchDB code</i> * * @param fullname - the fullname (including the document name) ex: foodoc:viewname * @return */ public ViewResults view(String fullname) { return view(new View(fullname), true); } /** * Runs an ad-hoc view from a string * * @param function - the Javascript function to use as the filter. * @return results */ public ViewResults adhoc(String function) { return adhoc(new AdHocView(function)); } /** * Runs an ad-hoc view from an AdHocView object. You probably won't use this much, unless * you want to add filtering to the view (reverse, startkey, etc...) * * @param view * @return */ public ViewResults adhoc(final AdHocView view) { String adHocBody = new JSONStringer() .object() .key("map").value(JSONUtils.stringSerializedFunction(view.getFunction())) .endObject() .toString(); // Bugfix - include query string for adhoc views to support // additional view options (setLimit, etc) CouchResponse resp = session.post(name + "/_temp_view", adHocBody, view.getQueryString()); if (resp.isOk()) { ViewResults results = new ViewResults(view, resp.getBodyAsJSONObject()); results.setDatabase(this); return results; } else { log.warn("Error executing view - " + resp.getErrorId() + " " + resp.getErrorReason()); } return null; } /** * Save a document at the given _id * <p/> * if the docId is null or empty, then this performs a POST to the database and retrieves a new * _id. * <p/> * Otherwise, a PUT is called. * <p/> * Either way, a new _id and _rev are retrieved and updated in the Document object * * @param doc * @param docId */ public void saveDocument(Document doc, String docId) throws IOException { CouchResponse resp; if (docId == null || docId.equals("")) { resp = session.post(name, doc.getJSONObject().toString()); } else { resp = session.put(name + "/" + urlEncodePath(docId), doc.getJSONObject().toString()); } if (resp.isOk()) { try { if (doc.getId() == null || doc.getId().equals("")) { doc.setId(resp.getBodyAsJSONObject().getString("id")); } doc.setRev(resp.getBodyAsJSONObject().getString("rev")); } catch (JSONException e) { e.printStackTrace(); } doc.setDatabase(this); } else { log.warn("Error adding document - " + resp.getErrorId() + " " + resp.getErrorReason()); System.err.println("RESP: " + resp); } } /** * Save a document w/o specifying an id (can be null) * * @param doc */ public void saveDocument(Document doc) throws IOException { saveDocument(doc, doc.getId()); } public void bulkSaveDocuments(Document[] documents) throws IOException { CouchResponse resp = null; resp = session.post(name + "/_bulk_docs", new JSONObject().accumulate("docs", documents).toString()); if (resp.isOk()) { // TODO set Ids and revs and name (db) final JSONArray respJsonArray = resp.getBodyAsJSONArray(); JSONObject respObj = null; String id = null; String rev = null; for (int i = 0; i < documents.length; i++) { respObj = respJsonArray.getJSONObject(i); id = respObj.getString("id"); rev = respObj.getString("rev"); if (StringUtils.isBlank(documents[i].getId())) { documents[i].setId(id); documents[i].setRev(rev); } else if (StringUtils.isNotBlank(documents[i].getId()) && documents[i].getId().equals(id)) { documents[i].setRev(rev); } else { log.warn("returned bulk save array in incorrect order, saved documents do not have updated rev or ids"); } documents[i].setDatabase(this); } } else { log.warn("Error bulk saving documents - " + resp.getErrorId() + " " + resp.getErrorReason()); } } /** * Retrieves a document from the CouchDB database * * @param id * @return */ public Document getDocument(String id) throws IOException { return getDocument(id, null, false); } /** * Retrieves a document from the database and asks for a list of it's revisions. * The list of revision keys can be retrieved from Document.getRevisions(); * * @param id * @return */ public Document getDocumentWithRevisions(String id) throws IOException { return getDocument(id, null, true); } /** * Retrieves a specific document revision * * @param id * @param revision * @return */ public Document getDocument(String id, String revision) throws IOException { return getDocument(id, revision, false); } /** * Retrieves a specific document revision and (optionally) asks for a list of all revisions * * @param id * @param revision * @param showRevisions * @return the document */ public Document getDocument(String id, String revision, boolean showRevisions) throws IOException { CouchResponse resp; Document doc = null; if (revision != null && showRevisions) { resp = session.get(name + "/" + urlEncodePath(id), "rev=" + revision + "&full=true"); } else if (revision != null && !showRevisions) { resp = session.get(name + "/" + urlEncodePath(id), "rev=" + revision); } else if (revision == null && showRevisions) { resp = session.get(name + "/" + urlEncodePath(id), "revs=true"); } else { resp = session.get(name + "/" + urlEncodePath(id)); } if (resp.isOk()) { doc = new Document(resp.getBodyAsJSONObject()); doc.setDatabase(this); } else { log.warn("Error getting document - " + resp.getErrorId() + " " + resp.getErrorReason()); } return doc; } /** * Deletes a document * * @param d * @return was the delete successful? * @throws IllegalArgumentException for blank document id */ public boolean deleteDocument(Document d) throws IOException { if (StringUtils.isBlank(d.getId())) { throw new IllegalArgumentException("cannot delete document, doc id is empty"); } CouchResponse resp = session.delete(name + "/" + urlEncodePath(d.getId()) + "?rev=" + d.getRev()); if (resp.isOk()) { return true; } else { log.warn("Error deleting document - "+resp.getErrorId()+" "+resp.getErrorReason()); return false; } } /** * Gets attachment * * @param id * @param attachment attachment body * @return attachment body */ public String getAttachment(String id, String attachment) throws IOException { CouchResponse resp = session.get(name + "/" + urlEncodePath(id) + "/" + attachment); return resp.getBody(); } /** * Puts attachment to a doc * * @param id document id * @param fname attachment name * @param ctype content type * @param attachment attachment body * @return was the put successful? */ public Document putAttachment(String id, String rev, String fname, String ctype, String attachment) throws IOException { CouchResponse resp = session.put(name + "/" + urlEncodePath(id) + "/" + fname + "?rev=" + urlEncodePath(rev), ctype, attachment); return new Document(resp.getBodyAsJSONObject()); } }