All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hazelcast.buildutils.FinalRemovalAgent Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * Copyright (c) 2008-2022, Hazelcast, Inc. All Rights Reserved.
 *
 * 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 com.hazelcast.buildutils;

import com.hazelcast.core.HazelcastException;
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.asm.ModifierAdjustment;
import net.bytebuddy.description.modifier.MethodManifestation;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.DynamicType;
import net.bytebuddy.jar.asm.Opcodes;

import java.lang.instrument.Instrumentation;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;

import static net.bytebuddy.matcher.ElementMatchers.nameStartsWith;
import static net.bytebuddy.matcher.ElementMatchers.named;

public final class FinalRemovalAgent {

    // Class name -> List to be processed for
    // removing final modifier
    private static final Map> FINAL_METHODS;

    private FinalRemovalAgent() {
    }

    static {
        Map> finalMethods = new HashMap<>();
        finalMethods.put("com.hazelcast.cp.internal.session.AbstractProxySessionManager",
                setOf("getSession", "getSessionAcquireCount"));
        finalMethods.put("com.hazelcast.spi.impl.AbstractInvocationFuture",
                setOf("get", "join"));
        finalMethods.put("com.hazelcast.cp.internal.datastructures.spi.blocking.AbstractBlockingService",
                setOf("getRegistryOrNull"));
        finalMethods.put("com.hazelcast.cp.internal.datastructures.spi.blocking.ResourceRegistry",
                setOf("getWaitTimeouts"));
        FINAL_METHODS = finalMethods;
    }

    /**
     * When running a compatibility test, all com.hazelcast.* classes are transformed so that none are
     * loaded with final modifier to allow subclass proxying.
     * We configure the agent with REDEFINE type strategy and NoOp initialization strategy.
     * This allows for the redefinition of the type (instead of default REBASE strategy) without
     * adding any methods (which result in modifying Serializable classes' serialVersionUid). For
     * more details see {@link net.bytebuddy.ByteBuddy#rebase(Class)} vs
     * {@link net.bytebuddy.ByteBuddy#redefine(Class)}.
     */
    public static void premain(String argument, Instrumentation instrumentation) {
        new AgentBuilder.Default()
                .with(AgentBuilder.TypeStrategy.Default.REDEFINE)
                .with(AgentBuilder.InitializationStrategy.NoOp.INSTANCE)
                .type(nameStartsWith("com.hazelcast"))
                .transform((builder, typeDescription, classLoader, module) -> {
                    builder = manifestMethodAsPlain(builder, typeDescription);
                    int actualModifiers = typeDescription.getActualModifiers(false);
                    // unset final modifier
                    int nonFinalModifiers = actualModifiers & ~Opcodes.ACC_FINAL;
                    if (actualModifiers != nonFinalModifiers) {
                        return builder.modifiers(nonFinalModifiers);
                    } else {
                        return builder;
                    }
                }).installOn(instrumentation);
    }

    /**
     * This method assigns the {@link MethodManifestation.PLAIN} to each method of the
     * given {@link TypeDescription} that is listed in {@link #FINAL_METHODS}.
     */
    private static DynamicType.Builder manifestMethodAsPlain(DynamicType.Builder builder,
                                                             TypeDescription typeDescription) {
        final String typeName = typeDescription.getName();
        if (FINAL_METHODS.containsKey(typeName)) {
            for (String methodName : FINAL_METHODS.get(typeName)) {
                builder = builder.visit(new ModifierAdjustment()
                        .withMethodModifiers(named(methodName), MethodManifestation.PLAIN)
                );
            }
        }
        return builder;
    }

    private static  Set setOf(T... items) {
        Set set = new HashSet<>(Arrays.asList(items));
        if (set.size() < items.length) {
            throw new HazelcastException("set cannot contain duplicate items");
        }
        return set;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy