Water: The Truth
*It's good for you!
* Next Page *
/*
* Copyright (c) 2013 Univocity Software Pty Ltd. All rights reserved.
* This file is subject to the terms and conditions defined in file
* 'LICENSE.txt', which is part of this source code package.
*/
package com.univocity.api.entity.html;
import com.univocity.api.*;
import com.univocity.api.common.*;
import com.univocity.api.entity.html.builders.*;
import com.univocity.api.net.*;
import com.univocity.parsers.remote.*;
import java.util.*;
/**
* Used by the {@link HtmlParser} to collect multiple pages of results in a website and to handle
* the files that have been downloaded for each page.
*
* @author Univocity Software Pty Ltd - [email protected]
* @see HtmlParser
* @see PaginationContext
* @see NextInputHandler
*/
public final class HtmlPaginator extends Paginator {
private Set requestParameters;
final Map fieldParameters = new HashMap();
final Map parameterData = new HashMap();
/**
* Creates a new `HtmlPaginator` and sets the `currentPageNumber` to 0
*
* @param parserSettings the parser settings to use
*/
protected HtmlPaginator(HtmlParserSettings parserSettings) {
super(parserSettings);
}
/**
* Creates a new {@link HtmlEntitySettings} which will be used to create fields specifically for this
* `HtmlPaginator`.
*
* @return the created {@link HtmlEntitySettings} to be used by this `HtmlPaginator`.
*/
@Override
protected final HtmlEntitySettings newEntitySettings(final RemoteParserSettings htmlParserSettings) {
return new HtmlEntitySettings(ENTITY_NAME, null) {
{
HtmlPaginator.this.requestParameters = this.requestParameters;
this.parserSettings = (HtmlParserSettings) htmlParserSettings;
}
};
}
/**
* Returns the {@link HtmlEntitySettings} that is associated with this `HtmlPaginator`.
*
* @return returns the paginator {@link HtmlEntitySettings}
*/
final HtmlEntitySettings getSettings() {
return entitySettings;
}
/**
* Creates a new field for the current page and returns a {@link PathStart} which can be used to define the path
* to the 'current page' element. The current page is a HTML element that indicates which page among a series of pages
* is being currently parsed.
*
* @return a {@link PathStart} used to define the path to the current page element
*/
public final PathStart setCurrentPage() {
return entitySettings.addField(CURRENT_PAGE);
}
/**
* Creates a new field for the current page and returns a {@link PathStart} which can be used to define the path
* to the 'current page' element as a number. The current page is a HTML element that indicates which page among
* a series of pages is being currently parsed.
*
* @return a {@link PathStart} used to define the path to the current page number
*/
public final PathStart setCurrentPageNumber() {
return entitySettings.addField(CURRENT_PAGE_NUMBER);
}
/**
* Creates a new field for the next page number and returns a {@link PathStart} which can be used to define the path
* to the next page number element. The next page number indicates that there are more pages after the current page.
* When the parser runs and completes the parsing of the page, it will read the next page number to decide whether
* to proceed a try to obtain the next page to parse. The parser will continue to access the next page until the
* next page number does not exist or the follow count set by {@link Paginator#setFollowCount(int)} is reached.
*
* @return a {@link PathStart} used to define the path to the next page number
*/
public final PathStart setNextPageNumber() {
return entitySettings.addField(NEXT_PAGE_NUMBER);
}
/**
* Creates a new field for the next page and returns a {@link PathStart} which can be used to define the path
* to the next page element. The next page is a HTML element that changes the current page to the next page in series.
* When the parser runs and completes the parsing of the page, it will 'click' on the next page element
* and process the result. The parser will continue to access the next page until a next page element does not
* exist or the follow count set by {@link HtmlPaginator#setFollowCount(int)}
* is reached.
*
* An example of setting the next page can be demonstrated using this HTML:
*
* ```html
*
*
*
* Water: The Truth
* It's good for you!
* Next Page
*
*
*
* ```
*
* Assume that the `paginationTarget.html` linked above contains the following HTML:
*
* ```html
*
*
*
* Bananas
* An excellent source of potassium
*
*
*
* ```
*
* You can get the text in the `h1` and `p` elements from both pages with:
*
* ```
* HtmlEntityList entities = new HtmlEntityList();
* HtmlEntitySettings entity = entities.configureEntity("entity");
*
* entity.addField("header").match("h1").containedBy("article").getText();
* entity.addField("text").match("p").containedBy("article").getText();
*
* entities.configurePaginator()
* .setNextPage()
* .match("a")
* .containedBy("article")
* .getAttribute("href");
* ```
*
* When the parser runs, it will parse the first page, collecting the first row of data:
*
* ```
* [Water: The Truth, It's good for you!]
* ```
*
* The paginator will then run, accessing the link's URL provided by the `href` attribute and opening the next page.
* This next page will then be parsed to collect the next row:
*
* ```
* [Bananas, An excellent source of potassium]
* ```
*
* As there is no `a` element on the second page with a link to the next, the paginator will be unable to run and
* the parsing will finish, returning the two records parsed from each page.
*
* @return a {@link PathStart} is used to define the path to the next page element
*/
public final PathStart setNextPage() {
return entitySettings.addField(NEXT_PAGE);
}
/**
* Creates a new field on this `HtmlPaginator` and returns a {@link PathStart} that allows the user to define
* a path to the field. Use method {@link #toRequestParameter} of this class to send the field name and its value
* in the body of the POST request to the next page.
*
* Any fields added to the paginator and their values can be read from a {@link NextInputHandler} through
* its {@link PaginationContext}.
*
* @param fieldName name of the new field
*
* @return a {@link PathStart} is used to define the path of the value for the given field
*/
public final PathStart addField(String fieldName) {
return entitySettings.addField(fieldName);
}
/**
* Creates a new request parameter and returns a {@link PathStart} that allows the user to define path to the
* parameter. Request parameters are values set on a page, usually internally in hidden fields of a form element,
* and which contain details about the page state. These are usually used to be sent back to the server in a POST
* {@link HttpRequest}, to validate the current session and return a next page of results.
*
* If you assign the path to the request parameters you are interested in, the paginator will automatically
* set them for you when requesting for the next page of results. Otherwise, you'd have to manually set the parameters
* using a {@link NextInputHandler}.
*
* @param paramName name of the request parameter
*
* @return a {@link PathStart} is used to define the path of the value for the given parameter
*/
public final PathStart addRequestParameter(String paramName) {
Args.notBlank(paramName, "Request parameter name");
requestParameters.add(paramName);
return entitySettings.addField(paramName);
}
/**
* Associates a constant value to a request parameter. Parameter values are submitted in POST requests to load
* the next page.
*
* @param paramName the name that will be associated with the parameter, which will be sent in the request
* for the next page.
* @param value the value of the corresponding parameter
*/
public final void setRequestParameter(String paramName, String value) {
requestParameters.add(paramName);
entitySettings.addField(paramName, value);
}
/**
* Assigns values captured for two fields declared in this `HtmlPaginator` to a request parameter. For
* example, if fields "scriptName" and "scriptValue" have been defined in this paginator, and and their values are
* collected as "thescript" and "myscript.js" respectively, the {@link HttpRequest} used to invoke the next page
* will have its {@link HttpRequest#addDataParameter(String, Object)} invoked with "thescript" as the parameter name
* and "myscript.js" as the parameter value.
*
* @param fieldForParamName name of the field available in this paginator whose value will be used as a the request
* parameter name.
* @param fieldForParamValue name of the field available in this paginator whose value will be used as the request
* parameter value.
*/
public final void toRequestParameter(String fieldForParamName, String fieldForParamValue) {
Args.notBlank(fieldForParamName, "Field to use as parameter name");
Args.notBlank(fieldForParamValue, "Field to use as parameter value");
this.fieldParameters.put(fieldForParamName, fieldForParamValue);
}
/**
* Defines a request parameter name and data value to be used when requesting the next page.
*
* @param fieldForParamName name of the request parameter to add to this paginator
* @param value the value associated with the given request parameter
*/
public final void setRequestParameterData(String fieldForParamName, Object value) {
Args.notBlank(fieldForParamName, "Field to use as parameter name");
Args.notNull(value, "Constant to be associated with " + fieldForParamName);
this.parameterData.put(fieldForParamName, value);
}
/**
* Returns a {@link PartialPathStart} that is used to define a reusable path of HTML elements. Fields then can
* added to this path using {@link PartialPath#addField(String)} and others, which associates the field with this entity.
*
* Example:
*
* ```java
* HtmlEntityList entityList = new HtmlEntityList();
* HtmlEntitySettings items = entityList.configureEntity("items");
* PartialPath path = items.newPath()
* .match("table").id("productsTable")
* .match("td").match("div").classes("productContainer");
*
* //uses the path to add new fields to it and further element matching rules from the initial, common path.
* path.addField("name").match("span").classes("prodName", "prodNameTro").getText();
* path.addField("URL").match("a").childOf("div").classes("productPadding").getAttribute("href")
* ```
*
* @return a {@link PartialPathStart} to specify the path of HTML elements
*/
public final PaginationPathStart newPath() {
return Builder.build(PaginationPathStart.class, entitySettings);
}
/**
* Creates a new {@link PaginationGroup} group for this paginator. Refer to the {@link Group} documentation to learn
* more about how element groups are used.
*
* @return a {@link PaginationGroupStart} which is the first step in determining which element demarcates the start
* of a {@link PaginationGroup}.
*/
public final PaginationGroupStart newGroup() {
return entitySettings.newPaginationGroup(this);
}
}