All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.permazen.ReferencePath Maven / Gradle / Ivy

There is a newer version: 5.1.0
Show newest version

/*
 * Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
 */

package io.permazen;

import com.google.common.base.Preconditions;
import com.google.common.primitives.Ints;
import com.google.common.reflect.TypeToken;

import io.permazen.core.ObjId;
import io.permazen.kv.KeyRange;
import io.permazen.kv.KeyRanges;
import io.permazen.schema.SchemaObjectType;
import io.permazen.util.ParseContext;

import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Reference paths.
 *
 * 

* A reference path is a {@link String} specifying a path from some implicit starting object type, through zero * or more reference fields, ending up at some target object type(s). * In other words, given some starting object(s), a reference path takes you through a path of references to the target object(s). * Note that the number of target objects can be vastly different than the number of starting objects, depending on * the fan-in/fan-out of the references traversed. * *

Specifying Fields * *

* The starting object type is always implicit from the context in which the path is used. The {@link String} path describes * the reference fields, i.e., the "steps" in the path that travel from the starting object type to the target object type. * The steps are separated by dot ({@code "."}) characters. * *

* The reference fields in the path may be traversed in either the forward or inverse direction: * to traverse a field in the forward direction, specify name of the field; to traverse a field in the inverse direction, * specify the name of the field and its containing type using the syntax {@code ^Type:field^}. * *

* For complex fields, specify both the field and sub-field: for {@link Set} and {@link java.util.List} fields, the sub-field is * always {@code "element"}, while for {@link java.util.Map} fields the sub-field is either {@code "key"} or {@code "value"}. * For example, to traverse a map field's {@code key} sub-field, specify {@code "mymap.key"}. * *

Target Fields * *

* A reference path may optionally have a target field appended to the end of the path. The target field does not have * to be a reference field. If a target field is specified, the target object types * are restricted to those types containing the target field. * *

* Note that to avoid ambiguity it must be known at the time the path is parsed whether the path contains a target field: for * example, consider the path {@code "parent.parent"} in the context of a {@code Child} object: if there is a target field, * the target object is the child's parent and the target field is the {@code "parent"} field of the child's parent * (which just happens to also be a reference field), but if there is no target field, the path simply refers to the child's * grandparent. * *

Examples * *

* Considering the following model classes: * *

 * @PermazenType
 * public class Employee {
 *
 *     public abstract String getName();
 *     public abstract void setName(String name);
 *
 *     public abstract Employee getManager();
 *     public abstract void setManager(Employee manager);
 *
 *     public abstract Set<Asset> getAssets();
 * }
 *
 * @PermazenType
 * public class Asset {
 *
 *     public abstract int getAssetId();
 *     public abstract void setAssetId(int assetId);
 *
 *     public abstract String getName();
 *     public abstract void setName(String name);
 * }
 * 
