io.permazen.UpgradeConversionPolicy Maven / Gradle / Ivy
Show all versions of permazen-main Show documentation
/*
* Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
*/
package io.permazen;
/**
* Policies to apply when a simple or counter field's type changes during a schema update.
*
*
* Type Changes
*
*
* Permazen fields are identified by their {@linkplain JSimpleField#getStorageId storage ID's}, which is typically
* {@linkplain StorageIdGenerator derived automatically} from the field's name.
* With one restriction*, the type of a field may change arbitrarily between schema versions.
*
*
* When changing an object's schema version, Permazen supports optional automatic conversion of simple field
* values from the old type to the new type. For example, an {@code int} field value {@code 1234} can be automatically
* converted into {@link String} field value {@code "1234"}.
*
*
* See {@link io.permazen.core.FieldType#convert} for details about conversions between simple field types. In addition,
* {@link io.permazen.Counter} fields can be converted to/from any numeric Java primitive (or primitive wrapper) type.
*
*
* This class is used to {@linkplain io.permazen.annotation.JField#upgradeConversion specify} whether such automatic
* conversion should occur when a simple field's type changes, and if so, whether the conversion must always succeed.
*
*
* *A simple field may not have different types across schema versions and be indexed in both versions.
*
*
* References and Enums
*
*
* Permazen considers {@link Enum} types with different identifier lists as different types. However, automatic
* conversion of {@link Enum} values will work if the existing value's name is valid for the new {@link Enum} type.
*
*
* Automatic conversion of reference fields also works as long as the referenced object's type is assignable
* to the field's new Java type (otherwise, the field is set to null).
*
*
* Conversion Policies
*
*
* With {@link #RESET}, no automatic conversion is attempted: the field is always reset to the default value of
* the new type. With {@link #ATTEMPT} and {@link #REQUIRE}, automatic conversion of field values is attempted.
*
*
* For some types and/or field values, conversion is not possible. In this case, {@link #REQUIRE} generates a
* {@link UpgradeConversionException}, while {@link #ATTEMPT} just reverts to the behavior of {@link #RESET}.
*
*
* Note that arbitrary conversion logic is always possible using
* {@link io.permazen.annotation.OnVersionChange @OnVersionChange}.
*
* @see io.permazen.annotation.JField#upgradeConversion
* @see io.permazen.core.FieldType#convert FieldType.convert()
*/
public enum UpgradeConversionPolicy {
/**
* Do not attempt to automatically convert values to the new type.
*
*
* Instead, during a schema version change, the field will be reset to the default value of the field's new type.
*/
RESET(false, false),
/**
* Attempt automatic conversion of field values to the new type, and if automatic conversion fails,
* set the value to the new type's default value as would {@link #RESET}.
*/
ATTEMPT(true, false),
/**
* Attempt automatic conversion of field values to the new type, and iIf automatic conversion fails,
* throw a {@link UpgradeConversionException}.
*/
REQUIRE(true, true);
private final boolean convertsValues;
private final boolean requireConversion;
UpgradeConversionPolicy(boolean convertsValues, boolean requireConversion) {
this.convertsValues = convertsValues;
this.requireConversion = requireConversion;
}
/**
* Determine whether this policy should attempt to convert field values from the old type to the new type.
*
*
* If this is false, the field's value will be set to the new type's default value.
* If this is true, the field's old value will be converted to the field's new type if possible;
* if the conversion fails, the behavior depends on {@link #isRequireConversion}.
*
* @return true if under this policy conversion should be attempted
*/
public boolean isConvertsValues() {
return this.convertsValues;
}
/**
* Determine whether failed attempts to convert a field's value from the old type to the new type should be fatal.
*
*
* If this is true, a failed conversion attempt results in a {@link UpgradeConversionException} being thrown.
* If this is false, a failed conversion attempt results in the field being set to the new type's default value.
*
* @return true if under this policy conversion is mandatory
*/
public boolean isRequireConversion() {
return this.requireConversion;
}
}