io.trino.sql.gen.IsolatedClass Maven / Gradle / Ivy
/*
* 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.trino.sql.gen;
import com.google.common.collect.ImmutableMap;
import com.google.common.io.ByteStreams;
import io.airlift.bytecode.DynamicClassLoader;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.lang.String.format;
public final class IsolatedClass
{
private IsolatedClass() {}
public static Class extends T> isolateClass(
DynamicClassLoader dynamicClassLoader,
Class publicBaseClass,
Class extends T> implementationClass,
Class>... additionalClasses)
{
ImmutableMap.Builder builder = ImmutableMap.builder();
builder.put(implementationClass.getName(), getBytecode(implementationClass));
for (Class> additionalClass : additionalClasses) {
builder.put(additionalClass.getName(), getBytecode(additionalClass));
}
// load classes into a private class loader
Map> isolatedClasses = dynamicClassLoader.defineClasses(builder.buildOrThrow());
Class> isolatedClass = isolatedClasses.get(implementationClass.getName());
// verify the isolated class
checkArgument(isolatedClass != null, "Could load class %s", implementationClass.getName());
checkArgument(publicBaseClass.isAssignableFrom(isolatedClass),
"Error isolating class %s, newly loaded class is not a sub type of %s",
implementationClass.getName(),
publicBaseClass.getName());
checkState(isolatedClass != implementationClass, "Isolation failed");
return isolatedClass.asSubclass(publicBaseClass);
}
private static byte[] getBytecode(Class> clazz)
{
try (InputStream stream = clazz.getClassLoader().getResourceAsStream(clazz.getName().replace('.', '/') + ".class")) {
checkArgument(stream != null, "Could not obtain byte code for class %s", clazz.getName());
return ByteStreams.toByteArray(stream);
}
catch (IOException e) {
throw new RuntimeException(format("Could not obtain byte code for class %s", clazz.getName()), e);
}
}
}