Skip to content

Commit

Permalink
[lang] Make the closure serializable.
Browse files Browse the repository at this point in the history
close #885

Signed-off-by: Stéphane Galland <galland@arakhne.org>
  • Loading branch information
gallandarakhneorg committed Nov 27, 2018
1 parent fb7e37e commit 13dfb0d
Show file tree
Hide file tree
Showing 18 changed files with 4,380 additions and 40 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
/*
* $Id$
*
* SARL is an general-purpose agent programming language.
* More details on http://www.sarl.io
*
* Copyright (C) 2014-2018 the original authors 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
*
* 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 io.sarl.lang.util;

import java.io.InvalidClassException;
import java.io.ObjectStreamException;
import java.io.Serializable;
import java.io.StreamCorruptedException;
import java.io.WriteAbortedException;
import java.lang.reflect.Constructor;

/**
* An object that could be used as a proxy for serializing an object.
*
* @author $Author: sgalland$
* @version $FullVersion$
* @mavengroupid $GroupId$
* @mavenartifactid $ArtifactId$
* @since 0.8.6
*/
public class SerializableProxy implements Serializable {

private static final long serialVersionUID = -2601677585237973795L;

private final Class<?> proxyType;

private final Object[] values;

/** Construct a proxy.
*
* @param type the inner or anonymous class of the object to create when deserializing.
* @param values the values to serialize.
*/
public SerializableProxy(Class<?> type, Object... values) {
assert type.isLocalClass() || type.isAnonymousClass();
this.proxyType = type;
this.values = values;
}

/** Replies the value at the given index.
*
* @param <T> the expected type of the data.
* @param index the index.
* @return the value.
*/
@SuppressWarnings("unchecked")
public <T> T getValue(int index) {
return (T) this.values[index];
}

/** This function enables to deserialize an instance of this proxy.
* @throws ObjectStreamException if the deserialization fails
*/
private Object readResolve() throws ObjectStreamException {
Constructor<?> compatible = null;
for (final Constructor<?> candidate : this.proxyType.getDeclaredConstructors()) {
if (candidate != null && isCompatible(candidate)) {
if (compatible != null) {
throw new IllegalStateException();
}
compatible = candidate;
}
}
if (compatible != null) {
if (!compatible.isAccessible()) {
compatible.setAccessible(true);
}
try {
final Object[] arguments = new Object[this.values.length + 1];
System.arraycopy(this.values, 0, arguments, 1, this.values.length);
return compatible.newInstance(arguments);
} catch (Exception exception) {
throw new WriteAbortedException(exception.getLocalizedMessage(), exception);
}
}
throw new InvalidClassException("compatible constructor not found"); //$NON-NLS-1$
}

private boolean isCompatible(Constructor<?> candidate) throws ObjectStreamException {
final Class<?>[] parameterTypes = candidate.getParameterTypes();
if (parameterTypes.length != this.values.length + 1) {
return false;
}
final Class<?> enclosingType = this.proxyType.getEnclosingClass();
if (!parameterTypes[0].isAssignableFrom(enclosingType)) {
return false;
}
for (int i = 0, j = 1; i < this.values.length; ++i, ++j) {
final Object param = this.values[i];
assert j < parameterTypes.length;
Class<?> clazz = parameterTypes[j];
if (clazz.isPrimitive()) {
clazz = wrapperTypeFor(clazz);
}
if (param != null && !clazz.isInstance(param)) {
return false;
}
}
return true;
}

@SuppressWarnings("checkstyle:npathcomplexity")
private static Class<?> wrapperTypeFor(Class<?> primitive) throws ObjectStreamException {
if (primitive == Boolean.TYPE) {
return Boolean.class;
}
if (primitive == Byte.TYPE) {
return Byte.class;
}
if (primitive == Character.TYPE) {
return Character.class;
}
if (primitive == Short.TYPE) {
return Short.class;
}
if (primitive == Integer.TYPE) {
return Integer.class;
}
if (primitive == Long.TYPE) {
return Long.class;
}
if (primitive == Float.TYPE) {
return Float.class;
}
if (primitive == Double.TYPE) {
return Double.class;
}
if (primitive == Void.TYPE) {
return Void.class;
}
throw new StreamCorruptedException();
}

}

0 comments on commit 13dfb0d

Please sign in to comment.