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

com.couchbase.client.protocol.views.Paginator Maven / Gradle / Ivy

/**
 * Copyright (C) 2009-2013 Couchbase, Inc.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALING
 * IN THE SOFTWARE.
 */

package com.couchbase.client.protocol.views;

import com.couchbase.client.CouchbaseClient;

import java.util.Iterator;

/**
 * The {@link Paginator} makes it possible to iterate over a
 * {@link ViewResponse} in pages.
 *
 * 

It is possible to iterate over both reduced and non-reduced results, but * iterating over non-reduced results is considerably faster, because a * more efficient pagination approach can be used.

* *

Usage:

*
{@code
 *
 *View view = client.getView("design_doc", "view_name");
 *Query query = new Query();
 *int docsPerPage = 20;
 *
 *Paginator paginator = client.paginatedQuery(view, query, docsPerPage);
 *while(paginator.hasNext()) {
 *  ViewResponse response = paginator.next();
 *  for(ViewRow row : response) {
 *    System.out.println(row.getKey());
 *  }
 *}
 * }
* *

Note that if a custom limit is set on the {@link Query} object when it * gets passed in into the {@link Paginator}, then it is considered as an * absolute limit. The {@link Paginator} will stop iterating once this absolute * limit is reached. If no limit is provided, the {@link Paginator} will move * forward until no more documents are returned by the view.

* *

If you encounter an infinite loop when emitting stringified numbers from * your View, see the {@link Paginator#forcedKeyType} method for instructions * to remedy this situation.

*/ public class Paginator implements Iterator { private final CouchbaseClient client; private final View view; private final Query query; private final int limit; /** * Contains the current state of the Paginator. */ private volatile State currentState; /** * Holds the next response that will be returned on {@link #next()}. */ private ViewResponse nextResponse = null; /** * A counter indicating the current page (used to for skipping). */ private int currentPage; /** * The next ID to start when paging with non-reduced views. */ private String nextStartKeyDocID = null; /** * The next key to start when paging with non-reduced views. */ private String nextStartKey = null; /** * Helps to prevent errors when {@link #hasNext()} is called twice or more * before actually calling {@link #next()}. */ private boolean alreadyCalled; /** * Used to make sure a total limit on the view result can still be * paginated correctly. */ private int totalLimit; /** * Defines into which the key will be casted into. */ private Class forcedKeyType = null; /** * Create a new Paginator by passing in the needed params. * * @param client the client object to work against. * @param view the corresponding view to query. * @param query the query object to customize the pages. * @param limit the amount of docs to return per page. */ public Paginator(final CouchbaseClient client, final View view, final Query query, final int limit) { if (limit <= 0) { throw new IllegalArgumentException("Number of documents per page " + "must be greater than zero."); } this.client = client; this.view = view; this.query = query.copy(); this.limit = limit; if (this.query.getLimit() > 0) { this.totalLimit = this.query.getLimit(); } else { this.totalLimit = -1; } this.query.setLimit(limit + 1); this.currentState = State.INITIALIZED; this.currentPage = 1; this.alreadyCalled = false; } /** * Check if another Page is available. * * @return true if a page is available, false otherwise. */ public final boolean hasNext() { if (currentState == State.FINISHED) { return false; } if (alreadyCalled) { return true; } else { alreadyCalled = true; } fetchNextPage(); if (currentState == State.INITIALIZED) { currentState = State.PAGING; } return true; } /** * Fetch the next page. * * Depending on if reduce is used or not, it uses a different approach on * how to handle paging. */ private void fetchNextPage() { if (currentState == State.PAGING) { if (query.willReduce()) { query.setSkip(limit * (currentPage - 1)); } else { query.setStartkeyDocID(nextStartKeyDocID); query.setRangeStart(convertKey(nextStartKey)); } } if (totalLimit > 0 && (currentPage * limit) >= totalLimit) { int reduceBy = (currentPage * limit) - totalLimit; query.setLimit(limit - reduceBy); } nextResponse = client.query(view, query); if (nextResponse.size() == limit + 1) { ViewRow nextRow = nextResponse.removeLastElement(); if (!query.willReduce()) { nextStartKeyDocID = nextRow.getId(); nextStartKey = nextRow.getKey(); } } else { currentState = State.FINISHED; } currentPage++; } /** * Returns the next {@link ViewResponse}. * * @return returns a {@link ViewResponse} which represents the next page * or null if there is none (check with {@link #hasNext()} first. */ public final ViewResponse next() { alreadyCalled = false; if (currentState == State.INITIALIZED) { return null; } return nextResponse; } /** * Allows one to override the type of the row key. * *

This should only be used to enforce a different type if absolutely * needed, especially if you emit a number as a string from the view. If * nothing else is specified, the data will be passed in 1:1 which may lead * to infinite loops on stringified numbers. To remedy this situation, * forcing to stringify the number again like this helps:

* *
{@code
   *paginatedQuery.forceKeyType(String.class);
   *}
* *

Setting it to Integer.class will force a conversion to integer * the other way round. Note that if the class type is not recognized, * the original value will be passed straight through as a String towards * the {@link ComplexKey} class.

* *

This is ignored on reduced and spatial views, because a different * strategy (skip) is used there.

* * @param clazz the enforced key type. */ public void forceKeyType(Class clazz) { this.forcedKeyType = clazz; } /** * Converts the paginator key to the intended type. * * @param original original Key from the View as string * @return the modified key. */ private String convertKey(String original) { if(forcedKeyType == null) { return original; } if(forcedKeyType.getSimpleName().equals("Integer")) { return ComplexKey.of(Integer.parseInt(original)).toJson(); } return ComplexKey.of(original).toJson(); } /** * The {@link #remove()} method is not supported in this context. */ public final void remove() { throw new UnsupportedOperationException("Remove is unsupported"); } /** * Defines the States in which the Paginator is in at any given time. */ enum State { /** No Page has yet been fetched, it has just been initialized. */ INITIALIZED, /** Currently in the process of paging, more pages available. */ PAGING, /** Last page reached and/or finished already. */ FINISHED } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy