 
                        
        
                        
        com.oracle.bedrock.junit.CoherenceClusterOrchestration Maven / Gradle / Ivy
Show all versions of coherence-12.1.3-testing-support Show documentation
/*
 * File: CoherenceClusterOrchestration.java
 *
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * The contents of this file are subject to the terms and conditions of 
 * the Common Development and Distribution License 1.0 (the "License").
 *
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the License by consulting the LICENSE.txt file
 * distributed with this file, or by consulting https://oss.oracle.com/licenses/CDDL
 *
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file LICENSE.txt.
 *
 * MODIFICATIONS:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 */
package com.oracle.bedrock.junit;
import com.oracle.bedrock.Option;
import com.oracle.bedrock.OptionsByType;
import com.oracle.bedrock.testsupport.deferred.Eventually;
import com.oracle.bedrock.runtime.LocalPlatform;
import com.oracle.bedrock.runtime.Platform;
import com.oracle.bedrock.runtime.coherence.CoherenceCluster;
import com.oracle.bedrock.runtime.coherence.CoherenceClusterBuilder;
import com.oracle.bedrock.runtime.coherence.CoherenceClusterMember;
import com.oracle.bedrock.runtime.coherence.callables.GetAutoStartServiceNames;
import com.oracle.bedrock.runtime.coherence.options.ClusterName;
import com.oracle.bedrock.runtime.coherence.options.ClusterPort;
import com.oracle.bedrock.runtime.coherence.options.LocalHost;
import com.oracle.bedrock.runtime.coherence.options.LocalStorage;
import com.oracle.bedrock.runtime.coherence.options.Multicast;
import com.oracle.bedrock.runtime.coherence.options.RoleName;
import com.oracle.bedrock.runtime.java.options.Headless;
import com.oracle.bedrock.runtime.java.options.HeapSize;
import com.oracle.bedrock.runtime.java.options.HotSpot;
import com.oracle.bedrock.runtime.java.options.SystemProperty;
import com.oracle.bedrock.runtime.options.ApplicationClosingBehavior;
import com.oracle.bedrock.runtime.options.Console;
import com.oracle.bedrock.runtime.options.DisplayName;
import com.oracle.bedrock.util.Capture;
import com.tangosol.net.CacheFactory;
import com.tangosol.net.ConfigurableCacheFactory;
import org.junit.rules.ExternalResource;
import org.junit.runner.Description;
import org.junit.runners.model.Statement;
import java.util.HashMap;
import java.util.Properties;
import java.util.Set;
import static com.oracle.bedrock.deferred.DeferredHelper.invoking;
import static org.hamcrest.core.Is.is;
/**
 * A JUnit {@link ExternalResource} to represent and orchestrate configuring,
 * establishing, accessing and tearing down a Oracle Coherence Cluster, with
 * support for requesting various ConfigurableCacheFactory implementations to
 * interact with the orchestrated Coherence Cluster.
 * 
 * Copyright (c) 2015. All Rights Reserved. Oracle Corporation.
 * Oracle is a registered trademark of Oracle Corporation and/or its affiliates.
 * 
 * DEPRECATED: Please consider migrating to use the {@link CoherenceClusterResource} instead. This
 * class may be removed in the next major release.
 *
 * @author Jonathan Knight
 * @author Brian Oliver
 * @author Harvey Raja
 * @author Aleks Seovic
 */