* * Then these paths have the following meanings: * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Start TypePathHas Target Field?Description
{@code Employee}{@code ""}NoThe {@code Employee}
{@code Employee}{@code ""}YesInvalid - no target field specified
{@code Employee}{@code "name"}YesThe {@code Employee}'s name
{@code Employee}{@code "name"}NoInvalid - {@code "name"} is not a reference field
{@code Asset}{@code "name"}YesThe {@code Asset}'s name
{@code Object}{@code "name"}YesThe {@code Employee}'s or {@code Asset}'s name
{@code Employee}{@code "manager"}NoThe {@code Employee}'s manager
{@code Employee}{@code "manager"}YesThe {@code "manager"} field of the {@code Employee}
{@code Employee}{@code "manager.name"}YesThe {@code Employee}'s manager's name
{@code Employee}{@code "^Employee:manager^.name"}YesThe names of all of the {@code Employee}'s direct reports
{@code Employee}{@code "manager.assets.description"}YesThe descriptions of the {@code Employee}'s manager's assets
{@code Employee}{@code "manager.^Employee:manager^"}NoAll of the {@code Employee}'s manager's direct reports
{@code Asset}{@code "^Employee:assets.element^"}NoThe employee owning the {@code Asset}
{@code Asset}{@code "^Employee:assets.element^"}YesInvalid - no target field specified
{@code Asset}{@code "^Employee:assets.element^}
  {@code .manager.^Employee:manager^.asset.assetId"}
YesID's of all {@code Asset}s owned by direct reports of the manager of the {@code Employee} owning the original * {@code Asset}
*
* *

Fields of Sub-Types * *

* The same field can appear in multiple types, e.g., {@code "name"} in the example above appears in both {@code Employee} * and {@code Asset}. The set of all possible object types is recalculated at each step in the reference path, including * at the last step, which gives the target object type(s). At each intermediate step, as long as the Java types do not * contain incompatible definitions for the named field, the step is valid. * *

* In rare cases where multiple sub-types of a common super-type type have fields with the same name but different storage IDs, * the storage ID may be explicitly specified as a suffix, for example, {@code "name#123"}. * *

Using Reference Paths * *

* Reference paths may be explicitly created via {@link Permazen#parseReferencePath Permazen.parseReferencePath()} * and traversed in the forward direction via {@link JTransaction#followReferencePath JTransaction.followReferencePath()} * or in the inverse direction via {@link JTransaction#invertReferencePath JTransaction.invertReferencePath()}. * *

* Reference paths are also used implicitly by {@link io.permazen.annotation.OnChange @OnChange} annotations to * specify non-local objects for change monitoring, and by {@link io.permazen.annotation.FollowPath @FollowPath} * annotations. * * @see Permazen#parseReferencePath Permazen.parseReferencePath() * @see JTransaction#followReferencePath JTransaction.followReferencePath() * @see JTransaction#invertReferencePath JTransaction.invertReferencePath() * @see io.permazen.annotation.OnChange @OnChange */ public class ReferencePath { private static final String IDENT = "\\p{javaJavaIdentifierStart}\\p{javaJavaIdentifierPart}*"; private static final String IDENT_ID = IDENT + "(?:#[0-9]+)?"; private static final String IDENT_ID_1OR2 = IDENT_ID + "(?:\\." + IDENT_ID + ")?"; private static final String IDENTS = IDENT + "(?:\\." + IDENT + ")*"; private static final String FWD_STEP = "(" + IDENT_ID + ")"; private static final String REV_STEP = "\\^((" + IDENTS + "):(" + IDENT_ID_1OR2 + "))\\^"; final Permazen jdb; final Class startType; final ArrayList>> pathTypes; final Set> targetFieldTypes; final int targetFieldStorageId; final int targetSuperFieldStorageId; final int[] referenceFieldStorageIds; final Set cursors; final String path; boolean someTargetFieldIndexed; private final Logger log = LoggerFactory.getLogger(this.getClass()); private volatile KeyRanges[] pathKeyRanges; /** * Constructor. * * @param jdb {@link Permazen} against which to resolve object and field names * @param startType starting Java type for the path * @param path dot-separated path of zero or more reference fields, followed by a target field * @param withTargetField true if this path has a target field * @param lastIsSubField true if the last field can be a complex sub-field but not a complex field, false for the reverse, * or null for don't care * @throws IllegalArgumentException if {@code jdb}, {@code startType}, or {@code path} is null * @throws IllegalArgumentException if {@code startType} is not compatible with any Java model types * @throws IllegalArgumentException if {@code path} is invalid */ ReferencePath(Permazen jdb, Class startType, String path, boolean withTargetField, Boolean lastIsSubField) { // Sanity check Preconditions.checkArgument(jdb != null, "null jdb"); Preconditions.checkArgument(startType != null, "null startType"); Preconditions.checkArgument(path != null, "null path"); final String errorPrefix = "invalid path `" + path + "': "; this.jdb = jdb; this.startType = startType; this.path = path; // Debug if (this.log.isTraceEnabled()) { this.log.trace("RefPath: START startType=" + startType.getName() + " path=\"" + path + "\" withTargetField=" + withTargetField + " lastIsSubField=" + lastIsSubField); } // Split the path into field names final ArrayDeque fieldNames = new ArrayDeque<>(); final ParseContext ctx = new ParseContext(path); while (!ctx.isEOF()) { // Gobble separator if (ctx.getIndex() > 0) ctx.expect('.'); // Parse next step (either forward or reverse) Matcher matcher = ctx.tryPattern(FWD_STEP); if (matcher == null && (matcher = ctx.tryPattern(REV_STEP)) == null) { throw new IllegalArgumentException(errorPrefix + "invalid path starting at `" + ParseContext.truncate(ctx.getInput(), 32) + "'"); } fieldNames.add(matcher.group()); } // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: fieldNames=" + fieldNames); // Get starting types final List> startJClasses = this.jdb.getJClasses(this.startType); if (startJClasses.isEmpty()) { throw new IllegalArgumentException(errorPrefix + "no model type is an instance of path start type " + this.startType.getName()); } // Initialize cursors final HashSet remainingCursors = new HashSet<>(); startJClasses.stream() .map(jclass -> new Cursor(jclass, jclass.getType(), fieldNames)) .forEach(remainingCursors::add); if (this.startType.isAssignableFrom(UntypedJObject.class)) remainingCursors.add(new Cursor(null, UntypedJObject.class, fieldNames)); // Recursively advance cursors IllegalArgumentException error = null; final HashSet completedCursors = new HashSet<>(); while (!remainingCursors.isEmpty()) { // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: remainingCursors=" + remainingCursors); // Advance outstanding cursors final HashSet previouslyRemainingCursors = new HashSet<>(remainingCursors); remainingCursors.clear(); for (Cursor cursor : previouslyRemainingCursors) { // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: processing remainingCursor " + cursor); // Cursors that have no remaining fields, and for which no target field is required, are completed if (!withTargetField && !cursor.hasMoreFieldNames()) { if (this.log.isTraceEnabled()) this.log.trace("RefPath: remainingCursor " + cursor + " is completed"); completedCursors.add(cursor); continue; } // Try to identify the next field in the path final Set newCursors; try { newCursors = cursor.identifyNextField(withTargetField, lastIsSubField); } catch (IllegalArgumentException e) { if (this.log.isTraceEnabled()) this.log.trace("RefPath: identifyNextField() on " + cursor + " failed: " + e.getMessage()); error = e; continue; } if (this.log.isTraceEnabled()) this.log.trace("RefPath: identifyNextField() on " + cursor + " succeeded: newCursors=" + newCursors); // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: after identifyNextField(), newCursors=" + newCursors); // Remove cursors for which we have reached the end of the path for (Iterator i = newCursors.iterator(); i.hasNext(); ) { final Cursor newCursor = i.next(); if (withTargetField && !newCursor.hasMoreFieldNames()) { if (newCursor.isReverseStep()) // target field cannot be a reverse step throw new IllegalArgumentException("Invalid reference path: missing target field"); if (this.log.isTraceEnabled()) this.log.trace("RefPath: newCursor " + cursor + " is completed"); completedCursors.add(newCursor); i.remove(); } } // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: after identifyNextField(), remaining newCursors=" + newCursors); // Advance the unfinished cursors through the next reference field for (Cursor newCursor : newCursors) { // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: invoking stepThroughReference() on " + newCursor); // "Dereference" the reference field, possibly branching to create multiple new cursors final Set dereferencedCursors; try { dereferencedCursors = newCursor.stepThroughReference(); } catch (IllegalArgumentException e) { error = e; continue; } // Debug if (this.log.isTraceEnabled()) this.log.trace("RefPath: stepThroughReference() returned " + dereferencedCursors); remainingCursors.addAll(dereferencedCursors); } } } // Check for error if (this.log.isTraceEnabled()) this.log.trace("RefPath: remainingCursors=" + remainingCursors + " completedCursors=" + completedCursors); if (completedCursors.isEmpty()) throw error; // Check that all cursors took the same path final ArrayList referenceFieldList = completedCursors.iterator().next().getReferenceFields(); for (Cursor cursor : completedCursors) { if (!cursor.getReferenceFields().equals(referenceFieldList)) { throw new IllegalArgumentException(errorPrefix + "path is ambiguous due to traversal of fields with different types"); } } // Get target field info if (withTargetField) { // Sanity check target field has a consistent storage ID and parent storage ID final Set targetFieldStorageIds = completedCursors.stream() .map(Cursor::getField) .map(JField::getStorageId) .collect(Collectors.toSet()); final Set targetSuperFieldStorageIds = completedCursors.stream() .map(Cursor::getSuperField) .map(sf -> sf != null ? sf.storageId : 0) .collect(Collectors.toSet()); if (targetFieldStorageIds.size() != 1 || targetSuperFieldStorageIds.size() != 1) { throw new IllegalArgumentException(errorPrefix + "the target field `" + fieldNames.pollLast() + "' is ambiguous: " + completedCursors.stream().map(Cursor::getField).collect(Collectors.toSet())); } // Calculate all possible target field types, and whether any target field is indexed final Set> targetFieldTypesSet = completedCursors.stream() .peek(cursor -> this.someTargetFieldIndexed |= cursor.isIndexedSimpleField()) .map(Cursor::getField) .map(JField::getTypeToken) .collect(Collectors.toSet()); // Initialize this.targetFieldTypes = Collections.unmodifiableSet(targetFieldTypesSet); this.targetFieldStorageId = targetFieldStorageIds.iterator().next(); this.targetSuperFieldStorageId = targetSuperFieldStorageIds.iterator().next(); } else { this.targetFieldTypes = null; this.targetFieldStorageId = 0; this.targetSuperFieldStorageId = 0; } // Gather and minimize path types this.pathTypes = new ArrayList<>(referenceFieldList.size() + 1); for (int i = 0; i <= referenceFieldList.size(); i++) { final HashSet> types = new HashSet<>(); for (Cursor cursor : completedCursors) { final Class type = cursor.getPathTypes().get(i); types.add(type); if (this.log.isTraceEnabled()) this.log.trace("RefPath: added " + type + " to pathTypes[" + i + "] from " + cursor); } this.pathTypes.add(ReferencePath.minimizeAndSeal(types)); } assert this.pathTypes.size() == referenceFieldList.size() + 1; // Done this.referenceFieldStorageIds = Ints.toArray(referenceFieldList); this.cursors = completedCursors; // Logging if (this.log.isTraceEnabled()) { this.log.trace("RefPath: DONE: targetFieldStorageId=" + this.targetFieldStorageId + " targetSuperFieldStorageId=" + this.targetSuperFieldStorageId + " targetFieldTypes=" + this.targetFieldTypes + " references=" + referenceFieldList + " cursors=" + this.cursors + " pathTypes=" + this.pathTypes); } } /** * Get the Java type of the object at which this path starts. * *

* If there are zero {@linkplain #getReferenceFields reference fields} in this path, then this will * equal the {@linkplain #getTargetType target object type}, or possibly a super-type if the target * field exists only in a sub-type. * * @return the Java type at which this reference path starts */ public Class getStartType() { return this.startType; } /** * Get the Java type of the object at which this path ends. * *

* The returned type will be as narrow as possible while still including all possibilities, but note that it's * possible for there to be multiple candidates for the "target type", none of which is a sub-type of any other. * To retrieve all such target types, use {@link #getTargetTypes}; this method just invokes * {@link Util#findLowestCommonAncestorOfClasses Util.findLowestCommonAncestorOfClasses()} on the result. * * @return the Java type at which this reference path ends */ public Class getTargetType() { return Util.findLowestCommonAncestorOfClasses(this.getTargetTypes()).getRawType(); } /** * Get the possible Java types of the object at which this path ends. * *

* If there are zero {@linkplain #getReferenceFields reference fields} in this path, then this method will * return only the Java type of the {@linkplain #getStartType starting object type}, or possibly a sub-type * if the target field exists only in a sub-type. * *

* The returned type(s) will be maximally narrow. The set will contain only one element if a unique such * type exists, otherwise it will contain multiple mutually incompatible supertypes of the object types * at which this path ends. * * @return the Java type(s) at which this reference path ends */ public Set> getTargetTypes() { return this.getPathTypes().get(this.pathTypes.size() - 1); } /** * Get the set of possible model object types at each step in the path. * *

* The returned list always has length one more than the length of the array returned by {@link #getReferenceFields}, * such that the set at index i contains all possible types found after the ith step. * The first element contains type(s) that are all assignable to the {@linkplain #getStartType starting type} * (possibly only the starting type if it was already maximally narrow), and the last element contains the * {@linkplain #getTargetTypes target type(s)}. * *

* Each set in the returned list will be maximally narrow: it will contain only one element if a unique such type exists, * otherwise it will contain multiple mutually incompatible supertypes of the object types at that step. * * @return list of possible {@link JClass} corresponding to each step */ public List>> getPathTypes() { return Collections.unmodifiableList(this.pathTypes); } KeyRanges[] getPathKeyRanges() { if (this.pathKeyRanges == null) { final int numJClasses = this.jdb.jclasses.size(); final KeyRanges[] array = new KeyRanges[this.pathTypes.size()]; for (int i = 0; i < this.pathTypes.size(); i++) { final HashSet> jclasses = new HashSet<>(); final Set> types = this.pathTypes.get(i); for (Class type : types) jclasses.addAll(this.jdb.getJClasses(type)); if (jclasses.size() == numJClasses && this.isAnyAssignableFrom(types, UntypedJObject.class)) continue; // no filter needed final ArrayList ranges = new ArrayList<>(jclasses.size()); for (JClass jclass : jclasses) ranges.add(ObjId.getKeyRange(jclass.storageId)); array[i] = new KeyRanges(ranges); } this.pathKeyRanges = array; } return this.pathKeyRanges; } private boolean isAnyAssignableFrom(Iterable> tos, Class from) { for (Class to : tos) { if (to.isAssignableFrom(from)) return true; } return false; } /** * Get the Java type(s) corresponding to the target field at which this path ends, if any. * *

* The returned type(s) will be maximally narrow. The set will contain only one element if a unique such * type exists, otherwise it will contain multiple mutually incompatible supertypes of the object types * at which this path ends. The latter case can only occur when the field is a reference field, and there * are multiple Java model classes compatible with the field's type. * * @return the type of the field at which this reference path ends, or null if this reference path does * not have a target field */ public Set> getTargetFieldTypes() { return this.targetFieldTypes; } /** * Get the storage ID associated with the target field, if any, in the {@linkplain #getTargetType target object type}. * *

* This is just the storage ID of the last field in the path. * * @return the storage ID of the field at which this reference path ends, or zero if this reference path does * not have a target field */ public int getTargetField() { return this.targetFieldStorageId; } /** * Get the storage ID associated with the complex field containing the {@linkplain #getTargetField target field}, * if any, in the case that the target field is a sub-field of a complex field. * * @return target field's complex super-field storage ID, or zero if the target field is not a complex sub-field * or this reference path does not have a target field */ public int getTargetSuperField() { return this.targetSuperFieldStorageId; } /** * Get the storage IDs of the reference fields in this path. * *

* Storage ID's will be negated to indicate reference fields traversed in the reverse direction. * *

* The path may be empty, i.e., zero references are traversed in the path. * *

* Otherwise, the first field is a field in the {@linkplain #getStartType starting object type} and * the last field is field in some object type that refers to the {@linkplain #getTargetType target object type}. * * @return zero or more possibly negated reference field storage IDs */ public int[] getReferenceFields() { return this.referenceFieldStorageIds.clone(); } /** * Get the {@link String} form of the path associated with this instance. */ @Override public String toString() { return this.path; } /** * Identify the named field at the start of the given field name list in the given {@link JClass}. * *

* Either one or two field names will be consumed, depending on whether the field is complex. * * @param jclass starting type * @param fieldNames list of field names * @param lastIsSubField true if the last field can be a complex sub-field but not a complex field, false for the reverse, * or null for don't care * @return resulting {@link JField} * @throws IllegalArgumentException if something is bogus */ private JField findField(JClass jclass, ArrayDeque fieldNames, Boolean lastIsSubField) { // Sanity check Preconditions.checkArgument(jclass != null, "null jclass"); Preconditions.checkArgument(fieldNames != null, "null fieldNames"); Preconditions.checkArgument(!fieldNames.isEmpty(), "empty reference path"); // Logging if (this.log.isTraceEnabled()) { this.log.trace("RefPath.findField(): jclass=" + jclass + " fieldNames=" + fieldNames + " lastIsSubField=" + lastIsSubField); } // Get field name and containing type final String fieldName = fieldNames.removeFirst(); String description = "field `" + fieldName + "' in " + jclass.getType(); // Parse explicit storage ID, if any final int hash = fieldName.indexOf('#'); int explicitStorageId = 0; final String searchName; if (hash != -1) { try { explicitStorageId = Integer.parseInt(fieldName.substring(hash + 1)); } catch (NumberFormatException e) { throw new IllegalArgumentException("invalid field name `" + fieldName + "'"); } searchName = fieldName.substring(0, hash); } else searchName = fieldName; // Find the JField matching 'fieldName' in jclass JField matchingField = jclass.jfieldsByName.get(searchName); if (matchingField == null || (explicitStorageId != 0 && matchingField.storageId != explicitStorageId)) { throw new IllegalArgumentException("there is no field named `" + searchName + "'" + (explicitStorageId != 0 ? " with storage ID " + explicitStorageId : "") + " in " + jclass.getType()); } // Logging if (this.log.isTraceEnabled()) this.log.trace("RefPath.findField(): found field " + matchingField + " in " + jclass.getType()); // Handle complex fields JComplexField superField = null; if (matchingField instanceof JComplexField) { superField = (JComplexField)matchingField; // Logging if (this.log.isTraceEnabled()) this.log.trace("RefPath.findField(): field is a complex field"); // Last field? if (fieldNames.isEmpty()) { // Verify it's OK to end on a complex field if (Boolean.TRUE.equals(lastIsSubField)) { final StringBuilder buf = new StringBuilder(); for (JSimpleField subField : superField.getSubFields()) { if (buf.length() == 0) { buf.append("path may not end on complex ") .append(description) .append("; a sub-field must be specified (e.g., "); } else buf.append(" or "); buf.append('`').append(matchingField.name).append('.').append(subField.name).append('\''); } buf.append(")"); throw new IllegalArgumentException(buf.toString()); } // Done if (this.log.isTraceEnabled()) this.log.trace("RefPath.findField(): ended on complex field; result=" + matchingField); return matchingField; } // Get the specified sub-field final String subFieldName = fieldNames.removeFirst(); description = "sub-field `" + subFieldName + "' of complex " + description; try { matchingField = superField.getSubField(subFieldName); } catch (IllegalArgumentException e) { throw new IllegalArgumentException("invalid " + description + ": " + e.getMessage(), e); } // Logging if (this.log.isTraceEnabled()) { this.log.trace("RefPath.findField(): also stepping through sub-field [" + searchName + "." + subFieldName + "] to reach " + matchingField); } } else if (this.log.isTraceEnabled()) { if (matchingField instanceof JSimpleField) { final JSimpleField simpleField = (JSimpleField)matchingField; this.log.trace("RefPath.findField(): field is a simple field of type " + simpleField.getTypeToken()); } else this.log.trace("RefPath.findField(): field is " + matchingField); } // Verify it's OK to end on a complex sub-field (if that's what happened) if (fieldNames.isEmpty() && superField != null && Boolean.FALSE.equals(lastIsSubField)) { throw new IllegalArgumentException("path may not end on " + description + "; instead, specify the complex field itself"); } // Done if (this.log.isTraceEnabled()) this.log.trace("RefPath.findField(): result=" + matchingField); return matchingField; } // Utility methods private static Set> minimizeAndSeal(Set> types) { final HashSet> minimalTypes = new HashSet<>(types.size()); for (TypeToken typeToken : Util.findLowestCommonAncestorsOfClasses(types)) minimalTypes.add((Class)typeToken.getRawType()); return Collections.unmodifiableSet(minimalTypes); } private static ArrayList> arrayListOf(Class type) { final ArrayList> list = new ArrayList<>(1); list.add(type); return list; } private static ArrayList copyAndAppend(List original, T elem) { final ArrayList list = new ArrayList<>(original.size() + 1); list.addAll(original); list.add(elem); return list; } // Cursor /** * A cursor position in a partially parsed reference path. * *

* An instance points to a {@link JClass}, and optionally to a {@link JField} within that class (or within * some referring class in the case of a reverse step). * It has a list of references it has already traversed, and a list of field names it has yet to traverse. * Traversing a field involves two steps: (a) finding the field via {@link #identifyNextField}, * and (if not done yet) (b) dereferencing that (presumably reference) field via {@link #stepThroughReference}. * Note prior to step (a), {@code this.jfield} is null; prior to step (b), {@code this.jfield} is the field * we just dereferenced in the previous {@link JClass}. * *

* Instances are immutable. */ final class Cursor { private final Logger log = LoggerFactory.getLogger(this.getClass()); private final ArrayList referenceFields = new ArrayList<>(); private final ArrayList> pathTypes = new ArrayList<>(); private final JClass jclass; // null means UntypedJObject private final JField jfield; private final ArrayDeque fieldNames; private final boolean reverseStep; private Cursor(JClass jclass, Class startType, ArrayDeque fieldNames) { this(new ArrayList<>(0), ReferencePath.arrayListOf(startType), jclass, null, fieldNames, false); } private Cursor(ArrayList referenceFields, ArrayList> pathTypes, JClass jclass, JField jfield, ArrayDeque fieldNames, boolean reverseStep) { assert pathTypes.size() == referenceFields.size() + 1; this.referenceFields.addAll(referenceFields); this.pathTypes.addAll(pathTypes); this.jclass = jclass; this.jfield = jfield; this.fieldNames = fieldNames.clone(); this.reverseStep = reverseStep; } public ArrayList getReferenceFields() { return this.referenceFields; } public ArrayList> getPathTypes() { return this.pathTypes; } public int getNumRefs() { return this.referenceFields.size(); } public JClass getJClass() { return this.jclass; } public JField getField() { return this.jfield; } public JComplexField getSuperField() { return this.jfield instanceof JSimpleField ? ((JSimpleField)this.jfield).getParentField() : null; } public boolean hasMoreFieldNames() { return !this.fieldNames.isEmpty(); } public boolean isReverseStep() { return this.reverseStep; } public boolean isIndexedSimpleField() { return this.jfield instanceof JSimpleField && ((JSimpleField)this.jfield).indexed; } /** * Get the type associated with the next reference field. * * @return referred-to type, or null if field is not a reference field * @throws IllegalStateException if field not identified yet */ public Class getFieldTargetType() { Preconditions.checkState(this.jfield != null, "have not yet stepped through field"); return this.reverseStep ? this.jfield.getJClass().type : this.jfield instanceof JReferenceField ? ((JReferenceField)this.jfield).typeToken.getRawType() : null; } /** * Step through the next field name and return the resulting cursor(s). * * @param lastIsSubField true if the last field can be a complex sub-field but not a complex field, false for the reverse, * or null for don't care * @return resulting cursor(s) * @throws IllegalArgumentException if step is bogus */ public Set identifyNextField(boolean withTargetField, Boolean lastIsSubField) { // Sanity check Preconditions.checkArgument(!this.fieldNames.isEmpty(), "empty reference path"); Preconditions.checkState(this.jfield == null, "already stepped through field"); // Get next field name and containing type final ArrayDeque remainingFieldNames = this.fieldNames.clone(); final String step = remainingFieldNames.peekFirst(); // Handle forward vs. reverse final HashSet newCursors = new HashSet<>(3); final Matcher reverseMatcher = Pattern.compile(REV_STEP).matcher(step); if (reverseMatcher.matches()) { // This field cannot be the target field if (withTargetField && remainingFieldNames.isEmpty()) { throw new IllegalArgumentException("Invalid reference path: missing target field after last step `" + step + "'"); } // Get type and field names final String typeName = reverseMatcher.group(2); final String fieldName = reverseMatcher.group(3); // Consume step remainingFieldNames.removeFirst(); if (this.log.isTraceEnabled()) { this.log.trace("RefPath.identifyNextField(): reverse step `" + step + "' -> type `" + typeName + "' field `" + fieldName + "'"); } // Resolve type into all assignable JClass's, plus null if untyped objects are possible final Class type; final SchemaObjectType schemaType = ReferencePath.this.jdb.getNameIndex().getSchemaObjectType(typeName); if (schemaType != null) type = ReferencePath.this.jdb.getJClass(schemaType.getStorageId()).getType(); else { try { type = Class.forName(typeName, false, Thread.currentThread().getContextClassLoader()); } catch (Exception e) { throw new IllegalArgumentException("Unknown type `" + typeName + "' in reference path reverse traversal step `" + step + "'"); } } List> jclasses = ReferencePath.this.jdb.getJClasses(type); if (type.isAssignableFrom(UntypedJObject.class)) { final ArrayList> jclasses2 = new ArrayList<>(jclasses.size() + 1); jclasses2.addAll(jclasses); jclasses2.add(null); jclasses = jclasses2; } // Any types found? if (jclasses.isEmpty()) { throw new IllegalArgumentException("Invalid type `" + typeName + "' in reference path reverse traversal step `" + step + "': no schema model types are assignable to `" + typeName + "'"); } // Find field in each type and create corresponding cursors for (JClass nextJClass : jclasses) { if (nextJClass == null) continue; final ArrayDeque stepFieldNames = new ArrayDeque<>(Arrays.asList(fieldName.split("\\."))); final JField nextJField; try { nextJField = ReferencePath.this.findField(nextJClass, stepFieldNames, true); } catch (IllegalArgumentException e) { continue; } newCursors.add(new Cursor(this.referenceFields, this.pathTypes, nextJClass, nextJField, remainingFieldNames, true)); } // Any fields found? if (newCursors.isEmpty()) { throw new IllegalArgumentException("Invalid reference path reverse traversal step `" + step + "': field `" + fieldName + "' does not exist in " + (schemaType == null ? "any model type assignable to " : "") + "`" + typeName + "'"); } } else { assert Pattern.compile(FWD_STEP).matcher(step).matches() : "`" + step + "' is not a forward step"; // Resolve field if (this.jclass != null) { final JField nextJField = ReferencePath.this.findField(this.jclass, remainingFieldNames, lastIsSubField); newCursors.add(new Cursor(this.referenceFields, this.pathTypes, this.jclass, nextJField, remainingFieldNames, false)); } } // Done if (this.log.isTraceEnabled()) this.log.trace("RefPath.identifyNextField(): result=" + newCursors); return newCursors; } /** * Step through the current reference field to the referred-to types. */ public Set stepThroughReference() { // Logging if (this.log.isTraceEnabled()) this.log.trace("RefPath.stepThroughReference(): this=" + this); // Sanity check Preconditions.checkState(this.jfield != null, "have not yet stepped through field"); assert this.jfield != null; assert this.reverseStep || this.jfield.parent == this.jclass || (this.jfield instanceof JSimpleField && ((JSimpleField)this.jfield).getParentField().parent == this.jclass); Preconditions.checkArgument(this.jfield instanceof JReferenceField, this.jfield + " is not a reference field"); // Append reference field to list final int stepStorageId = this.reverseStep ? -this.jfield.storageId : this.jfield.storageId; final ArrayList newReferenceFields = ReferencePath.copyAndAppend(this.referenceFields, stepStorageId); // Advance through the reference, either forward or inverse final Class targetType = this.reverseStep ? this.jfield.getJClass().type : ((JReferenceField)this.jfield).typeToken.getRawType(); if (this.log.isTraceEnabled()) { this.log.trace("RefPath.stepThroughReference(): targetType=" + targetType + " -> " + ReferencePath.this.jdb.getJClasses(targetType)); } final HashSet newCursors = new HashSet<>(); for (JClass targetJClass : ReferencePath.this.jdb.getJClasses(targetType)) { final ArrayList> newPathTypes = ReferencePath.copyAndAppend(this.pathTypes, targetJClass.getType()); newCursors.add(new Cursor(newReferenceFields, newPathTypes, targetJClass, null, this.fieldNames, false)); } if (targetType.isAssignableFrom(UntypedJObject.class)) { final ArrayList> newPathTypes = ReferencePath.copyAndAppend(this.pathTypes, UntypedJObject.class); newCursors.add(new Cursor(newReferenceFields, newPathTypes, null, null, this.fieldNames, false)); } // Done if (this.log.isTraceEnabled()) this.log.trace("RefPath.stepThroughReference(): result=" + newCursors); return newCursors; } // Object @Override public boolean equals(Object obj) { if (obj == this) return true; if (obj == null || obj.getClass() != this.getClass()) return false; final Cursor that = (Cursor)obj; return this.referenceFields.equals(that.referenceFields) && this.pathTypes.equals(that.pathTypes) && Objects.equals(this.jclass, that.jclass) && Objects.equals(this.jfield, that.jfield) && this.reverseStep == that.reverseStep; } @Override public int hashCode() { return this.referenceFields.hashCode() ^ this.pathTypes.hashCode() ^ Objects.hashCode(this.jclass) ^ Objects.hashCode(this.jfield) ^ (this.reverseStep ? 1 : 0); } @Override public String toString() { return "Cursor" + "[jclass=" + this.jclass + (this.jfield != null ? ",jfield=" + this.jfield : "") + (!this.fieldNames.isEmpty() ? ",fieldNames=" + this.fieldNames : "") + (!this.referenceFields.isEmpty() ? ",refs=" + this.referenceFields : "") + (!this.pathTypes.isEmpty() ? ",pathTypes=" + this.pathTypes : "") + (this.reverseStep ? ",reverseStep" : "") + "]"; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy