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

io.permazen.OnChangeScanner 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.Converter;
import com.google.common.reflect.TypeToken;

import io.permazen.annotation.OnChange;
import io.permazen.change.FieldChange;
import io.permazen.change.ListFieldAdd;
import io.permazen.change.ListFieldClear;
import io.permazen.change.ListFieldRemove;
import io.permazen.change.ListFieldReplace;
import io.permazen.change.MapFieldAdd;
import io.permazen.change.MapFieldClear;
import io.permazen.change.MapFieldRemove;
import io.permazen.change.MapFieldReplace;
import io.permazen.change.SetFieldAdd;
import io.permazen.change.SetFieldClear;
import io.permazen.change.SetFieldRemove;
import io.permazen.change.SimpleFieldChange;
import io.permazen.core.Field;
import io.permazen.core.ListField;
import io.permazen.core.MapField;
import io.permazen.core.ObjId;
import io.permazen.core.SetField;
import io.permazen.core.SimpleField;
import io.permazen.core.Transaction;
import io.permazen.core.TypeNotInSchemaVersionException;
import io.permazen.core.UnknownFieldException;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.NavigableSet;

/**
 * Scans for {@link OnChange @OnChange} annotations.
 */
class OnChangeScanner extends AnnotationScanner {

    OnChangeScanner(JClass jclass) {
        super(jclass, OnChange.class);
    }

    @Override
    protected boolean includeMethod(Method method, OnChange annotation) {
        this.checkReturnType(method, void.class);
        if (this.getParameterTypeTokens(method).size() > 1)
            throw new IllegalArgumentException(this.getErrorPrefix(method) + "method is required to take zero or one parameter");
        return true;                                    // we do further parameter type check in ChangeMethodInfo
    }

    @Override
    protected ChangeMethodInfo createMethodInfo(Method method, OnChange annotation) {
        return new ChangeMethodInfo(method, annotation);
    }

// ChangeMethodInfo

    class ChangeMethodInfo extends MethodInfo implements AllChangesListener {

        final HashSet paths;
        final Class[] genericTypes;      // derived from this.method, so there's no need to include it in equals() or hashCode()

