All Downloads are FREE. Search and download functionalities are using the official Maven repository.

main.java.com.cloudant.client.api.Changes Maven / Gradle / Ivy

/*
 * Copyright (C) 2011 lightcouch.org
 * Copyright © 2015, 2017 IBM Corp. All rights reserved.
 *
 * 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.cloudant.client.api;


import com.cloudant.client.api.model.ChangesResult;
import com.cloudant.client.api.model.ChangesResult.Row;
import com.cloudant.client.internal.DatabaseURIHelper;
import com.cloudant.client.org.lightcouch.CouchDbClient;
import com.cloudant.client.org.lightcouch.CouchDbException;
import com.cloudant.client.org.lightcouch.internal.CouchDbUtil;
import com.google.gson.Gson;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.UnsupportedEncodingException;
import java.net.URI;

/**
 * 

Contains the Change Notifications API, supports normal and continuous feed * Changes. *

Usage example for normal feed limiting to 10 results, with a filter:

*
 * {@code
 * // feed type normal
 * String since = db.info().getUpdateSeq(); // latest update seq
 * ChangesResult changeResult = db.changes()
 * 	.since(since)
 * 	.limit(10)
 * 	.filter("example/filter")
 * 	.getChanges();
 *
 * //process the ChangesResult
 * for (ChangesResult.Row row : changeResult.getResults()) {
 *   String docId = row.getId()
 *   JsonObject doc = row.getDoc();
 * }
 * }
 * 
*

Usage example for continuous feed, including document content:

*
 * {@code
 * // feed type continuous
 * Changes changes = db.changes()
 * 	.includeDocs(true)
 * 	.heartBeat(30000)
 * 	.continuousChanges();
 *
 * while (changes.hasNext()) {
 * 	ChangesResult.Row feed = changes.next();
 *  String docId = feed.getId();
 *  JsonObject doc = feed.getDoc();
 * }
 *
 * //while loop blocks; stop from another thread
 * changes.stop(); // stop continuous feed
 * }
 * 
* * @author Ganesh K Choudhary * @see ChangesResult * @see Databases - get changes * @since 0.0.1 */ public class Changes { private final CouchDbClient client; private final Gson gson; private final DatabaseURIHelper databaseHelper; private BufferedReader reader; private ChangesResult.Row nextRow; private boolean stop; Changes(CloudantClient client, Database database) { this.client = client.couchDbClient; this.gson = database.getGson(); this.databaseHelper = new DatabaseURIHelper(database.getDBUri()); } /** * Requests Change notifications of feed type continuous. *

Feed notifications are accessed in an iterator style.

*

* This method will connect to the changes feed; any configuration options applied after calling * it will be ignored. *

*

* Note that in some versions of Apache CouchDB and Cloudant the HTTP response headers will be * delayed until the first change or heartbeat is sent * (see https://github.com/apache/couchdb/issues/985). As such this method may block for up to * the heartbeat duration if there are no changes to be received from the database immediately. *

* @return this Changes instance connected to a continuous feed, use * {@link #hasNext()} and {@link #next()} to iterate the changes. */ public Changes continuousChanges() { final URI uri = this.databaseHelper.changesUri("feed", "continuous"); final InputStream in = client.get(uri); try { final InputStreamReader is = new InputStreamReader(in, "UTF-8"); setReader(new BufferedReader(is)); } catch (UnsupportedEncodingException e) { // This should never happen as every implementation of the java platform is required // to support UTF-8. throw new RuntimeException(e); } return this; } /** * Checks whether a feed is available in the continuous stream, blocking * until a feed is received. * * @return true if there is a change */ public boolean hasNext() { return readNextRow(); } /** * @return The next feed in the stream. */ public Row next() { return getNextRow(); } /** * Stops a running continuous feed. */ public void stop() { stop = true; } /** * Requests Change notifications of feed type normal. * * @return {@link ChangesResult} encapsulating the normal feed changes */ public ChangesResult getChanges() { final URI uri = this.databaseHelper.changesUri("feed", "normal"); return client.get(uri, ChangesResult.class); } // Query Params /** * Return only changes after the specified sequence identifier. * * @param since sequence identifier or {@code "now"} * @return this Changes instance */ public Changes since(String since) { this.databaseHelper.query("since", since); return this; } /** * Limit the number of rows to return. * * @param limit the number of rows * @return this Changes instance */ public Changes limit(int limit) { this.databaseHelper.query("limit", limit); return this; } /** * Enable an empty line heartbeat for longpoll or continuous feeds for when there have been no * changes. * * @param heartBeat time in milliseconds after which an empty line is sent * @return this Changes instance */ public Changes heartBeat(long heartBeat) { this.databaseHelper.query("heartbeat", heartBeat); return this; } /** * Configure a timeout for the changes feed. * * @param timeout time in milliseconds to wait for data * @return this Changes instance */ public Changes timeout(long timeout) { this.databaseHelper.query("timeout", timeout); return this; } /** * Specify a filter function to apply to the changes feed. * * @param filter name of the design document filter function e.g {@code * "designDoc/filterFunction"} * @return this Changes instance */ public Changes filter(String filter) { this.databaseHelper.query("filter", filter); return this; } /** * @param includeDocs whether to include document content in the returned rows * @return this Changes instance */ public Changes includeDocs(boolean includeDocs) { this.databaseHelper.query("include_docs", includeDocs); return this; } /** * Configures how many changes are returned "main_only" for the winning revision only or * "all_docs" to also include leaf revisions. * * @param style {@code "main_only"} or {@code "all_docs"} * @return this Changes instance */ public Changes style(String style) { this.databaseHelper.query("style", style); return this; } /** * @param descending {@code true} to return changes in descending order * @return this Changes instance * @since 2.5.0 */ public Changes descending(boolean descending) { this.databaseHelper.query("descending", descending); return this; } /** * Add a custom query parameter to the _changes request. Useful for specifying extra parameters * to a filter function for example. * * @param name the name of the query parameter * @param value the value of the query parameter * @return this Changes instance * @since 2.5.0 */ public Changes parameter(String name, String value) { this.databaseHelper.query(name, value); return this; } // Helper /** * Reads and sets the next feed in the stream. */ private boolean readNextRow() { while (!stop) { String row = getLineWrapped(); // end of stream - null indicates end of stream before we see last_seq which shouldn't // be possible but we should handle it if (row == null || row.startsWith("{\"last_seq\":")) { terminate(); return false; } else if (row.isEmpty()) { // heartbeat continue; } setNextRow(gson.fromJson(row, ChangesResult.Row.class)); return true; } // we were stopped, end of changes feed terminate(); return false; } private String getLineWrapped() { try { return getReader().readLine(); } catch (IOException ioe) { terminate(); throw new CouchDbException("Error reading continuous stream.", ioe); } } private BufferedReader getReader() { return reader; } private void setReader(BufferedReader reader) { this.reader = reader; } private ChangesResult.Row getNextRow() { return nextRow; } private void setNextRow(ChangesResult.Row nextRow) { this.nextRow = nextRow; } private void terminate() { CouchDbUtil.close(getReader()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy