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;
012
013import java.lang.invoke.MethodType;
014import java.lang.invoke.MethodHandle;
015import java.lang.reflect.Method;
016
017import static java.lang.reflect.Modifier.*;
018
019/**
020 * Encapsulate informations about a runtime method call.
021 */
022public class MethodInvocation {
023
024  private final String name;
025  private final Class<?> receiverClass;
026  private final Object[] arguments;
027  private final int arity;
028  private final String[] argumentNames;
029  private final MethodType type;
030
031  MethodInvocation(String name, MethodType type, Object[] args, String[] argNames) {
032    this.name = name;
033    this.receiverClass = args[0].getClass();
034    this.arguments = args;
035    this.arity = args.length;
036    this.type = type;
037    this.argumentNames = argNames;
038  }
039
040  public String name() {
041    return name;
042  }
043
044  public Class<?> receiverClass() {
045    return receiverClass;
046  }
047
048  public Object[] arguments() {
049    return arguments;
050  }
051
052  public int arity() {
053    return arity;
054  }
055
056  public String[] argumentNames() {
057    return argumentNames;
058  }
059
060  public MethodType type() {
061    return type;
062  }
063
064  private boolean isLastArgumentAnArray() {
065    return arity > 0
066      && arguments.length == arity
067      && arguments[arity - 1] instanceof Object[];
068  }
069
070  public MethodHandle coerce(MethodHandle target) {
071    if (target.isVarargsCollector() && isLastArgumentAnArray()) {
072      return target.asFixedArity().asType(type);
073    }
074    return target.asType(type);
075  }
076
077  public boolean match(Method method) {
078    return method.getName().equals(name)
079      && isPublic(method.getModifiers())
080      && !isAbstract(method.getModifiers())
081      && TypeMatching.argumentsMatch(method, arguments);
082  }
083
084  /**
085   * returns a new invocation having the given name.
086   */
087  MethodInvocation withName(String newName) {
088    return new MethodInvocation(newName, type, arguments, argumentNames);
089  }
090
091  @Override
092  public String toString() {
093    return name + type + java.util.Arrays.asList(arguments);
094  }
095
096
097}