Skip to content

Commit

Permalink
Introduce ReflectiveIndexAccessor in SpEL
Browse files Browse the repository at this point in the history
  • Loading branch information
sbrannen committed Apr 30, 2024
1 parent 9b85c93 commit d8fadc9
Show file tree
Hide file tree
Showing 5 changed files with 545 additions and 143 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,292 @@
/*
* Copyright 2002-2024 the original author or authors.
*
* 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
*
* https://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 org.springframework.expression.spel.support;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

import org.springframework.asm.MethodVisitor;
import org.springframework.expression.EvaluationContext;
import org.springframework.expression.IndexAccessor;
import org.springframework.expression.TypedValue;
import org.springframework.expression.spel.CodeFlow;
import org.springframework.expression.spel.CompilableIndexAccessor;
import org.springframework.expression.spel.SpelNode;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.ReflectionUtils;

/**
* A flexible {@link org.springframework.expression.IndexAccessor IndexAccessor}
* that uses reflection to read from and optionally write to an indexed structure
* of a target object.
*
* <p>The indexed structure can be accessed through a public read-method (when
* being read) or a public write-method (when being written). The relationship
* between the read-method and write-method is based on a convention that is
* applicable for typical implementations of indexed structures. See the example
* below for details.
*
* <p>{@code ReflectiveIndexAccessor} also implements {@link CompilableIndexAccessor}
* in order to support compilation to bytecode for read access. Note, however,
* that the configured read-method must be invokable via a public class or public
* interface for compilation to succeed.
*
* <h3>Example</h3>
*
* <p>The {@code FruitMap} class (the {@code targetType}) represents a structure
* that is indexed via the {@code Color} enum (the {@code indexType}). The name
* of the read-method is {@code "getFruit"}, and that method returns a
* {@code String} (the {@code indexedValueType}). The name of the write-method
* is {@code "setFruit"}, and that method accepts a {@code Color} enum (the
* {@code indexType}) and a {@code String} (the {@code indexedValueType} which
* must match the return type of the read-method).
*
* <p>A read-only {@code IndexAccessor} for {@code FruitMap} can be created via
* {@code new ReflectiveIndexAccessor(FruitMap.class, Color.class, "getFruit")}.
* With that accessor registered and a {@code FruitMap} registered as a variable
* named {@code #fruitMap}, the SpEL expression {@code #fruitMap[T(example.Color).RED]}
* will evaluate to {@code "cherry"}.
*
* <p>A read-write {@code IndexAccessor} for {@code FruitMap} can be created via
* {@code new ReflectiveIndexAccessor(FruitMap.class, Color.class, "getFruit", "setFruit")}.
* With that accessor registered and a {@code FruitMap} registered as a variable
* named {@code #fruitMap}, the SpEL expression
* {@code #fruitMap[T(example.Color).RED] = 'strawberry'} can be used to change
* the fruit mapping for the color red from {@code "cherry"} to {@code "strawberry"}.
*
* <pre class="code">
* package example;
*
* public enum Color {
* RED, ORANGE, YELLOW
* }</pre>
*
* <pre class="code">
* public class FruitMap {
*
* private final Map&lt;Color, String&gt; map = new HashMap&lt;&gt;();
*
* public FruitMap() {
* this.map.put(Color.RED, "cherry");
* this.map.put(Color.ORANGE, "orange");
* this.map.put(Color.YELLOW, "banana");
* }
*
* public String getFruit(Color color) {
* return this.map.get(color);
* }
*
* public void setFruit(Color color, String fruit) {
* this.map.put(color, fruit);
* }
* }</pre>
*
* @author Sam Brannen
* @since 6.2
* @see IndexAccessor
* @see CompilableIndexAccessor
* @see StandardEvaluationContext
* @see SimpleEvaluationContext
*/
public class ReflectiveIndexAccessor implements CompilableIndexAccessor {

private final Class<?> targetType;

private final Class<?> indexType;

private final Method readMethod;

private final Method readMethodToInvoke;

@Nullable
private final Method writeMethodToInvoke;


/**
* Construct a new {@code ReflectiveIndexAccessor} for read-only access.
* <p>See {@linkplain ReflectiveIndexAccessor class-level documentation} for
* further details and an example.
* @param targetType the type of indexed structure which serves as the target
* of index operations
* @param indexType the type of index used to read from the indexed structure
* @param readMethodName the name of the method used to read from the indexed
* structure
*/
public ReflectiveIndexAccessor(Class<?> targetType, Class<?> indexType, String readMethodName) {
this(targetType, indexType, readMethodName, null);
}

/**
* Construct a new {@code ReflectiveIndexAccessor} for read-write access.
* <p>See {@linkplain ReflectiveIndexAccessor class-level documentation} for
* further details and an example.
* @param targetType the type of indexed structure which serves as the target
* of index operations
* @param indexType the type of index used to read from or write to the indexed
* structure
* @param readMethodName the name of the method used to read from the indexed
* structure
* @param writeMethodName the name of the method used to write to the indexed
* structure, or {@code null} if writing is not supported
*/
public ReflectiveIndexAccessor(Class<?> targetType, Class<?> indexType, String readMethodName,
@Nullable String writeMethodName) {

this.targetType = targetType;
this.indexType = indexType;

try {
this.readMethod = targetType.getMethod(readMethodName, indexType);
}
catch (Exception ex) {
throw new IllegalArgumentException("Failed to find public read-method '%s(%s)' in class '%s'."
.formatted(readMethodName, getName(indexType), getName(targetType)));
}

this.readMethodToInvoke = ClassUtils.getInterfaceMethodIfPossible(this.readMethod, targetType);
ReflectionUtils.makeAccessible(this.readMethodToInvoke);

if (writeMethodName != null) {
Class<?> indexedValueType = this.readMethod.getReturnType();
Method writeMethod;
try {
writeMethod = targetType.getMethod(writeMethodName, indexType, indexedValueType);
}
catch (Exception ex) {
throw new IllegalArgumentException("Failed to find public write-method '%s(%s, %s)' in class '%s'."
.formatted(writeMethodName, getName(indexType), getName(indexedValueType),
getName(targetType)));
}
this.writeMethodToInvoke = ClassUtils.getInterfaceMethodIfPossible(writeMethod, targetType);
ReflectionUtils.makeAccessible(this.writeMethodToInvoke);
}
else {
this.writeMethodToInvoke = null;
}
}


/**
* Return an array containing the {@code targetType} configured via the constructor.
*/
@Override
public Class<?>[] getSpecificTargetClasses() {
return new Class<?>[] { this.targetType };
}

/**
* Return {@code true} if the supplied {@code target} and {@code index} can
* be assigned to the {@code targetType} and {@code indexType} configured
* via the constructor.
* <p>Considers primitive wrapper classes as assignable to the corresponding
* primitive types.
*/
@Override
public boolean canRead(EvaluationContext context, Object target, Object index) {
return (ClassUtils.isAssignableValue(this.targetType, target) &&
ClassUtils.isAssignableValue(this.indexType, index));
}

/**
* Invoke the configured read-method via reflection and return the result
* wrapped in a {@link TypedValue}.
*/
@Override
public TypedValue read(EvaluationContext context, Object target, Object index) {
Object value = ReflectionUtils.invokeMethod(this.readMethodToInvoke, target, index);
return new TypedValue(value);
}

/**
* Return {@code true} if a write-method has been configured and
* {@link #canRead} returns {@code true} for the same arguments.
*/
@Override
public boolean canWrite(EvaluationContext context, Object target, Object index) {
return (this.writeMethodToInvoke != null && canRead(context, target, index));
}

/**
* Invoke the configured write-method via reflection.
* <p>Should only be invoked if {@link #canWrite} returns {@code true} for the
* same arguments.
*/
@Override
public void write(EvaluationContext context, Object target, Object index, @Nullable Object newValue) {
Assert.state(this.writeMethodToInvoke != null, "Write-method cannot be null");
ReflectionUtils.invokeMethod(this.writeMethodToInvoke, target, index, newValue);
}

@Override
public boolean isCompilable() {
return true;
}

/**
* Get the return type of the configured read-method.
*/
@Override
public Class<?> getIndexedValueType() {
return this.readMethod.getReturnType();
}

@Override
public void generateCode(SpelNode index, MethodVisitor mv, CodeFlow cf) {
// Find the public declaring class.
Class<?> publicDeclaringClass = this.readMethodToInvoke.getDeclaringClass();
if (!Modifier.isPublic(publicDeclaringClass.getModifiers())) {
publicDeclaringClass = CodeFlow.findPublicDeclaringClass(this.readMethod);
}
Assert.state(publicDeclaringClass != null && Modifier.isPublic(publicDeclaringClass.getModifiers()),
() -> "Failed to find public declaring class for read-method: " + this.readMethod);

String classDesc = publicDeclaringClass.getName().replace('.', '/');
String lastDesc = cf.lastDescriptor();
boolean isStatic = Modifier.isStatic(this.readMethod.getModifiers());

if (!isStatic) {
if (lastDesc == null || !classDesc.equals(lastDesc.substring(1))) {
// Ensure the current object on the stack is available as the required type.
mv.visitTypeInsn(CHECKCAST, classDesc);
}
}
else if (lastDesc != null) {
// A static method invocation will not consume what is on the stack,
// so it needs to be popped off.
mv.visitInsn(POP);
}

// Push the index onto the stack.
cf.generateCodeForArgument(mv, index, this.indexType);

// Invoke the read-method.
String methodName = this.readMethod.getName();
String methodDescr = CodeFlow.createSignatureDescriptor(this.readMethod);
boolean isInterface = publicDeclaringClass.isInterface();
int opcode = (isStatic ? INVOKESTATIC : isInterface ? INVOKEINTERFACE : INVOKEVIRTUAL);
mv.visitMethodInsn(opcode, classDesc, methodName, methodDescr, isInterface);
}


private static String getName(Class<?> clazz) {
String canonicalName = clazz.getCanonicalName();
return (canonicalName != null ? canonicalName : clazz.getName());
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -23,4 +23,6 @@ public interface PublicInterface {

String getText();

String getFruit(int index);

}
Original file line number Diff line number Diff line change
Expand Up @@ -37,4 +37,12 @@ public String greet(String name) {
return "Super, " + name;
}

public String getIndex(int index) {
return "value-" + index;
}

public String getIndex2(int index) {
return "value-" + (2 * index);
}

}

0 comments on commit d8fadc9

Please sign in to comment.