com.caoccao.javet.interop.proxy.ScoredExecutable Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of javet-macos Show documentation
Show all versions of javet-macos Show documentation
Javet is Java + V8 (JAVa + V + EighT). It is an awesome way of embedding Node.js and V8 in Java.
/*
* Copyright (c) 2021-2023. caoccao.com Sam Cao
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.caoccao.javet.interop.proxy;
import com.caoccao.javet.annotations.V8Function;
import com.caoccao.javet.exceptions.JavetException;
import com.caoccao.javet.utils.JavetTypeUtils;
import com.caoccao.javet.utils.JavetVirtualObject;
import com.caoccao.javet.values.V8Value;
import com.caoccao.javet.values.reference.V8ValueFunction;
import com.caoccao.javet.values.reference.V8ValueObject;
import com.caoccao.javet.values.reference.V8ValueProxy;
import java.lang.reflect.*;
import java.util.ArrayList;
import java.util.List;
/**
* The type Scored executable.
*
* @param the type parameter
* @since 0.9.6
*/
final class ScoredExecutable {
/**
* The constant V8_VALUE_CLASS.
*
* @since 0.9.10
*/
private static final Class> V8_VALUE_CLASS = V8Value.class;
/**
* The constant V8_VALUE_FUNCTION_CLASS.
*
* @since 0.9.10
*/
private static final Class> V8_VALUE_FUNCTION_CLASS = V8ValueFunction.class;
/**
* The constant V8_VALUE_OBJECT_CLASS.
*
* @since 0.9.10
*/
private static final Class> V8_VALUE_OBJECT_CLASS = V8ValueObject.class;
/**
* The constant V8_VALUE_PROXY_CLASS.
*
* @since 0.9.10
*/
private static final Class> V8_VALUE_PROXY_CLASS = V8ValueProxy.class;
private final IJavetDynamicObjectFactory dynamicObjectFactory;
private final E executable;
private final Object targetObject;
private final V8ValueObject thisObject;
private JavetVirtualObject[] javetVirtualObjects;
private double score;
/**
* Instantiates a new Scored executable.
*
* @param dynamicObjectFactory the dynamic object factory
* @param targetObject the target object
* @param thisObject this object
* @param executable the executable
* @param javetVirtualObjects the javet virtual objects
* @since 0.9.10
*/
public ScoredExecutable(
IJavetDynamicObjectFactory dynamicObjectFactory,
Object targetObject,
V8ValueObject thisObject,
E executable,
JavetVirtualObject[] javetVirtualObjects) {
this.executable = executable;
this.dynamicObjectFactory = dynamicObjectFactory;
this.javetVirtualObjects = javetVirtualObjects;
this.score = 0;
this.targetObject = targetObject;
this.thisObject = thisObject;
}
/**
* Calculate score double.
*
* @throws JavetException the javet exception
* @since 0.9.10
*/
public void calculateScore() throws JavetException {
final boolean isConstructor = executable instanceof Constructor;
// Max score is 1. Min score is 0.
Class>[] parameterTypes = isConstructor
? ((Constructor>) executable).getParameterTypes()
: ((Method) executable).getParameterTypes();
boolean isExecutableVarArgs = isConstructor
? ((Constructor>) executable).isVarArgs()
: ((Method) executable).isVarArgs();
boolean thisObjectRequired = false;
if (!isConstructor) {
Method method = (Method) executable;
if (method.isAnnotationPresent(V8Function.class)) {
V8Function v8Function = method.getAnnotation(V8Function.class);
thisObjectRequired = v8Function.thisObjectRequired();
}
}
if (thisObjectRequired) {
JavetVirtualObject[] javetVirtualObjectsWithThis = new JavetVirtualObject[javetVirtualObjects.length + 1];
javetVirtualObjectsWithThis[0] = new JavetVirtualObject(thisObject);
System.arraycopy(javetVirtualObjects, 0, javetVirtualObjectsWithThis, 1, javetVirtualObjects.length);
javetVirtualObjects = javetVirtualObjectsWithThis;
}
final int parameterCount = parameterTypes.length;
score = 0;
final int length = javetVirtualObjects.length;
if (length == 0) {
if (isExecutableVarArgs) {
if (parameterCount == 1) {
score = 0.99;
}
} else {
if (parameterCount == 0) {
score = 1;
}
}
} else {
boolean isVarArgs = isExecutableVarArgs && length >= parameterCount - 1;
boolean isFixedArgs = !isExecutableVarArgs && length == parameterCount;
if (isVarArgs || isFixedArgs) {
double totalScore = 0;
final int fixedParameterCount = isExecutableVarArgs ? parameterCount - 1 : parameterCount;
final JavetDynamicProxyFactory dynamicProxyFactory = JavetDynamicProxyFactory.getInstance();
for (int i = 0; i < fixedParameterCount; i++) {
Class> parameterType = parameterTypes[i];
final V8Value v8Value = javetVirtualObjects[i].getV8Value();
final Object object = javetVirtualObjects[i].getObject();
if (v8Value != null) {
if (V8_VALUE_CLASS.isAssignableFrom(parameterType)
&& parameterType.isAssignableFrom(v8Value.getClass())) {
totalScore += 1;
continue;
} else if (object != null && parameterType.isAssignableFrom(object.getClass())) {
totalScore += 0.9;
continue;
} else if (dynamicProxyFactory.isSupportedFunction(parameterType, v8Value)) {
totalScore += 0.95;
continue;
} else if (dynamicProxyFactory.isSupportedObject(parameterType, v8Value)) {
totalScore += 0.85;
continue;
} else if (dynamicObjectFactory != null && dynamicObjectFactory.isSupported(parameterType, v8Value)) {
totalScore += 0.5;
continue;
}
}
if (object == null) {
if (parameterType.isPrimitive()) {
totalScore = 0;
break;
}
totalScore += 0.9;
} else if (parameterType.isAssignableFrom(object.getClass())) {
totalScore += 0.9;
} else if (parameterType.isPrimitive()
&& JavetTypeUtils.toExactPrimitive(parameterType, object) != null) {
totalScore += 0.8;
} else if (JavetTypeUtils.toApproximatePrimitive(parameterType, object) != null) {
totalScore += 0.7;
} else {
totalScore = 0;
break;
}
}
if ((fixedParameterCount == 0 || (fixedParameterCount > 0 && totalScore > 0)) && isVarArgs) {
Class> componentType = parameterTypes[fixedParameterCount].getComponentType();
for (int i = fixedParameterCount; i < length; ++i) {
final V8Value v8Value = javetVirtualObjects[i].getV8Value();
final Object object = javetVirtualObjects[i].getObject();
if (v8Value != null) {
if (V8_VALUE_CLASS.isAssignableFrom(componentType)
&& componentType.isAssignableFrom(v8Value.getClass())) {
totalScore += 0.95;
continue;
} else if (object != null && componentType.isAssignableFrom(object.getClass())) {
totalScore += 0.85;
continue;
} else if (dynamicProxyFactory.isSupportedFunction(componentType, v8Value)) {
totalScore += 0.95;
continue;
} else if (dynamicProxyFactory.isSupportedObject(componentType, v8Value)) {
totalScore += 0.85;
continue;
} else if (dynamicObjectFactory != null && dynamicObjectFactory.isSupported(componentType, v8Value)) {
totalScore += 0.5;
continue;
}
}
if (object == null) {
if (componentType.isPrimitive()) {
totalScore = 0;
break;
} else {
totalScore += 0.85;
}
} else if (componentType.isAssignableFrom(object.getClass())) {
totalScore += 0.85;
} else if (componentType.isPrimitive()
&& JavetTypeUtils.toExactPrimitive(componentType, object) != null) {
totalScore += 0.75;
} else if (JavetTypeUtils.toApproximatePrimitive(componentType, object) != null) {
totalScore += 0.65;
} else {
totalScore = 0;
break;
}
}
}
if (totalScore > 0) {
score = totalScore / length;
if (isConstructor) {
if (targetObject != null &&
((Constructor>) executable).getDeclaringClass() != targetObject.getClass()) {
score *= 0.9;
}
} else {
if (targetObject != null &&
((Method) executable).getDeclaringClass() != targetObject.getClass()) {
score *= 0.9;
}
}
}
}
}
}
/**
* Execute.
*
* @return the object
* @throws Throwable the throwable
* @since 0.9.10
*/
public Object execute() throws Throwable {
final int length = javetVirtualObjects.length;
Object callee = Modifier.isStatic(((Member) executable).getModifiers()) ? null : targetObject;
Class>[] parameterTypes = executable instanceof Constructor ?
((Constructor>) executable).getParameterTypes() : ((Method) executable).getParameterTypes();
final int parameterCount = parameterTypes.length;
boolean isExecutableVarArgs = executable instanceof Constructor ?
((Constructor>) executable).isVarArgs() : ((Method) executable).isVarArgs();
if (length == 0) {
if (isExecutableVarArgs) {
Class> componentType = parameterTypes[parameterCount - 1].getComponentType();
Object varObject = Array.newInstance(componentType, 0);
if (executable instanceof Constructor) {
return ((Constructor>) executable).newInstance(varObject);
} else {
return ((Method) executable).invoke(callee, varObject);
}
} else {
if (executable instanceof Constructor) {
return ((Constructor>) executable).newInstance();
} else {
return ((Method) executable).invoke(callee);
}
}
} else {
List