io.permazen.annotation.JField Maven / Gradle / Ivy
Show all versions of permazen-main Show documentation
/*
* Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
*/
package io.permazen.annotation;
import io.permazen.UpgradeConversionPolicy;
import io.permazen.core.DeleteAction;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* Java annotation for defining simple fields, including reference fields that refer to other Java model object types,
* and {@link io.permazen.Counter} fields.
*
*
* This annotation is used in two scenarios:
*
* - To describe a simple or counter database field by annotating the corresponding abstract Java bean
* property `getter' method
* - To describe the sub-field of a complex database field (i.e., set, list, or map), that is,
* a collection {@code element} field, or a map {@code key} or {@code value} field. In this case this annotation
* nests within the corresponding {@link JListField @JListField}, {@link JSetField @JSetField},
* or {@link JMapField @JMapField} annotation.
*
*
*
* This annotation can be applied to superclass and interface methods to have the corresponding field defined in all
* {@link PermazenType @PermazenType}-annotated sub-types.
*
*
* When auto-generation of properties is enabled, use of this annotation is not required unless you need to override
* the defaults; see {@link PermazenType#autogenFields}.
*
*
Non-Reference Fields
*
*
* If the field is not a reference field, the property type is inferred from the type of the annotated method or,
* in the case of complex sub-fields, the generic type of the collection class. The name of the property type
* must be registered in the {@link io.permazen.core.FieldTypeRegistry} (perhaps via {@link JFieldType @JFieldType}),
* and the corresponding {@link io.permazen.core.FieldType} is then used to encode/decode field values.
* See {@link io.permazen.core.FieldTypeRegistry} for a list of built-in (pre-defined) field types.
* The type name may also be specified explicitly by {@link #name}.
*
*
* Simple fields may be {@link #indexed}; see {@link io.permazen.index} for information on querying indexes.
* {@link io.permazen.Counter} fields may not be indexed.
*
*
* Two or more simple fields may be indexed together in a composite index; see {@link JCompositeIndex @JCompositeIndex}.
*
*
Reference Fields
*
*
* If the type of the field is (assignable to) a {@link PermazenType @PermazenType}-annotated Java model object type,
* or any supertype thereof, then the field is a reference field.
*
*
* Reference fields are always indexed; the value of {@link #indexed} is ignored.
*
*
Referential Integrity
*
*
* In general, reference fields may reference objects that don't actually exist. This can happen in one of two ways:
* (a) a field is set to an invalid reference, or (b) a field references a valid object that is subsequently deleted.
* The {@link #allowDeleted} and {@link #onDelete} properties, respectively, control whether (a) or (b) is permitted.
*
*
* By default, neither (a) nor (b) is allowed; if attempted, a {@link io.permazen.core.DeletedObjectException} is thrown.
* This ensures references are always valid.
*
*
Copy Cascades
*
*
* The {@link io.permazen.JObject} methods {@link io.permazen.JObject#cascadeCopyIn cascadeCopyIn()},
* {@link io.permazen.JObject#cascadeCopyOut cascadeCopyOut()}, and {@link io.permazen.JObject#cascadeCopyTo cascadeCopyTo()}
* copy a graph of related objects between transactions by first copying a starting object, then cascading through matching
* reference fields and repeating recursively. This cascade operation is capable of traversing references in both the
* forward and inverse directions.
*
*
* Which reference fields are traversed in a particular copy operation is determined by the supplied cascade name.
* Outgoing references are traversed if the cascade name is in the reference field's {@link #cascades} property,
* while incoming references from other objects are traversed (in the reverse direction) if the cascade name is in the
* referring object's reference field's {@link #inverseCascades}.
*
*
* For example:
*
* @PermazenType
* public interface TreeNode extends JObject {
*
* /**
* * Get the parent of this node, or null if node is a root.
* */
* @JField(cascades = { "tree", "ancestors" }, inverseCascades = { "tree", "descendants" })
* TreeNode getParent();
* void setParent(TreeNode parent);
*
* /**
* * Get the children of this node.
* */
* @FollowPath(inverseOf = "parent", startingFrom = TreeNode.class)
* NavigableSet<TreeNode> getChildren();
*
* default TreeNode copySubtreeTo(JTransaction dest) {
* return (TreeNode)this.cascadeCopyTo(dest, "descendants", false);
* }
*
* default TreeNode copyWithAnscestorsTo(JTransaction dest) {
* return (TreeNode)this.cascadeCopyTo(dest, "ancestors", false);
* }
*
* default TreeNode copyEntireTreeTo(JTransaction dest) {
* return (TreeNode)this.cascadeCopyTo(dest, "tree", false);
* }
*
* default TreeNode cloneEntireTreeTo(JTransaction dest) {
* return (TreeNode)this.cascadeCopyTo(dest, "tree", true);
* }
*
* default TreeNode cloneEntireTree() {
* return (TreeNode)this.cascadeCopyTo(this.getTransaction(), "tree", true);
* }
* }
*
*
* Delete Cascades
*
*
* Reference fields have configurable behavior when the referring object or the referred-to object is deleted;
* see {@link #onDelete} and {@link #cascadeDelete}.
*
*
Uniqueness Constraints
*
*
* Fields that are not complex sub-fields may be marked as {@link #unique} to impose a uniqueness constraint on the value.
* Fields with uniqueness constraints must be indexed. Uniqueness constraints are handled at the Permazen layer and function as
* an implicit validation constraint. In other words, the constraint is verified when the validation queue is processed
* and is affected by the transaction's configured {@link io.permazen.ValidationMode}.
*
*
* Optionally, specific field values may be marked as excluded from the uniqueness constraint via {@link #uniqueExclude}.
* If so, the specified values may appear in multiple objects without violating the constraint. Because null values
* are not allowed in annotations, include {@link #NULL} to indicate that null values should be excluded.
*
*
* In {@link io.permazen.ValidationMode#AUTOMATIC}, any upgraded {@link io.permazen.JObject}s are automatically
* added to the validation queue, so a uniqueness constraint added in a new schema version will be automatically verified
* when any object is upgraded.
*
*
* Beware however, that like all other types of validation constraint, uniqueness constraints can be added or changed on a field
* without any schema version change. Therefore, after such changes, it's possible for pre-existing database objects that were
* previously valid to suddenly become invalid, and these invalid objects would not be detected until they are validated in some
* future transaction and a validation exception is thrown.
*
*
Upgrade Conversions
*
*
* When a field's type has changed in a new schema version, the old field value can be automatically converted into the
* new type. See {@link #upgradeConversion} for how to control this behavior.
*
*
Meta-Annotations
*
*
* This annotation may be configured indirectly as a Spring
* meta-annotation
* when {@code spring-core} is on the classpath.
*/
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.ANNOTATION_TYPE, ElementType.METHOD })
@Documented
public @interface JField {
/**
* Value for use with {@link #uniqueExclude} to represent a null value.
*
*
* Note: this particular {@link String} will never conflict with any actual field values because it contains a character
* that is not allowed in the return value from {@link io.permazen.core.FieldType#toString(Object) FieldType.toString()}.
*/
String NULL = "\u0000";
/**
* The name of this field.
*
*
* If equal to the empty string (default value), the name is inferred from the name of the annotated Java bean getter method.
*
*
* For sub-fields of complex fields, this property must be left unset.
*
* @return the name of the field
*/
String name() default "";
/**
* Optional override for the type of this field.
*
*
* If set, this must equal the name of a type registered in the {@link io.permazen.core.FieldTypeRegistry}
* associated with the {@link io.permazen.core.Database} instance, and the annotated method's return type must match the
* {@link io.permazen.core.FieldType}'s {@linkplain io.permazen.core.FieldType#getTypeToken supported Java type}.
*
*
* If equal to the empty string (default value), then the Java type is inferred from the return type of the getter method
* and the {@link io.permazen.core.FieldType} is found via
* {@link io.permazen.core.FieldTypeRegistry#getFieldType(com.google.common.reflect.TypeToken)
* FieldTypeRegistry.getFieldType()}.
*
*
* For reference fields (i.e., methods with return value equal to a {@link PermazenType @PermazenType}-annotated class),
* this property must be left unset.
*
*
* For sub-fields of complex fields, this property can be used to force a primitive sub-field type instead of a
* primitive wrapper type. In that case, the complex field will disallow null values. For example:
*
* @JSetField(element = @JField(type = "float")) // nulls will be disallowed
* public abstract List<Float> getScores();
*
*
* @return the name of the field's type
* @see io.permazen.core.FieldType
* @see io.permazen.core.FieldTypeRegistry#getFieldType(String, long)
*/
String type() default "";
/**
* Optional override for the {@linkplain io.permazen.core.FieldType#getEncodingSignature encoding signature}
* associated with this field's {@link io.permazen.core.FieldType} used to encode/decode field values.
*
* @return the encoding signature of the field's type
* @see io.permazen.core.FieldType
* @see io.permazen.core.FieldTypeRegistry#getFieldType(String, long)
*/
long typeSignature() default 0;
/**
* Storage ID for this field.
*
*
* Value should be positive and unique within the contained class.
*
*
* If zero, the configured {@link io.permazen.StorageIdGenerator} will be consulted to auto-generate a value
* unless {@link PermazenType#autogenFields} is false (in which case an error occurs).
*
* @see io.permazen.StorageIdGenerator#generateFieldStorageId StorageIdGenerator.generateFieldStorageId()
* @see io.permazen.StorageIdGenerator#generateSetElementStorageId StorageIdGenerator.generateSetElementStorageId()
* @see io.permazen.StorageIdGenerator#generateListElementStorageId StorageIdGenerator.generateListElementStorageId()
* @see io.permazen.StorageIdGenerator#generateMapKeyStorageId StorageIdGenerator.generateMapKeyStorageId()
* @see io.permazen.StorageIdGenerator#generateMapValueStorageId StorageIdGenerator.generateMapValueStorageId()
*
* @return the field's storage ID
*/
int storageId() default 0;
/**
* Whether this field is indexed or not.
*
*
* Setting this property to true creates a simple index on this field. To have this field participate in
* a composite index on multiple fields, use {@link JCompositeIndex @JCompositeIndex}.
*
*
* Note: reference fields are always indexed (for reference fields, this property is ignored).
*
* @return whether the field is indexed
* @see JCompositeIndex @JCompositeIndex
*/
boolean indexed() default false;
/**
* Define forward copy cascades for the annotated reference field.
*
*
* When {@link io.permazen.JObject#cascadeCopyIn JObject.cascadeCopyIn()},
* {@link io.permazen.JObject#cascadeCopyOut JObject.cascadeCopyOut()}, or
* {@link io.permazen.JObject#cascadeCopyTo JObject.cascadeCopyTo()} is invoked, if the given cascade name is one
* of the names listed here, and an object with the annotated reference field is copied, then the reference field will
* will be traversed in the forward direction and the referred-to object will also be copied.
*
*
* For non-reference fields this property must be equal to its default value.
*
* @return whether the field is indexed
* @see io.permazen.JObject#cascadeCopyTo JObject.cascadeCopyTo()
* @see io.permazen.JTransaction#cascadeFindAll JTransaction.cascadeFindAll()
*/
String[] cascades() default {};
/**
* Define inverse copy cascades for the annotated reference field.
*
*
* When {@link io.permazen.JObject#cascadeCopyIn JObject.cascadeCopyIn()},
* {@link io.permazen.JObject#cascadeCopyOut JObject.cascadeCopyOut()}, or
* {@link io.permazen.JObject#cascadeCopyTo JObject.cascadeCopyTo()} is invoked, if the given cascade name is one
* of the names listed here, and an object with the annotated reference field refers to an object that is copied, then the
* reference field will be traversed in the inverse direction and the referring object will also be copied.
*
*
* For non-reference fields this property must be equal to its default value.
*
* @return whether the field is indexed
* @see io.permazen.JObject#cascadeCopyTo JObject.cascadeCopyTo()
* @see io.permazen.JTransaction#cascadeFindAll JTransaction.cascadeFindAll()
*/
String[] inverseCascades() default {};
/**
* For reference fields, configure the behavior when the referred-to object is
* {@linkplain io.permazen.JObject#delete deleted}.
*
*
* For non-reference fields this property must be equal to its default value.
*
* @return desired behavior when a referenced object is deleted
* @see #cascadeDelete
* @see io.permazen.JObject#delete
*/
DeleteAction onDelete() default DeleteAction.EXCEPTION;
/**
* For reference fields, configure cascading behavior when the referring object is
* {@linkplain io.permazen.JObject#delete deleted}. If set to true, the referred-to object
* is automatically deleted as well.
*
*
* For non-reference fields this property must be equal to its default value.
*
* @return whether deletion should cascade to the referred-to object
* @see #onDelete
* @see io.permazen.JObject#delete
*/
boolean cascadeDelete() default false;
/**
* Require this field's value to be unique among all database objects.
*
*
* This property creates an implicit uniqueness validation constraint.
*
*
* The constraint will be checked any time normal validation is performed on an object containing the field.
* More precisely, a uniqueness constraint behaves like a JSR 303
* validation constraint with {@code groups() = }{
{@link javax.validation.groups.Default}{@code .class,
* }{@link io.permazen.UniquenessConstraints}{@code .class} }
. Therefore, uniqueness constraints
* are included in default validation, but you can also validate only uniqueness constraints via
* {@link io.permazen.JObject#revalidate myobj.revalidate(UniquenessConstraints.class)}.
*
*
* This property must be false for sub-fields of complex fields, and for any field that is not indexed.
*
* @return whether the field's value should be unique
* @see #uniqueExclude
* @see io.permazen.UniquenessConstraints
*/
boolean unique() default false;
/**
* Specify field value(s) which are excluded from the uniqueness constraint.
*
*
* The specified values must be valid {@link String} encodings of the associated field (as returned by
* {@link io.permazen.core.FieldType#toString(Object) FieldType.toString(T)}), or the constant {@link #NULL}
* to indicate a null value. For example:
*
* @JField(indexed = true, unique = true, uniqueExclude = { "Infinity", "-Infinity" })
* public abstract float getPriority();
*
* @JField(indexed = true, unique = true, uniqueExclude = { JField.NULL })
* public abstract String getName();
*
*
*
* This property must be left empty when {@link #unique} is false.
*
* @return values to exclude from the uniqueness constraint
* @see #unique
*/
String[] uniqueExclude() default {};
/**
* Allow the field to reference non-existent objects in normal transactions.
*
*
* For non-reference fields, this property must be equal to its default value.
*
*
* Otherwise, if this property is set to false, the field is disallowed from ever referring to a non-existent object;
* instead, a {@link io.permazen.core.DeletedObjectException} will be thrown. When used together with
* {@link DeleteAction#EXCEPTION} (see {@link #onDelete}), the field is guaranteed to never be a dangling reference.
*
*
* This property only controls validation in regular (non-snapshot transactions); {@link #allowDeletedSnapshot}
* separately controls validation for {@link io.permazen.SnapshotJTransaction}s.
*
*
* For consistency, this property must be set to true when {@link #onDelete} is set to {@link DeleteAction#NOTHING}.
*
* @return whether the reference field should allow assignment to deleted objects in normal transactions
* @see #onDelete
* @see #allowDeletedSnapshot
* @see PermazenType#autogenAllowDeleted
*/
boolean allowDeleted() default false;
/**
* Allow the field to reference non-existent objects in snapshot transactions.
*
*
* For non-reference fields, this property must be equal to its default value.
*
*
* This property is equivalent to {@link #allowDeleted}, but applies to {@link io.permazen.SnapshotJTransaction}s
* instead of normal {@link io.permazen.JTransaction}s; see {@link #allowDeleted} for details.
*
*
* Snapshot transactions typically hold a copy of some small portion of the database. If this property is set to false,
* then it effectively creates a requirement that this "small portion" be transitively closed under object references.
*
*
* For consistency, this property must be set to true when {@link #onDelete} is set to {@link DeleteAction#NOTHING}.
*
* @return whether the reference field should allow assignment to deleted objects in snapshot transactions
* @see #onDelete
* @see #allowDeleted
* @see PermazenType#autogenAllowDeletedSnapshot
*/
boolean allowDeletedSnapshot() default true;
/**
* Specify the {@link UpgradeConversionPolicy} policy to apply when a schema change occurs and this field's type changes.
*
*
* With one restriction*, Permazen supports schema changes that alter a field's type, and in some cases
* can automatically convert field values from the old to the new type (for example, from the {@code int} value {@code 1234}
* to the {@link String} 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 property defines the {@link UpgradeConversionPolicy} for the annotated field when upgrading an object from some
* other schema version to the current schema version. Note custom conversion logic is also possible using
* {@link OnVersionChange @OnVersionChange} methods.
*
*
* For sub-fields of complex fields, this property is ignored.
*
*
* *A simple field may not have different types across schema versions and be indexed in both versions.
*
* @return upgrade conversion policy for this field
* @see UpgradeConversionPolicy
* @see io.permazen.core.FieldType#convert FieldType.convert()
*/
UpgradeConversionPolicy upgradeConversion() default UpgradeConversionPolicy.ATTEMPT;
}