package com.sap.runlet.interpreter.nativestdlib; import behavioral.actions.Statement; import com.sap.ap.metamodel.utils.StringFormatter; import com.sap.runlet.abstractinterpreter.Interpreter; import com.sap.runlet.abstractinterpreter.objects.RunletObject; import com.sap.runlet.abstractinterpreter.repository.Repository; import com.sap.runlet.abstractinterpreter.repository.Snapshot; import com.sap.runlet.interpreter.RunletInterpreter; import com.sap.runlet.interpreter.RunletStackFrame; import com.sap.runlet.interpreter.objects.NativeObject; import com.sap.runlet.interpreter.statements.ReturnInterpreter; import data.classes.Association; import data.classes.AssociationEnd; import data.classes.ClassTypeDefinition; import data.classes.MethodSignature; import data.classes.NativeImpl; import data.classes.Parameter; import data.classes.SapClass; import data.classes.SignatureImplementation; import data.classes.TypeDefinition; import dataaccess.expressions.Expression; public class SnapshotInterpreter implements Interpreter<NativeImpl, SapClass, TypeDefinition, ClassTypeDefinition, Association, AssociationEnd, Statement, Expression, SignatureImplementation, RunletStackFrame, NativeImpl, RunletInterpreter> { private NativeImpl nativeImpl; public SnapshotInterpreter(NativeImpl nativeImpl) { this.nativeImpl = nativeImpl; } @Override public RunletObject<AssociationEnd, TypeDefinition, ClassTypeDefinition> evaluate(RunletInterpreter interpreter) throws SecurityException, IllegalArgumentException { MethodSignature sig = nativeImpl.getImplements_(); Snapshot thiz = (Snapshot) ((NativeObject) interpreter.getCallstack().peek().getThis()).getNativeObject(); if (sig.getName().equals("when")) { return new ReturnInterpreter.ReturnResult(new NativeObject((ClassTypeDefinition) sig.getOutput(), thiz.when(), interpreter.getDefaultSnapshot(), interpreter)); } if (sig.getName().equals("precedes")) { Parameter op = sig.getInput().get(0); boolean result = true; for (RunletObject<AssociationEnd, TypeDefinition, ClassTypeDefinition> argument : interpreter.getCallstack().peek().getValue(op)) { result = result && interpreter.getRepository().precedes(thiz, (Snapshot) ((NativeObject) argument).getNativeObject()); if (!result) { break; } } return new ReturnInterpreter.ReturnResult(new NativeObject((ClassTypeDefinition) sig.getOutput(), result, interpreter .getDefaultSnapshot(), interpreter)); } if (sig.getName().equals("resolve")) { Parameter op = sig.getInput().get(0); Snapshot ss = (Snapshot) ((NativeObject) interpreter.getCallstack().peek().getValue(op)).getNativeObject(); Repository<Association, AssociationEnd, SapClass, TypeDefinition, ClassTypeDefinition> repository = interpreter.getRepository(); Snapshot result = repository.merge(thiz, ss); return new ReturnInterpreter.ReturnResult(new NativeObject((ClassTypeDefinition) sig.getOutput(), result, interpreter .getDefaultSnapshot(), interpreter)); } if (sig.getName().equals("equals")) { Parameter op = sig.getInput().get(0); Snapshot other = (Snapshot) ((NativeObject) interpreter.getCallstack().peek().getValue(op)).getNativeObject(); Repository<Association, AssociationEnd, SapClass, TypeDefinition, ClassTypeDefinition> repository = interpreter.getRepository(); boolean result = repository.testSnapshotEquality(thiz, other); return new ReturnInterpreter.ReturnResult(new NativeObject((ClassTypeDefinition) sig.getOutput(), result, interpreter .getDefaultSnapshot(), interpreter)); } throw new RuntimeException("Unknown native method " + StringFormatter.toString(nativeImpl.getImplements_()) + " and no native interpreter registered for class " + sig.getOwner().getName()); } }