org.snapscript.tree.dispatch.ScopeDispatcher Maven / Gradle / Ivy
package org.snapscript.tree.dispatch;
import java.util.concurrent.Callable;
import org.snapscript.core.Context;
import org.snapscript.core.InternalStateException;
import org.snapscript.core.Module;
import org.snapscript.core.Result;
import org.snapscript.core.Scope;
import org.snapscript.core.Type;
import org.snapscript.core.TypeExtractor;
import org.snapscript.core.Value;
import org.snapscript.core.ValueType;
import org.snapscript.core.bind.FunctionBinder;
public class ScopeDispatcher implements InvocationDispatcher {
private final Scope object;
private final Scope scope;
public ScopeDispatcher(Scope scope, Object object) {
this.object = (Scope)object;
this.scope = scope;
}
@Override
public Value dispatch(String name, Object... arguments) throws Exception {
Callable match = bind(name, arguments);
if(match == null) {
Module module = scope.getModule();
Context context = module.getContext();
TypeExtractor extractor = context.getExtractor();
Type type = extractor.getType(object);
throw new InternalStateException("Method '" + name + "' not found for '" + type + "'");
}
Result result = match.call();
Object data = result.getValue();
return ValueType.getTransient(data);
}
private Callable bind(String name, Object... arguments) throws Exception {
Module module = scope.getModule();
Context context = module.getContext();
FunctionBinder binder = context.getBinder();
Callable local = binder.bind(scope, object, name, arguments);
if(local == null) {
Callable external = binder.bind(scope, module, name, arguments); // maybe closure should be first
if(external != null) {
return external;
}
Callable closure = binder.bind(object, name, arguments); // closure
if(closure != null) {
return closure;
}
}
return local;
}
}