@Deprecated
public class CoherenceClusterOrchestration extends ExternalResource
{
    /**
     * The number of storage enable members that will be started in the cluster.
     */
    private int storageMemberCount = 2;
    /**
     * The {@link LocalPlatform} on which the Coherence Cluster will be orchestrated.
     */
    private LocalPlatform platform;
    /**
     * The {@link OptionsByType} to use as a basis for constructing a variety of {@link CoherenceClusterMember}s.
     */
    private OptionsByType commonMemberOptions;
    /**
     * The {@link OptionsByType} to use as a basis for constructing storage-enabled {@link CoherenceClusterMember}s.
     */
    private OptionsByType storageMemberOptions;
    /**
     * The {@link OptionsByType} to use as a basis for constructing proxy {@link CoherenceClusterMember}s.
     */
    private OptionsByType proxyMemberOptions;
    /**
     * The {@link OptionsByType} to use when creating the {@link CoherenceCluster}.
     */
    private OptionsByType clusterCreationOptions;
    /**
     * The {@link OptionsByType} to use when closing the {@link CoherenceCluster}.
     */
    private OptionsByType clusterClosingOptions;
    /**
     * The {@link CoherenceCluster} established by the orchestrator.
     */
    private CoherenceCluster cluster;
    /**
     * The original system properties to be restored when the orchestration is complete.
     */
    private Properties systemProperties;
    /**
     * The {@link ConfigurableCacheFactory} sessions that have been
     * created against the orchestrated {@link CoherenceCluster}.
     */
    private HashMap sessions;
    /**
     * The cluster port.
     */
    private Capture clusterPort;
    /**
     * The extend port.
     */
    private Capture extendPort;
    /**
     * Constructs a {@link CoherenceClusterOrchestration}.
     */
    public CoherenceClusterOrchestration()
    {
        // we're going to orchestrate the cluster using the LocalPlatform
        this.platform = LocalPlatform.get();
        // establish a Cluster port
        this.clusterPort = new Capture<>(platform.getAvailablePorts());
        // establish the Extend port (is the same as the cluster port)
        this.extendPort = new Capture<>(platform.getAvailablePorts());
        // establish a common member options on which to base storage enabled and proxy members
        this.commonMemberOptions = OptionsByType.empty();
        // establish the common server schema address and port details
        String hostAddress = platform.getLoopbackAddress().getHostAddress();
        this.commonMemberOptions.add(LocalHost.of(hostAddress));
        this.commonMemberOptions.add(ClusterPort.of(clusterPort));
        this.commonMemberOptions.add(Multicast.ttl(0));
        // we also define the proxy configuration (this will only be used if it's enabled)
        this.commonMemberOptions.add(SystemProperty.of("tangosol.coherence.extend.address", hostAddress));
        this.commonMemberOptions.add(SystemProperty.of("tangosol.coherence.extend.port", extendPort));
        // establish default java process configuration
        this.commonMemberOptions.add(Headless.enabled());
        this.commonMemberOptions.add(HotSpot.Mode.SERVER);
        this.commonMemberOptions.add(HeapSize.of(256, HeapSize.Units.MB, 1024, HeapSize.Units.MB));
        // by default we'll use the SystemApplicationConsole
        this.commonMemberOptions.add(Console.system());
        // by default we don't have any special options for storage or proxy members
        this.storageMemberOptions = OptionsByType.empty();
        this.proxyMemberOptions   = OptionsByType.empty();
        // by default we don't have a cluster
        this.cluster = null;
        // by default the creation and closing options aren't set
        this.clusterCreationOptions = OptionsByType.empty();
        this.clusterClosingOptions  = OptionsByType.empty();
        // by default we have no sessions
        this.sessions = new HashMap<>();
    }
    @Override
    public Statement apply(Statement   base,
                           Description description)
    {
        // automatically set the cluster name to the test class name
        // if the cluster name isn't configured
        if (commonMemberOptions.get(ClusterName.class) == null)
        {
            commonMemberOptions.add(ClusterName.of(description.getClassName()));
        }
        return super.apply(base, description);
    }
    @Override
    protected void before() throws Throwable
    {
        // take a snapshot of the system properties
        systemProperties = com.oracle.bedrock.util.SystemProperties.createSnapshot();
        // establish a CoherenceClusterBuilder with the required configuration
        CoherenceClusterBuilder clusterBuilder = new CoherenceClusterBuilder();
        // define the options for the storage enabled members of the cluster
        OptionsByType storageServerOptions = createStorageEnabledMemberOptions();
        storageServerOptions.addAll(clusterCreationOptions);
        clusterBuilder.include(storageMemberCount, CoherenceClusterMember.class, storageServerOptions.asArray());
        // define the schema for the proxy enabled members of the cluster
        OptionsByType proxyServerOptions = createProxyServerOptions();
        proxyServerOptions.addAll(clusterCreationOptions);
        int proxyMemberCount = 1;
        clusterBuilder.include(proxyMemberCount, CoherenceClusterMember.class, proxyServerOptions.asArray());
        int preferredClusterSize = storageMemberCount + proxyMemberCount;
        // establish the cluster
        cluster = clusterBuilder.build();
        // ensure that the cluster has been orchestrated correctly
        Eventually.assertThat(invoking(cluster).getClusterSize(), is(preferredClusterSize));
        // ensure that all services marked as autostart on the proxy have started
        CoherenceClusterMember proxyServer     = cluster.get("proxy-1");
        Set            setServiceNames = proxyServer.invoke(new GetAutoStartServiceNames());
        for (String sServiceName : setServiceNames)
        {
            Eventually.assertThat(invoking(proxyServer).isServiceRunning(sServiceName), is(true));
        }
        // let's ensure that we don't have a local cluster member
        CacheFactory.setCacheFactoryBuilder(null);
        CacheFactory.shutdown();
        // let the super-class perform it's initialization
        super.before();
    }
    @Override
    protected void after()
    {
        // clean up the sessions
        synchronized (sessions)
        {
            for (ConfigurableCacheFactory session : sessions.values())
            {
                CacheFactory.getCacheFactoryBuilder().release(session);
            }
        }
        CacheFactory.shutdown();
        // close the cluster
        cluster.close(clusterClosingOptions.asArray());
        // restore the system properties
        com.oracle.bedrock.util.SystemProperties.replaceWith(systemProperties);
        // let the super-class perform it's cleanup as well
        super.after();
    }
    protected OptionsByType createStorageEnabledMemberOptions()
    {
        // define the options for the storage enabled members of the cluster
        OptionsByType optionsByType = OptionsByType.of(commonMemberOptions);
        optionsByType.add(DisplayName.of("storage"));
        optionsByType.add(RoleName.of("storage"));
        optionsByType.add(LocalStorage.enabled());
        optionsByType.addAll(storageMemberOptions);
        return optionsByType;
    }
    protected OptionsByType createProxyServerOptions()
    {
        OptionsByType optionsByType = OptionsByType.of(commonMemberOptions);
        optionsByType.add(DisplayName.of("proxy"));
        optionsByType.add(RoleName.of("proxy"));
        optionsByType.add(LocalStorage.disabled());
        optionsByType.add(SystemProperty.of("tangosol.coherence.extend.enabled", true));
        optionsByType.addAll(proxyMemberOptions);
        return optionsByType;
    }
    /**
     * Sets the {@link Option}s to be used when closing the {@link CoherenceCluster}.
     *
     * @param options the {@link Option}s
     *
     * @return the {@link CoherenceClusterOrchestration} to permit fluent-style method-calls
     *
     * @see ApplicationClosingBehavior
     */
    public CoherenceClusterOrchestration withClosingOptions(Option... options)
    {
        this.clusterClosingOptions = OptionsByType.of(options);
        return this;
    }
    /**
     * Sets the {@link Option}s to be passed to the {@link CoherenceClusterBuilder} when building the
     * {@link CoherenceCluster}.
     *
     * @param options the {@link Option}s
     *
     * @return the {@link CoherenceClusterOrchestration} to permit fluent-style method-calls
     */
    public CoherenceClusterOrchestration withBuilderOptions(Option... options)
    {
        this.clusterCreationOptions = OptionsByType.of(options);
        return this;
    }
    /**
     * Obtains the {@link CoherenceCluster} that was established by the orchestration.
     *
     * @return the {@link CoherenceCluster}
     */
    public CoherenceCluster getCluster()
    {
        return cluster;
    }
    /**
     * Obtains the {@link LocalPlatform} on which the {@link CoherenceClusterOrchestration} will
     * create sessions.
     * 
     * NOTE: The orchestrated {@link CoherenceCluster} may be orchestrated on one or more other
     * {@link Platform}s.
     *
     * @return the {@link LocalPlatform}
     */
    public LocalPlatform getLocalPlatform()
    {
        return platform;
    }
    /**
     * Adds the specified {@link Option}s to the {@link CoherenceClusterOrchestration} for
     * launching all {@link CoherenceClusterMember}s.
     *
     * @param options  the {@link Option}s
     *
     * @return  the {@link CoherenceClusterOrchestration}
     */
    public CoherenceClusterOrchestration withOptions(Option... options)
    {
        commonMemberOptions.addAll(options);
        return this;
    }
    /**
     * Adds the specified {@link Option}s to the {@link CoherenceClusterOrchestration} for
     * launching all storage-enabled {@link CoherenceClusterMember}s.
     *
     * @param options  the {@link Option}s for storage-enabled {@link CoherenceClusterMember}s
     *
     * @return  the {@link CoherenceClusterOrchestration}
     */
    public CoherenceClusterOrchestration withStorageMemberOptions(Option... options)
    {
        storageMemberOptions.addAll(options);
        return this;
    }
    /**
     * Adds the specified {@link Option}s to the {@link CoherenceClusterOrchestration} for
     * launching all proxy server {@link CoherenceClusterMember}s.
     *
     * @param options  the {@link Option}s for proxy server {@link CoherenceClusterMember}s
     *
     * @return  the {@link CoherenceClusterOrchestration}
     */
    public CoherenceClusterOrchestration withProxyMemberOptions(Option... options)
    {
        proxyMemberOptions.addAll(options);
        return this;
    }
    /**
     * Obtains a session (represented as a {@link ConfigurableCacheFactory} against the
     * orchestrated Coherence Cluster for interacting with Coherence.
     * 
     * Only a single session may be created by a {@link CoherenceClusterOrchestration}
     * against an orchestrated {@link CoherenceCluster}.
     * 
     * Attempts to request a session multiple times with the same {@link SessionBuilder}
     * will return the same session.
     *
     * @param builder the builder for the specific type of session
     *
     * @return a {@link ConfigurableCacheFactory} representing the Coherence Session.
     *
     * @throws IllegalStateException when an attempt to request sessions for
     *                               different {@link SessionBuilder}s is made
     */
    public synchronized ConfigurableCacheFactory getSessionFor(SessionBuilder builder)
    {
        // restore the system properties (as the session needs to start cleanly)
        com.oracle.bedrock.util.SystemProperties.replaceWith(systemProperties);
        ConfigurableCacheFactory session = sessions.get(builder);
        if (session == null)
        {
            OptionsByType optionsByType = OptionsByType.of(commonMemberOptions);
            optionsByType.add(RoleName.of("client"));
            optionsByType.add(LocalStorage.disabled());
            session = builder.build(platform, cluster, optionsByType);
            sessions.put(builder, session);
        }
        return session;
    }
    /**
     * Sets the number of storage enabled members that will be started in the cluster.
     *
     * @param count the number of storage enabled members
     *
     * @return the {@link CoherenceClusterOrchestration}
     */
    public CoherenceClusterOrchestration setStorageMemberCount(int count)
    {
        storageMemberCount = count;
        return this;
    }
}