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.adapters;
012
013import gololang.FunctionReference;
014
015import java.lang.reflect.Method;
016import java.util.*;
017
018import static java.lang.invoke.MethodType.genericMethodType;
019import static java.lang.reflect.Modifier.*;
020import static java.util.Collections.unmodifiableMap;
021import static java.util.Collections.unmodifiableSet;
022
023public final class AdapterDefinition {
024
025  private final ClassLoader classLoader;
026  private final String name;
027  private final String parent;
028  private final TreeSet<String> interfaces = new TreeSet<>();
029  private final LinkedHashMap<String, FunctionReference> implementations = new LinkedHashMap<>();
030  private final LinkedHashMap<String, FunctionReference> overrides = new LinkedHashMap<>();
031
032  public AdapterDefinition(ClassLoader classLoader, String name, String parent) {
033    this.classLoader = classLoader;
034    this.name = name;
035    this.parent = parent;
036  }
037
038  public ClassLoader getClassLoader() {
039    return classLoader;
040  }
041
042  public String getName() {
043    return name;
044  }
045
046  public String getParent() {
047    return parent;
048  }
049
050  public Set<String> getInterfaces() {
051    return unmodifiableSet(interfaces);
052  }
053
054  public Map<String, FunctionReference> getImplementations() {
055    return unmodifiableMap(implementations);
056  }
057
058  public Map<String, FunctionReference> getOverrides() {
059    return unmodifiableMap(overrides);
060  }
061
062  public AdapterDefinition implementsInterface(String iface) {
063    interfaces.add(iface);
064    return this;
065  }
066
067  public AdapterDefinition implementsMethod(String name, FunctionReference target) throws AdapterDefinitionProblem {
068    checkForImplementation(target);
069    checkStarImplementationType(name, target);
070    implementations.put(name, target);
071    return this;
072  }
073
074  public AdapterDefinition overridesMethod(String name, FunctionReference target) throws AdapterDefinitionProblem {
075    checkForOverriding(target);
076    checkStarOverrideType(name, target);
077    overrides.put(name, target);
078    return this;
079  }
080
081  public boolean hasStarImplementation() {
082    return implementations.containsKey("*");
083  }
084
085  public boolean hasStarOverride() {
086    return overrides.containsKey("*");
087  }
088
089  public AdapterDefinition validate() throws AdapterDefinitionProblem {
090    checkSuperTypesExistence();
091    checkStarConflict();
092    checkMethodsToBeImplemented();
093    checkOverridesImplementationsConflict();
094    checkAllOverridesAndImplementationsExist();
095    return this;
096  }
097
098  private void checkOverridesImplementationsConflict() {
099    for (String key : implementations.keySet()) {
100      if (!"*".equals(key) && overrides.containsKey(key)) {
101        throw new AdapterDefinitionProblem("Conflict: there is both an implementation and an override for method " + key);
102      }
103    }
104  }
105
106  private void checkStarImplementationType(String name, FunctionReference target) {
107    if ("*".equals(name) && !target.type().equals(genericMethodType(2))) {
108      throw new AdapterDefinitionProblem("A * implementation must be of type (Object methodName, Object args)Object: " + target);
109    }
110  }
111
112  private void checkStarOverrideType(String name, FunctionReference target) {
113    if ("*".equals(name) && !target.type().equals(genericMethodType(3))) {
114      throw new AdapterDefinitionProblem("A * override must be of type (Object superHandle, Object methodName, Object args)Object: " + target);
115    }
116  }
117
118  private void checkAllOverridesAndImplementationsExist() {
119    try {
120      Class<?> parentClass = Class.forName(parent, true, classLoader);
121      HashSet<String> canBeOverridden = new HashSet<>();
122      for (Method method : parentClass.getMethods()) {
123        if (!isStatic(method.getModifiers())) {
124          canBeOverridden.add(method.getName());
125        }
126      }
127      for (Method method : parentClass.getDeclaredMethods()) {
128        if (!isStatic(method.getModifiers())) {
129          canBeOverridden.add(method.getName());
130        }
131      }
132      for (Method method : parentClass.getMethods()) {
133        if (!isStatic(method.getModifiers())) {
134          canBeOverridden.add(method.getName());
135        }
136      }
137      for (String key : overrides.keySet()) {
138        if (!"*".equals(key) && !canBeOverridden.contains(key)) {
139          throw new AdapterDefinitionProblem("There is no method named " + key + " to be overridden in " + parentClass);
140        }
141      }
142      for (String iface : interfaces) {
143        for (Method method : Class.forName(iface, true, classLoader).getMethods()) {
144          canBeOverridden.add(method.getName());
145        }
146      }
147      for (String key : implementations.keySet()) {
148        if (!"*".equals(key) && !canBeOverridden.contains(key)) {
149          throw new AdapterDefinitionProblem("There is no method named " + key + " to be implemented in " + parentClass + " or interfaces " + interfaces);
150        }
151      }
152    } catch (ClassNotFoundException e) {
153      throw new AdapterDefinitionProblem(e);
154    }
155  }
156
157  private Set<Method> abstractMethodsIn(Class<?> klass) {
158    LinkedHashSet<Method> abstractMethods = new LinkedHashSet<>();
159    for (Method method : klass.getMethods()) {
160      if (isAbstract(method.getModifiers())) {
161        abstractMethods.add(method);
162      }
163    }
164    for (Method method : klass.getDeclaredMethods()) {
165      if (isAbstract(method.getModifiers())) {
166        abstractMethods.add(method);
167      }
168    }
169    return abstractMethods;
170  }
171
172  private void checkMethodsToBeImplemented() {
173    try {
174      LinkedHashSet<Method> abstractMethods = new LinkedHashSet<>();
175      abstractMethods.addAll(abstractMethodsIn(Class.forName(parent, true, classLoader)));
176      for (String iface : interfaces) {
177        abstractMethods.addAll(abstractMethodsIn(Class.forName(iface, true, classLoader)));
178      }
179      for (Method abstractMethod : abstractMethods) {
180        String name = abstractMethod.getName();
181        if (!implementations.containsKey(name) && !hasStarImplementation()) {
182          throw new AdapterDefinitionProblem("There is no implementation or override for: " + abstractMethod);
183        }
184        if (implementations.containsKey(name)) {
185          FunctionReference target = implementations.get(name);
186          if (argsDifferForImplementation(abstractMethod, target) || varargsMismatch(abstractMethod, target)) {
187            throw new AdapterDefinitionProblem("Types do not match to implement " + abstractMethod + " with " + target);
188          }
189        }
190        if (overrides.containsKey(name)) {
191          FunctionReference target = overrides.get(name);
192          if (argsDifferForOverride(abstractMethod, target) || varargsMismatch(abstractMethod, target)) {
193            throw new AdapterDefinitionProblem("Types do not match to implement " + abstractMethod + " with " + target);
194          }
195        }
196      }
197    } catch (ClassNotFoundException e) {
198      throw new AdapterDefinitionProblem(e);
199    }
200  }
201
202  private boolean varargsMismatch(Method abstractMethod, FunctionReference target) {
203    return abstractMethod.isVarArgs() != target.isVarargsCollector();
204  }
205
206  private boolean argsDifferForImplementation(Method abstractMethod, FunctionReference target) {
207    return (target.type().parameterCount() - 1 != abstractMethod.getParameterTypes().length);
208  }
209
210  private boolean argsDifferForOverride(Method abstractMethod, FunctionReference target) {
211    return (target.type().parameterCount() - 2 != abstractMethod.getParameterTypes().length);
212  }
213
214  private void checkStarConflict() {
215    if (hasStarImplementation() && hasStarOverride()) {
216      throw new AdapterDefinitionProblem("Having both a star implementation and a star override is forbidden.");
217    }
218  }
219
220  private void checkSuperTypesExistence() {
221    try {
222      Class<?> parentClass = Class.forName(parent, true, classLoader);
223      if (parentClass.isInterface()) {
224        throw new AdapterDefinitionProblem("The parent class cannot be an interface: " + parentClass.getName());
225      }
226      if (isFinal(parentClass.getModifiers())) {
227        throw new AdapterDefinitionProblem("The parent class is final: " + parentClass.getName());
228      }
229      for (String iface : interfaces) {
230        Class.forName(iface, true, classLoader);
231      }
232    } catch (ClassNotFoundException e) {
233      throw new AdapterDefinitionProblem(e);
234    }
235  }
236
237  private void checkForImplementation(FunctionReference target) throws AdapterDefinitionProblem {
238    if (target.type().parameterCount() < 1) {
239      throw new AdapterDefinitionProblem("An implemented method target must take at least 1 argument (the receiver): " + target);
240    }
241  }
242
243  private void checkForOverriding(FunctionReference target) throws AdapterDefinitionProblem {
244    if (target.type().parameterCount() < 2) {
245      throw new AdapterDefinitionProblem("An overriden method target must take at least 2 arguments (the 'super' function reference followed by the receiver): " + target);
246    }
247  }
248}