001/*
002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others
003 *
004 * This program and the accompanying materials are made available under the
005 * terms of the Eclipse Public License 2.0 which is available at
006 * http://www.eclipse.org/legal/epl-2.0.
007 *
008 * SPDX-License-Identifier: EPL-2.0
009 */
010
011package org.eclipse.golo.runtime.augmentation;
012
013import java.lang.reflect.Method;
014import java.util.Objects;
015
016/**
017 * Encapsulate runtime information about augmentation method resolution.
018 */
019public final class AugmentationMethod implements Comparable<AugmentationMethod> {
020
021  private final AugmentationApplication.Kind kind;
022  private final DefiningModule.Scope scope;
023  private final Class<?> target;
024  private final Method method;
025
026  AugmentationMethod(AugmentationApplication.Kind kind, DefiningModule.Scope scope, Class<?> target, Method method) {
027    this.kind = kind;
028    this.scope = scope;
029    this.target = target;
030    this.method = method;
031  }
032
033  public Method method() {
034    return method;
035  }
036
037  @Override
038  public String toString() {
039    return String.format("AugmentationMethod<%s,%s,%s,%s>",
040        kind, scope, target, method);
041  }
042
043  /**
044   * Compare applications for priority.
045   * <p>
046   * The greater application is the one with the lower priority so that in a stable-sorted list of
047   * application, we always take the first one.
048   * <p>
049   * The natural order defined is:
050   * {@code target specificity < scope < kind}
051   * so that the implied priority is to use the augmentation defined:
052   * <ol>
053   * <li> on the more specific class: to allow to override augmentations by target specificity,
054   * similar to method overriding;
055   * <li> in the more local scope: to allow to override augmentations by defining them locally;
056   * <li> by direct definition instead of named application: to allow to augment a class with a
057   * named augmentation (mixin-like) but override some of the methods by directly defining them;
058   * <li> before any other augmentation: to control augmentation application using the import or
059   * definition order.
060   * </ol>
061   * Moreover, augmentation method with fixed arity are used in preference to variable-arity ones.
062   */
063  @Override
064  public int compareTo(AugmentationMethod other) {
065    if (this.target.isAssignableFrom(other.target) && !this.target.equals(other.target)) {
066      return 1;
067    }
068    if (other.target.isAssignableFrom(this.target) && !this.target.equals(other.target)) {
069      return -1;
070    }
071    if (this.scope != other.scope) {
072      return this.scope.compareTo(other.scope);
073    }
074    if (this.kind != other.kind) {
075      return this.kind.compareTo(other.kind);
076    }
077    if (this.method.isVarArgs() && !other.method.isVarArgs()) {
078      return 1;
079    }
080    if (other.method.isVarArgs() && !this.method.isVarArgs()) {
081      return -1;
082    }
083    return 0;
084  }
085
086  @Override
087  public boolean equals(Object o) {
088    if (o == null) { return false; }
089    if (this == o) { return true; }
090    if (!this.getClass().equals(o.getClass())) { return false; }
091    AugmentationMethod that = (AugmentationMethod) o;
092    return this.kind == that.kind
093      && this.scope == that.scope
094      && this.target.equals(that.target)
095      && this.method.equals(that.method);
096  }
097
098  @Override
099  public int hashCode() {
100    return Objects.hash(kind, scope, target, method);
101  }
102}
103