/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.el.lang;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.lang.reflect.Method;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import jakarta.el.FunctionMapper;
import org.apache.el.util.MessageFactory;
import org.apache.el.util.ReflectionUtil;
/**
* @author Jacob Hookom [jacob@hookom.net]
*/
public class FunctionMapperImpl
extends FunctionMapper
implements
Externalizable {
private static final long serialVersionUID = 1L;
protected ConcurrentMap<String,
Function> functions =
new ConcurrentHashMap<>();
/*
* (non-Javadoc)
*
* @see jakarta.el.FunctionMapper#resolveFunction(String, String)
*/
@Override
public Method resolveFunction(String prefix, String localName) {
Function f =
this.functions.get(prefix +
":" + localName);
if (f ==
null) {
return null;
}
return f.getMethod();
}
@Override
public void mapFunction(String prefix, String localName, Method m) {
String key = prefix +
":" + localName;
if (m ==
null) {
functions.remove(key);
}
else {
Function f =
new Function(prefix, localName, m);
functions.put(key, f);
}
}
/*
* (non-Javadoc)
*
* @see java.io.Externalizable#writeExternal(java.io.ObjectOutput)
*/
@Override
public void writeExternal(ObjectOutput out)
throws IOException {
out.writeObject(
this.functions);
}
/*
* (non-Javadoc)
*
* @see java.io.Externalizable#readExternal(java.io.ObjectInput)
*/
@SuppressWarnings(
"unchecked")
@Override
public void readExternal(ObjectInput in)
throws IOException,
ClassNotFoundException {
this.functions = (ConcurrentMap<String,
Function>) in.readObject();
}
public static class Function implements Externalizable {
protected transient Method m;
protected String owner;
protected String name;
protected String[] types;
protected String prefix;
protected String localName;
public Function(String prefix, String localName, Method m) {
if (localName ==
null) {
throw new NullPointerException(MessageFactory.get(
"error.nullLocalName"));
}
if (m ==
null) {
throw new NullPointerException(MessageFactory.get(
"error.nullMethod"));
}
this.prefix = prefix;
this.localName = localName;
this.m = m;
}
public Function() {
// for serialization
}
/*
* (non-Javadoc)
*
* @see java.io.Externalizable#writeExternal(java.io.ObjectOutput)
*/
@Override
public void writeExternal(ObjectOutput out)
throws IOException {
out.writeUTF((
this.prefix !=
null) ?
this.prefix :
"");
out.writeUTF(
this.localName);
if (
this.owner !=
null &&
this.name !=
null &&
this.types !=
null) {
out.writeUTF(
this.owner);
out.writeUTF(
this.name);
out.writeObject(
this.types);
}
else {
out.writeUTF(
this.m.getDeclaringClass().getName());
out.writeUTF(
this.m.getName());
out.writeObject(ReflectionUtil.toTypeNameArray(
this.m.getParameterTypes()));
}
}
/*
* (non-Javadoc)
*
* @see java.io.Externalizable#readExternal(java.io.ObjectInput)
*/
@Override
public void readExternal(ObjectInput in)
throws IOException,
ClassNotFoundException {
this.prefix = in.readUTF();
if (
this.prefix.isEmpty()) {
this.prefix =
null;
}
this.localName = in.readUTF();
this.owner = in.readUTF();
this.name = in.readUTF();
this.types = (String[]) in.readObject();
}
public Method getMethod() {
if (
this.m ==
null) {
try {
Class<?> t = ReflectionUtil.forName(
this.owner);
Class<?>[] p = ReflectionUtil.toTypeArray(
this.types);
this.m = t.getMethod(
this.name, p);
}
catch (Exception e) {
// Ignore: this results in ELException after further resolution
}
}
return this.m;
}
/* (non-Javadoc)
* @see java.lang.Object#equals(java.lang.Object)
*/
@Override
public boolean equals(Object obj) {
if (obj
instanceof Function) {
return this.hashCode() == obj.hashCode();
}
return false;
}
/* (non-Javadoc)
* @see java.lang.Object#hashCode()
*/
@Override
public int hashCode() {
return (
this.prefix +
this.localName).hashCode();
}
}
}