        ChangeMethodInfo(Method method, OnChange annotation) {
            super(method, annotation);

            // Get database
            final Permazen jdb = OnChangeScanner.this.jclass.jdb;

            // Get start type
            Class startType = method.getDeclaringClass();
            if (annotation.startType() != void.class) {
                if ((method.getModifiers() & Modifier.STATIC) == 0) {
                    throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method)
                      + "startType() may only be used for annotations on static methods");
                }
                if (annotation.startType().isPrimitive() || annotation.startType().isArray()) {
                    throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method)
                      + "invalid startType() " + annotation.startType());
                }
                startType = annotation.startType();
            }

            // Initialize path list
            final List unexpandedPathList = new ArrayList<>(Arrays.asList(annotation.value()));

            // An empty list is the same as @OnChange("*")
            if (unexpandedPathList.isEmpty())
                unexpandedPathList.add("*");

            // Replace paths ending in "*" them with an iteration of all fields in the corresponding type(s)
            final List expandedPathList = new ArrayList<>(unexpandedPathList.size());
            final HashSet expandedPathWasWildcard = new HashSet<>();
            for (String unexpandedPath : unexpandedPathList) {

                // Check for immediate wildcard: "*"
                if (unexpandedPath.equals("*")) {

                    // Replace path with non-wildcard paths for every field in the start type
                    for (JClass jclass : jdb.getJClasses(startType)) {
                        for (JField jfield : jclass.jfields.values()) {
                            expandedPathWasWildcard.add(expandedPathList.size());
                            expandedPathList.add(jfield.name + "#" + jfield.storageId);
                        }
                    }
                    continue;
                }

                // Check for a reference path ending with wildcard: "foo.bar.*"
                if (unexpandedPath.length() > 2 && unexpandedPath.endsWith(".*")) {

                    // Parse the reference path up to the wildcard
                    final String prefixPath = unexpandedPath.substring(0, unexpandedPath.length() - 2);
                    final ReferencePath prefixReferencePath;
                    try {
                        prefixReferencePath = jdb.parseReferencePath(startType, prefixPath, false, true);
                    } catch (IllegalArgumentException e) {
                        throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method) + e.getMessage(), e);
                    }

                    // Create a new non-wildcard path from each field in the dereferenced object types
                    final HashSet expandedCursors = new HashSet<>(prefixReferencePath.cursors.size() * 2);
                    for (ReferencePath.Cursor cursor : prefixReferencePath.cursors) {
                        for (JField jfield : cursor.getJClass().jfields.values()) {
                            if (!jfield.supportsChangeNotifications())
                                continue;
                            expandedPathWasWildcard.add(expandedPathList.size());
                            expandedPathList.add(prefixPath + "." + jfield.name + "#" + jfield.storageId);
                        }
                    }
                    continue;
                }

                // Non-wildcard path
                expandedPathList.add(unexpandedPath);
            }

            // Get method parameter type (generic and raw), if any, and extract generic types from the FieldChange parameter
            final TypeToken genericParameterType;
            final Class rawParameterType;
            switch (method.getParameterTypes().length) {
            case 1:
                rawParameterType = method.getParameterTypes()[0];
                genericParameterType = OnChangeScanner.this.getParameterTypeTokens(method).get(0);
                final Type firstParameterType = method.getGenericParameterTypes()[0];
                if (firstParameterType instanceof ParameterizedType) {
                    final ArrayList> genericTypeList = new ArrayList<>(3);
                    for (Type type : ((ParameterizedType)firstParameterType).getActualTypeArguments())
                        genericTypeList.add(TypeToken.of(type).getRawType());
                    this.genericTypes = genericTypeList.toArray(new Class[genericTypeList.size()]);
                } else
                    this.genericTypes = new Class[] { rawParameterType };
                break;
            case 0:
                rawParameterType = null;
                genericParameterType = null;
                this.genericTypes = null;
                break;
            default:
                throw new RuntimeException("internal error");
            }

            // Parse reference paths
            boolean anyFieldsFound = false;
            this.paths = new HashSet<>(expandedPathList.size());
            for (int i = 0; i < expandedPathList.size(); i++) {
                final String stringPath = expandedPathList.get(i);
                final boolean wildcard = expandedPathWasWildcard.contains(i);           // path was auto-generated from a wildcard

                // Parse reference path
                final ReferencePath path;
                try {
                    path = jdb.parseReferencePath(startType, stringPath, true, false);
                } catch (IllegalArgumentException e) {
                    throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method) + e.getMessage(), e);
                }

                // Validate the parameter type against the types of possible change events
                if (rawParameterType != null) {

                    // Get all possible (concrete) change types emitted by the target field
                    final ArrayList> possibleChangeTypes = new ArrayList>();
                    for (ReferencePath.Cursor cursor : path.cursors) {
                        try {
                            cursor.getField().addChangeParameterTypes(possibleChangeTypes, cursor.getJClass().getType());
                        } catch (UnsupportedOperationException e) {
                            if (wildcard)
                                continue;
                        }
                        anyFieldsFound = true;
                    }
                    if (possibleChangeTypes.isEmpty() && !wildcard) {
                        throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method) + "path `" + stringPath
                          + "' is invalid because change notifications are not supported for any target field");
                    }

                    // Check whether method parameter type accepts as least one of them; it must do so consistently raw vs. generic
                    boolean anyChangeMatch = false;
                    for (TypeToken possibleChangeType : possibleChangeTypes) {
                        final boolean matchesGeneric = genericParameterType.isSupertypeOf(possibleChangeType);
                        final boolean matchesRaw = rawParameterType.isAssignableFrom(possibleChangeType.getRawType());
                        assert !matchesGeneric || matchesRaw;
                        if (matchesGeneric != matchesRaw) {
                            throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method)
                              + "parameter type " + genericParameterType + " will match change events of type "
                              + possibleChangeType + " from field `" + stringPath + "' at runtime due to type erasure,"
                              + " but its generic type is does not match " + possibleChangeType
                              + "; try narrowing or widening the parameter type, keeping it compatible with "
                              + (possibleChangeTypes.size() != 1 ?
                                  "one or more of: " + possibleChangeTypes : possibleChangeTypes.get(0)));
                        }
                        if (matchesGeneric) {
                            anyChangeMatch = true;
                            break;
                        }
                    }

                    // If not wildcard match, then at least one change type must match method
                    if (!anyChangeMatch) {
                        if (wildcard)
                            continue;
                        throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method) + "path `" + stringPath
                          + "' is invalid because no changes emitted by the target field match the method's"
                          + " parameter type " + genericParameterType + "; the emitted change type is "
                          + (possibleChangeTypes.size() != 1 ? "one of: " + possibleChangeTypes : possibleChangeTypes.get(0)));
                    }
                }

                // Add path
                this.paths.add(path);
            }

            // No matching fields?
            if (this.paths.isEmpty()) {                                                         // must be wildcard
                if (!anyFieldsFound) {
                    throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method)
                      + "there are no fields that will generate change events");
                }
                throw new IllegalArgumentException(OnChangeScanner.this.getErrorPrefix(method) + "no changes emitted by any field"
                  + " will match the method's parameter type " + genericParameterType);
            }
        }

        // Register listeners for this method
        void registerChangeListener(Transaction tx) {
            for (ReferencePath path : this.paths)
                tx.addFieldChangeListener(path.targetFieldStorageId, path.getReferenceFields(), path.getPathKeyRanges(), this);
        }

        @Override
        public boolean equals(Object obj) {
            if (obj == this)
                return true;
            if (!super.equals(obj))
                return false;
            final OnChangeScanner.ChangeMethodInfo that = (OnChangeScanner.ChangeMethodInfo)obj;
            return this.paths.equals(that.paths);
        }

        @Override
        public int hashCode() {
            return super.hashCode() ^ this.paths.hashCode();
        }

    // SimpleFieldChangeListener

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onSimpleFieldChange(Transaction tx, ObjId id,
          SimpleField field, int[] path, NavigableSet referrers, T oldValue, T newValue) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JSimpleField jfield = this.getJField(jtx, id, field, JSimpleField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object joldValue = this.convertCoreValue(jtx, jfield, oldValue);
            final Object jnewValue = this.convertCoreValue(jtx, jfield, newValue);
            final JObject jobj = this.checkTypes(jtx, SimpleFieldChange.class, id, joldValue, jnewValue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new SimpleFieldChange(jobj, field.getStorageId(), field.getName(), joldValue, jnewValue));
        }

    // SetFieldChangeListener

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onSetFieldAdd(Transaction tx, ObjId id,
          SetField field, int[] path, NavigableSet referrers, E value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JSetField jfield = this.getJField(jtx, id, field, JSetField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jvalue = this.convertCoreValue(jtx, jfield.elementField, value);
            final JObject jobj = this.checkTypes(jtx, SetFieldAdd.class, id, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new SetFieldAdd(jobj, jfield.storageId, jfield.name, jvalue));
        }

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onSetFieldRemove(Transaction tx, ObjId id,
          SetField field, int[] path, NavigableSet referrers, E value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JSetField jfield = this.getJField(jtx, id, field, JSetField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jvalue = this.convertCoreValue(jtx, jfield.elementField, value);
            final JObject jobj = this.checkTypes(jtx, SetFieldRemove.class, id, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new SetFieldRemove(jobj, jfield.storageId, jfield.name, jvalue));
        }

        @Override
        public void onSetFieldClear(Transaction tx, ObjId id, SetField field, int[] path, NavigableSet referrers) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JSetField jfield = this.getJField(jtx, id, field, JSetField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final JObject jobj = this.checkTypes(jtx, SetFieldClear.class, id);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new SetFieldClear<>(jobj, jfield.storageId, jfield.name));
        }

    // ListFieldChangeListener

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onListFieldAdd(Transaction tx, ObjId id,
          ListField field, int[] path, NavigableSet referrers, int index, E value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JListField jfield = this.getJField(jtx, id, field, JListField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jvalue = this.convertCoreValue(jtx, jfield.elementField, value);
            final JObject jobj = this.checkTypes(jtx, ListFieldAdd.class, id, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new ListFieldAdd(jobj, jfield.storageId, jfield.name, index, jvalue));
        }

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onListFieldRemove(Transaction tx, ObjId id,
          ListField field, int[] path, NavigableSet referrers, int index, E value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JListField jfield = this.getJField(jtx, id, field, JListField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jvalue = this.convertCoreValue(jtx, jfield.elementField, value);
            final JObject jobj = this.checkTypes(jtx, ListFieldRemove.class, id, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new ListFieldRemove(jobj, jfield.storageId, jfield.name, index, jvalue));
        }

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onListFieldReplace(Transaction tx, ObjId id,
          ListField field, int[] path, NavigableSet referrers, int index, E oldValue, E newValue) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JListField jfield = this.getJField(jtx, id, field, JListField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object joldValue = this.convertCoreValue(jtx, jfield.elementField, oldValue);
            final Object jnewValue = this.convertCoreValue(jtx, jfield.elementField, newValue);
            final JObject jobj = this.checkTypes(jtx, ListFieldReplace.class, id, joldValue, jnewValue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers,
              new ListFieldReplace(jobj, jfield.storageId, jfield.name, index, joldValue, jnewValue));
        }

        @Override
        public void onListFieldClear(Transaction tx, ObjId id, ListField field, int[] path, NavigableSet referrers) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JListField jfield = this.getJField(jtx, id, field, JListField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final JObject jobj = this.checkTypes(jtx, ListFieldClear.class, id);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new ListFieldClear<>(jobj, jfield.storageId, jfield.name));
        }

    // MapFieldChangeListener

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onMapFieldAdd(Transaction tx, ObjId id,
          MapField field, int[] path, NavigableSet referrers, K key, V value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JMapField jfield = this.getJField(jtx, id, field, JMapField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jkey = this.convertCoreValue(jtx, jfield.keyField, key);
            final Object jvalue = this.convertCoreValue(jtx, jfield.valueField, value);
            final JObject jobj = this.checkTypes(jtx, MapFieldAdd.class, id, jkey, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new MapFieldAdd(jobj, jfield.storageId, jfield.name, jkey, jvalue));
        }

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onMapFieldRemove(Transaction tx, ObjId id,
          MapField field, int[] path, NavigableSet referrers, K key, V value) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JMapField jfield = this.getJField(jtx, id, field, JMapField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jkey = this.convertCoreValue(jtx, jfield.keyField, key);
            final Object jvalue = this.convertCoreValue(jtx, jfield.valueField, value);
            final JObject jobj = this.checkTypes(jtx, MapFieldRemove.class, id, jkey, jvalue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new MapFieldRemove(jobj, jfield.storageId, jfield.name, jkey, jvalue));
        }

        @Override
        @SuppressWarnings({ "unchecked", "rawtypes" })
        public  void onMapFieldReplace(Transaction tx, ObjId id,
          MapField field, int[] path, NavigableSet referrers, K key, V oldValue, V newValue) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JMapField jfield = this.getJField(jtx, id, field, JMapField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final Object jkey = this.convertCoreValue(jtx, jfield.keyField, key);
            final Object joldValue = this.convertCoreValue(jtx, jfield.valueField, oldValue);
            final Object jnewValue = this.convertCoreValue(jtx, jfield.valueField, newValue);
            final JObject jobj = this.checkTypes(jtx, MapFieldReplace.class, id, jkey, joldValue, jnewValue);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers,
              new MapFieldReplace(jobj, jfield.storageId, jfield.name, jkey, joldValue, jnewValue));
        }

        @Override
        public void onMapFieldClear(Transaction tx, ObjId id, MapField field, int[] path, NavigableSet referrers) {
            final JTransaction jtx = (JTransaction)tx.getUserObject();
            assert jtx != null && jtx.tx == tx;
            final JMapField jfield = this.getJField(jtx, id, field, JMapField.class);
            if (jfield == null)
                return;
            if (this.genericTypes == null) {
                this.invoke(jtx, referrers);
                return;
            }
            final JObject jobj = this.checkTypes(jtx, MapFieldClear.class, id);
            if (jobj == null)
                return;
            this.invoke(jtx, referrers, new MapFieldClear<>(jobj, jfield.storageId, jfield.name));
        }

    // Internal methods

        private  T getJField(JTransaction jtx, ObjId id, Field field, Class type) {
            try {
                return jtx.jdb.getJField(id, field.getStorageId(), type);
            } catch (TypeNotInSchemaVersionException | UnknownFieldException e) {
                return null;        // somebody changed the field directly via the core API without first upgrading the object
            }
        }

        @SuppressWarnings({ "rawtypes", "unchecked" })
        private Object convertCoreValue(JTransaction jtx, JSimpleField jfield, Object value) {
            final Converter converter = jfield.getConverter(jtx);
            return converter != null ? converter.convert(value) : value;
        }

        private JObject checkTypes(JTransaction jtx, Class*/> changeType, ObjId id, Object... values) {

            // Check method parameter type
            final Method method = this.getMethod();
            if (!method.getParameterTypes()[0].isAssignableFrom(changeType))
                return null;

            // Check first generic type parameter which is the JObject corresponding to id
            final JObject jobj = jtx.get(id);
            if (!this.genericTypes[0].isInstance(jobj))
                return null;

            // Check other generic type parameter(s)
            for (int i = 1; i < this.genericTypes.length; i++) {
                final Object value = values[Math.min(i, values.length) - 1];
                if (value != null && !this.genericTypes[i].isInstance(value))
                    return null;
            }

            // OK types agree
            return jobj;
        }

        // Used when @OnChange method takes zero parameters
        private void invoke(JTransaction jtx, NavigableSet referrers) {
            final Method method = this.getMethod();
            if ((method.getModifiers() & Modifier.STATIC) != 0)
                Util.invoke(method, null);
            else {
                for (ObjId id : referrers) {
                    final JObject target = jtx.get(id);             // type of 'id' should always be found

                    // Avoid invoking subclass's @OnChange method on superclass instance;
                    // this can happen when the field is in superclass but wildcard @OnChange is in the subclass
                    if (method.getDeclaringClass().isInstance(target))
                        Util.invoke(method, target);
                }
            }
        }

        // Used when @OnChange method takes one parameter
        private void invoke(JTransaction jtx, NavigableSet referrers, FieldChange change) {
            assert change != null;
            final Method method = this.getMethod();
            if ((method.getModifiers() & Modifier.STATIC) != 0)
                Util.invoke(method, null, change);
            else {
                for (ObjId id : referrers) {
                    final JObject target = jtx.get(id);             // type of 'id' should always be found

                    // Avoid invoking subclass's @OnChange method on superclass instance;
                    // this can happen when the field is in superclass but wildcard @OnChange is in the subclass
                    if (method.getDeclaringClass().isInstance(target))
                        Util.invoke(method, target, change);
                }
            }
        }
    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy