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

org.jsimpledb.OnVersionChangeScanner Maven / Gradle / Ivy


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

package org.jsimpledb;

import com.google.common.base.Function;
import com.google.common.collect.Maps;
import com.google.common.reflect.TypeToken;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Map;

import org.jsimpledb.annotation.OnVersionChange;
import org.jsimpledb.core.EnumValue;

/**
 * Scans for {@link OnVersionChange @OnVersionChange} annotations.
 */
class OnVersionChangeScanner extends AnnotationScanner
  implements Comparator.MethodInfo> {

    private final TypeToken> byStorageIdType;
    private final TypeToken> byNameType;

    @SuppressWarnings("serial")
    OnVersionChangeScanner(JClass jclass) {
        super(jclass, OnVersionChange.class);
        this.byStorageIdType = new TypeToken>() { };
        this.byNameType = new TypeToken>() { };
    }

    @Override
    protected boolean includeMethod(Method method, OnVersionChange annotation) {

        // Sanity check annotation
        if (annotation.oldVersion() < 0)
            throw new IllegalArgumentException("@" + this.annotationType.getSimpleName() + " has illegal negative oldVersion");
        if (annotation.newVersion() < 0)
            throw new IllegalArgumentException("@" + this.annotationType.getSimpleName() + " has illegal negative newVersion");

        // Check method types
        this.checkNotStatic(method);
        this.checkReturnType(method, void.class);
        int index = 0;
        if (annotation.oldVersion() == 0)
            this.checkParameterType(method, index++, TypeToken.of(int.class));
        if (annotation.newVersion() == 0)
            this.checkParameterType(method, index++, TypeToken.of(int.class));
        final ArrayList> choices = new ArrayList>(2);
        choices.add(this.byStorageIdType);
        choices.add(this.byNameType);
        this.checkParameterType(method, index, choices);

        // Done
        return true;
    }

    @Override
    protected VersionChangeMethodInfo createMethodInfo(Method method, OnVersionChange annotation) {
        return new VersionChangeMethodInfo(method, annotation);
    }

// Comparator

    @Override
    public int compare(MethodInfo info1, MethodInfo info2) {
        final OnVersionChange annotation1 = info1.getAnnotation();
        final OnVersionChange annotation2 = info2.getAnnotation();
        int diff = Boolean.compare(annotation1.oldVersion() == 0, annotation2.oldVersion() == 0);
        if (diff != 0)
            return diff;
        diff = Boolean.compare(annotation1.newVersion() == 0, annotation2.newVersion() == 0);
        if (diff != 0)
            return diff;
        diff = info1.getMethod().getName().compareTo(info2.getMethod().getName());
        if (diff != 0)
            return diff;
        return 0;
    }

// VersionChangeMethodInfo

    class VersionChangeMethodInfo extends MethodInfo {

        private final boolean byName;

        @SuppressWarnings("unchecked")
        VersionChangeMethodInfo(Method method, OnVersionChange annotation) {
            super(method, annotation);
            final List> actuals = OnVersionChangeScanner.this.getParameterTypeTokens(method);
            final TypeToken oldValuesType = actuals.get(actuals.size() - 1);
            if (oldValuesType.equals(OnVersionChangeScanner.this.byStorageIdType))
                this.byName = false;
            else if (oldValuesType.equals(OnVersionChangeScanner.this.byNameType))
                this.byName = true;
            else
                throw new RuntimeException("internal error");
        }

        // Invoke method
        void invoke(JObject jobj, int oldVersion, int newVersion,
          Map oldValuesByStorageId, Map oldValuesByName) {

            // Get method info
            final OnVersionChange annotation = this.getAnnotation();
            final Method method = this.getMethod();

            // Check old & new version numbers
            if ((annotation.oldVersion() != 0 && annotation.oldVersion() != oldVersion)
              || (annotation.newVersion() != 0 && annotation.newVersion() != newVersion))
                return;

            // Determine which map to provide
            Map oldValues = this.byName ? oldValuesByName : oldValuesByStorageId;

            // Map Enum -> EnumValue if so configured TODO: deprecate & remove alwaysUseEnumValue(), change default
            if (this.getAnnotation().alwaysUseEnumValue())
                oldValues = this.mapEnumValues(oldValues);

            // Invoke method
            switch ((annotation.oldVersion() != 0 ? 2 : 0) + (annotation.newVersion() != 0 ? 1 : 0)) {
            case 0:
                Util.invoke(method, jobj, oldVersion, newVersion, oldValues);
                break;
            case 1:
                Util.invoke(method, jobj, oldVersion, oldValues);
                break;
            case 2:
                Util.invoke(method, jobj, newVersion, oldValues);
                break;
            case 3:
            default:
                Util.invoke(method, jobj, oldValues);
                break;
            }
        }

        @Override
        public boolean equals(Object obj) {
            if (obj == this)
                return true;
            if (!super.equals(obj))
                return false;
            final OnVersionChangeScanner.VersionChangeMethodInfo that = (OnVersionChangeScanner.VersionChangeMethodInfo)obj;
            return this.byName == that.byName;
        }

        @Override
        public int hashCode() {
            return super.hashCode()
              ^ (this.byName ? 1 : 0);
        }

        private  Map mapEnumValues(Map map) {
            return Maps.transformValues(map, new Function() {
                @Override
                public Object apply(Object value) {
                    return value instanceof Enum ? new EnumValue((Enum)value) : value;
                }
            });
        }
    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy