/** * Copyright (c) 2014, University of Warsaw * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are permitted * provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list of * conditions and the following disclaimer in the documentation and/or other materials provided * with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY * WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package pl.edu.mimuw.cloudatlas.model; import pl.edu.mimuw.cloudatlas.model.Value; import java.io.Serializable; /** * A single value stored as an attribute. */ public abstract class Value implements Serializable { /** * An operation that may be performed on values. */ public enum Operation { EQUAL, COMPARE, ADD, SUBTRACT, MULTIPLY, DIVIDE, MODULO, AND, OR, REG_EXPR, NEGATE, VALUE_SIZE, } /** * Gets the type of this value. * * @return type of this value */ public abstract Type getType(); /** * Indicates whether this value is null. Distinct from a Value reference that is null * itself. * * @return true if and only if this value is null */ public abstract boolean isNull(); protected final void sameTypesOrThrow(Value value, Operation operation) { if(!getType().isCompatible(value.getType())) throw new IncompatibleTypesException(getType(), value.getType(), operation); } /** * Checks whether this value is equal to the specified one (operator ==). * * @param value the right side of the operator * @return a ValueBoolean representing true if and only if both values are equal * @throws UnsupportedValueOperationException if this operation is unsupported for these values */ public Value isEqual(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.EQUAL); } /** * Indicates whether this object is equal to another one. * * @param object the object to check * @return whether two objects are equal */ @Override public boolean equals(Object object) { if(!(object instanceof Value)) return false; return ((ValueBoolean)isEqual((Value)object)).getValue(); } /** * Checks whether this value is lower than the specified one (operator <=). * * @param value the right side of the operator * @return a ValueBoolean representing true if and only if this value is lower than the provided one * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value isLowerThan(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.COMPARE); } /** * Returns a new value created by adding argument to this value (operator +). * * @param value the right side of the operator * @return a sum of two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value addValue(Value value) { // name clash with add from List interface throw new UnsupportedValueOperationException(getType(), Operation.ADD); } /** * Returns a new value created by subtracting argument from this value (operator -). * * @param value the right side of the operator * @return a difference of two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value subtract(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.SUBTRACT); } /** * Returns a new value created by multiplying this value by an argument (operator *). * * @param value the right side of the operator * @return a product of two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value multiply(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.MULTIPLY); } /** * Returns a new value created by dividing this value by an argument (operator /). * * @param value the right side of the operator * @return a quotient of two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value divide(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.DIVIDE); } /** * Returns the remainder of division of this value by an argument (operator %). * * @param value the right side of the operator * @return a remainder * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * incompatible or non-numeric types) */ public Value modulo(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.MODULO); } /** * Returns the result of a logical AND (operator &&). * * @param value the right side of the operator * @return a conjunction of the two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * non-boolean types) */ public Value and(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.AND); } /** * Returns the result of a logical OR (operator ||). * * @param value the right side of the operator * @return an alternative of two values * @throws UnsupportedValueOperationException if this operator is unsupported for these values (for example * non-boolean types) */ public Value or(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.OR); } /** * Returns a result of trying to match to this value with a regular expression specified as an argument. * * @param value the regular expression to match against * @return a ValueBoolean representing true if and only if this value matches provided regular * expression * @throws UnsupportedValueOperationException if this operator is unsupported for these values */ public Value regExpr(Value value) { throw new UnsupportedValueOperationException(getType(), Operation.REG_EXPR); } /** * Returns the negation (numeric or logical) of this value. This may refer to operator - or !, depending on type. * * @return a value that is the negation of this value * @throws UnsupportedValueOperationException if this operator is unsupported for this value */ public Value negate() { // !, - throw new UnsupportedValueOperationException(getType(), Operation.NEGATE); } /** * Returns the size of this value. Semantics depend on type. * * @return a size of this value * @throws UnsupportedValueOperationException if this operation is unsupported for this value */ public Value valueSize() { // name clash with size from List interface throw new UnsupportedValueOperationException(getType(), Operation.VALUE_SIZE); } /** * Returns this value converted to another type. * * @param to a desired type * @return this value converted to the type * @throws UnsupportedConversionException if a requested conversion is unsupported */ public abstract Value convertTo(Type to); /** * Returns a textual representation of this value. This method uses conversion to ValueString. * * @return a textual representation of this value * @see #convertTo(Type) */ @Override public String toString() { return ((ValueString)convertTo(TypePrimitive.STRING)).getValue(); } /** * Returns a default value (such as an uninitialized variable). This may be 0 for integer types, * false for boolean, null for complex types, etc. * * @return a default value of this type */ public abstract Value getDefaultValue(); }