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

mil.nga.geopackage.db.Pagination Maven / Gradle / Ivy

package mil.nga.geopackage.db;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

import mil.nga.geopackage.GeoPackageException;

/**
 * Query pagination configuration
 * 
 * @author osbornb
 * @since 6.2.0
 */
public class Pagination {

	/**
	 * Limit SQL statement
	 */
	private static final String LIMIT = "LIMIT";

	/**
	 * Offset SQL statement
	 */
	private static final String OFFSET = "OFFSET";

	/**
	 * Expression 1 pattern group
	 */
	private static final String EXPRESSION1 = "expr1";

	/**
	 * Expression Separator pattern group
	 */
	private static final String SEPARATOR = "separator";

	/**
	 * Expression 1 pattern group
	 */
	private static final String EXPRESSION2 = "expr2";

	/**
	 * Limit pattern
	 */
	private static final Pattern limitPattern = Pattern
			.compile(
					"\\s" + LIMIT + "\\s+(?<" + EXPRESSION1
							+ ">-?+\\d+)(\\s*(?<" + SEPARATOR + ">(" + OFFSET
							+ "|,))\\s*(?<" + EXPRESSION2 + ">-?\\d+))?",
					Pattern.CASE_INSENSITIVE);

	/**
	 * Find the pagination offset and limit from the SQL statement
	 * 
	 * @param sql
	 *            SQL statement
	 * @return pagination or null if not found
	 */
	public static Pagination find(String sql) {
		Pagination pagination = null;
		Matcher matcher = limitPattern.matcher(sql);
		if (matcher.find()) {
			int limit;
			Long offset = null;
			String expr1 = matcher.group(EXPRESSION1);
			String separator = matcher.group(SEPARATOR);
			if (separator != null) {
				String expr2 = matcher.group(EXPRESSION2);
				if (separator.equalsIgnoreCase(OFFSET)) {
					limit = Integer.valueOf(expr1);
					offset = Long.valueOf(expr2);
				} else {
					limit = Integer.valueOf(expr2);
					offset = Long.valueOf(expr1);
				}
			} else {
				limit = Integer.valueOf(expr1);
			}
			pagination = new Pagination(limit, offset);
		}
		return pagination;
	}

	/**
	 * Replace the pagination limit and offset in the SQL statement
	 * 
	 * @param pagination
	 *            pagination settings
	 * @param sql
	 *            SQL statement
	 * @return modified SQL statement
	 */
	public static String replace(Pagination pagination, String sql) {
		String replaced = null;
		Matcher matcher = limitPattern.matcher(sql);
		if (matcher.find()) {
			replaced = matcher.replaceFirst(" " + pagination.toString());
		} else {
			throw new GeoPackageException(
					"SQL statement is not a paginated query: " + sql);
		}
		return replaced;
	}

	/**
	 * Limit
	 */
	private int limit;

	/**
	 * Offset
	 */
	private Long offset;

	/**
	 * Constructor
	 * 
	 * @param limit
	 *            upper bound number of rows
	 */
	public Pagination(int limit) {
		this(limit, null);
	}

	/**
	 * Constructor
	 * 
	 * @param limit
	 *            upper bound number of rows
	 * @param offset
	 *            row result starting offset
	 */
	public Pagination(int limit, long offset) {
		this(limit, Long.valueOf(offset));
	}

	/**
	 * Constructor
	 * 
	 * @param limit
	 *            upper bound number of rows
	 * @param offset
	 *            row result starting offset
	 */
	public Pagination(int limit, Long offset) {
		setLimit(limit);
		setOffset(offset);
	}

	/**
	 * Get the limit
	 * 
	 * @return upper bound number of rows
	 */
	public int getLimit() {
		return limit;
	}

	/**
	 * Set the limit
	 * 
	 * @param limit
	 *            upper bound number of rows
	 */
	public void setLimit(int limit) {
		this.limit = limit;
	}

	/**
	 * Is there positive limit
	 * 
	 * @return true if limit above 0
	 */
	public boolean hasLimit() {
		return limit > 0;
	}

	/**
	 * Get the offset
	 * 
	 * @return row result starting offset
	 */
	public Long getOffset() {
		return offset;
	}

	/**
	 * Is there an offset
	 * 
	 * @return true if has an offset
	 */
	public boolean hasOffset() {
		return offset != null;
	}

	/**
	 * Set the offset
	 * 
	 * @param offset
	 *            row result starting offset
	 */
	public void setOffset(Long offset) {
		if (offset != null && offset < 0) {
			offset = 0L;
		}
		this.offset = offset;
	}

	/**
	 * If the limit is positive, increment the offset
	 */
	public void incrementOffset() {
		if (limit > 0) {
			incrementOffset(limit);
		}
	}

	/**
	 * Increment the offset by the count
	 * 
	 * @param count
	 *            count to increment
	 */
	public void incrementOffset(long count) {
		if (this.offset == null) {
			this.offset = 0L;
		}
		this.offset += count;
	}

	/**
	 * Replace the limit and offset in the SQL statement with the pagination
	 * values
	 * 
	 * @param sql
	 *            SQL statement
	 * @return modified SQL statement
	 */
	public String replace(String sql) {
		return replace(this, sql);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public String toString() {
		StringBuilder sql = new StringBuilder();
		sql.append(LIMIT);
		sql.append(" ");
		sql.append(limit);
		if (hasOffset()) {
			sql.append(" ");
			sql.append(OFFSET);
			sql.append(" ");
			sql.append(offset);
		}
		return sql.toString();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy