com.hp.autonomy.aci.content.fieldtext.FieldTextBuilder Maven / Gradle / Ivy
/*
* Copyright 2009-2015 Hewlett-Packard Development Company, L.P.
* Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License.
*/
package com.hp.autonomy.aci.content.fieldtext;
import org.apache.commons.lang.Validate;
import static com.hp.autonomy.aci.content.fieldtext.FieldTexts.MATCHNOTHING;
/**
* {@code FieldTextBuilder} allows for the building of complex fieldtext expressions from other fieldtext expressions
* or specifiers. e.g.:
*
*
* FieldTextBuilder builder = new FieldTextBuilder(new MATCH("MYFIELD", "VALUE"));
* builder.AND(new EQUAL("NUMERIC_FIELD", 6));
*
* evaluates to:
*
* MATCH{VALUE}:MYFIELD+AND+EQUAL{6}:NUMERIC_FIELD
*
* Note that calls to {@code AND}, {@code OR}, {@code XOR} and {@code NOT} return the same builder on which they were
* called so the return value is usually discarded.
*
* {@code FieldTextBuilder} manages boolean operators to ensure that there are no leading or trailing operators.
* Further, it will omit parentheses where possible.
*
*
This class is {@code final} to ensure that the assumptions made in the optimizations for combining
* {@code FieldTextBuilder}s cannot be violated.
*
*
This class makes no attempt at internal synchronization and is not safe to be used by multiple threads without
* external synchronization.
*
*
The lack of an empty constructor is not an oversight, it caused some confusion in earlier versions of the class
* and has been removed. The problem occurred in the treatment of the OR operator, which is logically expected to
* increase the size of the results set. However, an empty fieldtext matches all documents so appending more fieldtext
* via OR should have no effect. This was not the behaviour in earlier versions of this class and this counter-intuitive
* behaviour led to problems. The problem was fixed using the {@link FieldTexts#MATCHNOTHING} specifier but introduced
* another problem in that it was now much more difficult to build up a sequence of expressions separated by OR. Static
* factory methods have been added to make this easier, see the examples below for current usage patterns.
*
*
Example 1 - Some of the fieldtext is fixed and can be added at construction time:
*
*
* FieldTextBuilder builder = new FieldTextBuilder(new MATCH("FIELD", "VALUE"));
*
* or:
*
*
* FieldTextBuilder builder = FieldTextBuilder.from(new MATCH("FIELD", "VALUE"));
*
* Example 2 - Creating a sequence of ORs with multiple expressions (likewise for AND and XOR):
*
*
* FieldTextBuilder builder = FieldTexts.OR(
* new MATCH("FIELD", "VALUE"),
* new EQUAL("NUMERIC", 7, 8),
* new EXISTS("UUID")
* );
*
* Example 3 - Conditionally building ORs when none of the fieldtext is fixed (likewise for AND and XOR):
*
*
* // Create a builder suitable for appending via OR
* FieldTextBuilder builder = FieldTexts.OR();
*
* for (fieldText : fieldTextList) {
builder.OR(fieldText);
* }
*
* if (builder.isMatchNothing()) {
* // Nothing was added as part of the loop so the fieldtext won't match any documents
* }
*
*
*/
public final class FieldTextBuilder extends AbstractFieldText {
private final StringBuilder fieldTextString = new StringBuilder();
private int componentCount;
private String lastOperator; // Values: null (no last operator), "" (unknown last operator), "AND", "OR", "XOR", "WHEN", "WHENn"
private boolean not;
// Creating an empty FieldTextBuilder leads to confusing behaviour under boolean operators. Given this class is
// final there's no need to worry about subclasses, so the static factory methods can be used instead to create
// 'empty' builders
FieldTextBuilder() {
}
/**
* Creates a {@code FieldTextBuilder} containing the specified fieldtext. The new object will not be backed by the
* {@code FieldText} provided.
*
* @param fieldText The initial contents of the builder.
*/
public FieldTextBuilder(final FieldText fieldText) {
Validate.notNull(fieldText, "FieldText should not be null");
setFieldText(fieldText);
}
/**
* Appends the specified fieldtext onto the builder using the AND operator. Parentheses are omitted where possible
* and logical simplifications are made in certain cases:
*
* (A AND A) {@literal =>} A
* (* AND A) {@literal =>} A
* (0 AND A) {@literal =>} 0
*
* @param fieldText A fieldtext expression or specifier.
* @return {@code this}
*/
@Override
public FieldTextBuilder AND(final FieldText fieldText) {
Validate.notNull(fieldText, "FieldText should not be null");
if (fieldText == this || toString().equals(fieldText.toString())) {
return this;
}
if (isEmpty()) {
return setFieldText(fieldText);
}
if (fieldText.isEmpty()) {
return this;
}
if (MATCHNOTHING.equals(this) || MATCHNOTHING.equals(fieldText)) {
return setFieldText(MATCHNOTHING);
}
return binaryOperation("AND", fieldText);
}
/**
* Appends the specified fieldtext onto the builder using the OR operator. Parentheses are omitted where possible
* and logical simplifications are made in certain cases:
*
* (A OR A) {@literal =>} A
* (* OR A) {@literal =>} *
* (0 OR A) {@literal =>} A
*
* @param fieldText A fieldtext expression or specifier.
* @return {@code this}
*/
@Override
public FieldTextBuilder OR(final FieldText fieldText) {
Validate.notNull(fieldText, "FieldText should not be null");
if (fieldText == this || toString().equals(fieldText.toString())) {
return this;
}
if (isEmpty() || MATCHNOTHING.equals(fieldText)) {
return this;
}
if (fieldText.isEmpty() || MATCHNOTHING.equals(this)) {
return setFieldText(fieldText);
}
return binaryOperation("OR", fieldText);
}
/**
* Appends the specified fieldtext onto the builder using the XOR operator. Parentheses are omitted where possible
* and logical simplifications are made in certain cases:
*
* (A XOR A) {@literal =>} 0
* (* XOR A) {@literal =>} NOT A
* (0 XOR A) {@literal =>} A
*
* @param fieldText A fieldtext expression or specifier.
* @return {@code this}.
*/
@Override
public FieldTextBuilder XOR(final FieldText fieldText) {
Validate.notNull(fieldText, "FieldText should not be null");
if (fieldText == this || toString().equals(fieldText.toString())) {
return setFieldText(MATCHNOTHING);
}
if (isEmpty()) {
return setFieldText(fieldText).NOT();
}
if (fieldText.isEmpty()) {
return NOT();
}
if (MATCHNOTHING.equals(this)) {
return setFieldText(fieldText);
}
if (MATCHNOTHING.equals(fieldText)) {
return this;
}
return binaryOperation("XOR", fieldText);
}
/**
* Appends the specified fieldtext onto the builder using the WHEN operator. A simplification is made in the case
* where the passed {@code fieldText} is equal to {@code this}:
*
* (A WHEN A) {@literal =>} A
*
* @param fieldText A fieldtext expression or specifier.
* @return {@code this}
*/
@Override
public FieldTextBuilder WHEN(final FieldText fieldText) {
Validate.notNull(fieldText, "FieldText should not be null");
Validate.isTrue(fieldText.size() >= 1, "FieldText must have a size greater or equal to 1");
if(size() < 1) {
throw new IllegalStateException("Size must be greater or equal to 1");
}
// Here we assume that A WHEN A => A but is there an edge case where this doesn't work?
if(fieldText == this || toString().equals(fieldText.toString())) {
return this;
}
if(MATCHNOTHING.equals(this) || MATCHNOTHING.equals(fieldText)) {
return setFieldText(MATCHNOTHING);
}
return binaryOperation("WHEN", fieldText);
}
/**
* Appends the specified fieldtext onto the builder using the WHENn operator.
*
* @param depth The n in WHENn
* @param fieldText A fieldtext expression or specifier.
* @return {@code this}.
*/
@Override
public FieldTextBuilder WHEN(final int depth, final FieldText fieldText) {
Validate.isTrue(depth >= 1, "Depth must be at least 1");
Validate.notNull(fieldText, "FieldText should not be null");
Validate.isTrue(fieldText.size() >= 1, "FieldText must have a size greater or equal to 1");
if(size() < 1) {
throw new IllegalStateException("Size must be greater or equal to 1");
}
// We omit this 'optimization' because for large n it doesn't work, X WHENn A => 0, even if X = A
//if (fieldText == this || toString().equals(fieldText.toString())) {
// return this;
//}
if(fieldText == this) {
return binaryOperation("WHEN" + depth, new FieldTextWrapper(this));
}
if(MATCHNOTHING.equals(this) || MATCHNOTHING.equals(fieldText)) {
return setFieldText(MATCHNOTHING);
}
return binaryOperation("WHEN" + depth, fieldText);
}
/**
* Clears any existing fieldtext and replaces it with the specified fieldtext.
*
* @param fieldText
* @return
*/
private FieldTextBuilder setFieldText(final FieldText fieldText) {
return clear().binaryOperation(null, fieldText);
}
/**
* Does the work for the OR, AND, XOR and WHEN methods.
*
* @param operator
* @param fieldText
* @return
*/
private FieldTextBuilder binaryOperation(final String operator, final FieldText fieldText) {
// This should never happen as the AND, OR, XOR and WHEN methods should already have checked this
Validate.isTrue(fieldText != this);
// Optimized case when fieldText is a FieldTextBuilder
if (fieldText instanceof FieldTextBuilder) {
return binaryOp(operator, (FieldTextBuilder)fieldText);
}
// Special case when we're empty
if (componentCount == 0) {
fieldTextString.append(fieldText.toString());
if (fieldText.size() > 1) {
lastOperator = "";
}
}
else {
// Add the NOTs, parentheses and operator
addOperator(operator);
// Size 1 means a single specifier, so no parentheses
if (fieldText.size() == 1) {
fieldTextString.append(fieldText.toString());
}
else {
fieldTextString.append('(').append(fieldText.toString()).append(')');
}
}
componentCount += fieldText.size();
not = false;
return this;
}
/**
* Does the work for the OR, AND and XOR methods in the optimized case where fieldText is a FieldTextBuilder.
*
* @param operator
* @param fieldText
* @return
*/
private FieldTextBuilder binaryOp(final String operator, final FieldTextBuilder fieldText) {
// Special case when we're empty
if (componentCount == 0) {
// Just copy the argument
fieldTextString.append(fieldText.fieldTextString);
lastOperator = fieldText.lastOperator;
not = fieldText.not;
componentCount = fieldText.componentCount;
}
else {
// Add the NOTs, parentheses and operator
addOperator(operator);
if (fieldText.lastOperator == null || fieldText.lastOperator.equals(operator) || fieldText.not) {
fieldTextString.append(fieldText.toString());
}
else {
fieldTextString.append('(').append(fieldText.toString()).append(')');
}
componentCount += fieldText.size();
not = false;
}
return this;
}
/**
* The first section of the fieldtext manipulation process, where the NOTs are handled and the operator is added,
* are common to both binaryOp methods so this method saves us repeating the code.
*
* @param operator
*/
private void addOperator(final String operator) {
// This should never happen but sanity check...
Validate.notNull(operator);
if (lastOperator == null) {
lastOperator = operator;
}
/* Logic to determine whether to put parentheses around the existing fieldtext. This is tied to the use of the
* NOT operator as it also adds parentheses.
*/
if (not) {
if (componentCount == 1) {
fieldTextString.insert(0, "NOT+");
}
else {
fieldTextString.insert(0, "NOT(").append(')');
}
}
else if (!lastOperator.equals(operator)) {
fieldTextString.insert(0, '(').append(')');
}
lastOperator = operator;
fieldTextString.append('+').append(operator).append('+');
}
/**
* Applies a NOT operator to the {@code FieldTextBuilder}. Note that two calls to {@code NOT()} will cancel out.
*
* @return {@code this}
*/
@Override
public FieldTextBuilder NOT() {
// NOT * => 0
// NOT 0 => *
if (isEmpty()) {
return setFieldText(MATCHNOTHING);
}
if (MATCHNOTHING.equals(this)) {
return clear();
}
not ^= true;
return this;
}
/**
* The total number of fieldtext specifiers in the current expression.
*
* @return The size of the expression.
*/
@Override
public int size() {
return componentCount;
}
/**
* Whether or not the builder is empty.
*
* @return {@code true} if and only if size is 0.
*/
@Override
public boolean isEmpty() {
return componentCount == 0;
}
/**
* The {@code String} representation of the fieldtext expression, exactly as it should be sent to IDOL.
*
* @return The {@code String} representation.
*/
@Override
public String toString() {
if (!not) {
return fieldTextString.toString();
}
if (componentCount == 1) {
return "NOT+" + fieldTextString;
}
return "NOT(" + fieldTextString + ')';
}
/**
* Clears all fields back to an empty builder.
*
* @return {@code this}
*/
public FieldTextBuilder clear() {
fieldTextString.setLength(0);
componentCount = 0;
not = false;
lastOperator = null;
return this;
}
/**
* Whether or not the builder is equal to {@link FieldTexts#MATCHNOTHING}.
*
* @return {@code true} if and only if {@code this} is equal to {@link FieldTexts#MATCHNOTHING MATCHNOTHING}
*/
// TODO: Promote to the FieldText interface?
public boolean isMatchNothing() {
return equals(MATCHNOTHING);
}
/**
* Converts a {@link FieldText} object into a {@code FieldTextBuilder}. This method can be more efficient than using
* the equivalent constructor but the returned object might be backed by the {@link FieldText} object provided.
*
* @param fieldText The fieldtext expression to convert
* @return An equivalent instance of {@code FieldTextBuilder}
*/
public static FieldTextBuilder from(final FieldText fieldText) {
return (fieldText instanceof FieldTextBuilder) ? (FieldTextBuilder)fieldText : new FieldTextBuilder(fieldText);
}
}