/* * This file is part of LanternServer, licensed under the MIT License (MIT). * * Copyright (c) LanternPowered <https://www.lanternpowered.org> * Copyright (c) SpongePowered <https://www.spongepowered.org> * Copyright (c) contributors * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the Software), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED AS IS, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package org.lanternpowered.server.event.filter.delegate; import static org.objectweb.asm.Opcodes.ACONST_NULL; import static org.objectweb.asm.Opcodes.ALOAD; import static org.objectweb.asm.Opcodes.ARETURN; import static org.objectweb.asm.Opcodes.ASTORE; import static org.objectweb.asm.Opcodes.CHECKCAST; import static org.objectweb.asm.Opcodes.DUP; import static org.objectweb.asm.Opcodes.IFEQ; import static org.objectweb.asm.Opcodes.IFNE; import static org.objectweb.asm.Opcodes.ILOAD; import static org.objectweb.asm.Opcodes.INSTANCEOF; import static org.objectweb.asm.Opcodes.INVOKEINTERFACE; import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL; import static org.objectweb.asm.Opcodes.ISTORE; import org.objectweb.asm.ClassWriter; import org.objectweb.asm.Label; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Type; import org.spongepowered.api.event.filter.Getter; import org.spongepowered.api.util.Tuple; import java.lang.reflect.Method; import java.lang.reflect.Parameter; import java.util.Optional; public class GetterFilterSourceDelegate implements ParameterFilterSourceDelegate { private final Getter anno; public GetterFilterSourceDelegate(Getter a) { this.anno = a; } @Override public Tuple<Integer, Integer> write(ClassWriter cw, MethodVisitor mv, Method method, Parameter param, int local) { Class<?> targetType = param.getType(); Class<?> eventClass = method.getParameterTypes()[0]; String targetMethod = this.anno.value(); Method targetMethodObj; try { targetMethodObj = eventClass.getMethod(targetMethod); } catch (NoSuchMethodException e) { throw new IllegalArgumentException(String.format("Method %s specified by getter annotation was not found in type %s", targetMethod, eventClass.getName())); } if (targetMethodObj.getParameterCount() != 0) { throw new IllegalArgumentException( "Method " + targetMethodObj.toGenericString() + " specified by getter annotation has non-zero parameter count"); } if (!targetMethodObj.getReturnType().equals(Optional.class) && !targetMethodObj.getReturnType().isAssignableFrom(targetType)) { throw new IllegalArgumentException("Method " + targetMethodObj.toGenericString() + " does not return the correct type. Expected: " + targetType.getName() + " Found: " + targetMethodObj.getReturnType().getName()); } Type returnType = Type.getReturnType(targetMethodObj); Class<?> declaringClass = targetMethodObj.getDeclaringClass(); mv.visitVarInsn(ALOAD, 1); mv.visitTypeInsn(CHECKCAST, Type.getInternalName(declaringClass)); int op = declaringClass.isInterface() ? INVOKEINTERFACE : INVOKEVIRTUAL; mv.visitMethodInsn(op, Type.getInternalName(declaringClass), targetMethod, "()" + returnType.getDescriptor(), declaringClass.isInterface()); int paramLocal = local++; mv.visitVarInsn(returnType.getOpcode(ISTORE), paramLocal); if (!targetMethodObj.getReturnType().isPrimitive()) { Label failure = new Label(); Label success = new Label(); if (Optional.class.equals(targetMethodObj.getReturnType()) && !Optional.class.equals(targetType)) { // Unwrap the optional mv.visitVarInsn(ALOAD, paramLocal); mv.visitMethodInsn(INVOKEVIRTUAL, "java/util/Optional", "isPresent", "()Z", false); mv.visitJumpInsn(IFEQ, failure); mv.visitVarInsn(ALOAD, paramLocal); mv.visitMethodInsn(INVOKEVIRTUAL, "java/util/Optional", "get", "()Ljava/lang/Object;", false); mv.visitInsn(DUP); mv.visitVarInsn(ASTORE, paramLocal); } else { mv.visitVarInsn(returnType.getOpcode(ILOAD), paramLocal); } mv.visitTypeInsn(INSTANCEOF, Type.getInternalName(targetType)); mv.visitJumpInsn(IFNE, success); mv.visitLabel(failure); mv.visitInsn(ACONST_NULL); mv.visitInsn(ARETURN); mv.visitLabel(success); } return new Tuple<>(local, paramLocal); } }