Package voldemort.utils

Source Code of voldemort.utils.Ec2GossipTest$Ec2GossipTestConfig

/*
* Copyright 2013 LinkedIn, Inc
*
* 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 voldemort.utils;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import static voldemort.TestUtils.assertWithBackoff;
import static voldemort.utils.Ec2RemoteTestUtils.createInstances;
import static voldemort.utils.Ec2RemoteTestUtils.destroyInstances;
import static voldemort.utils.RemoteTestUtils.deploy;
import static voldemort.utils.RemoteTestUtils.generateClusterDescriptor;
import static voldemort.utils.RemoteTestUtils.startClusterAsync;
import static voldemort.utils.RemoteTestUtils.stopCluster;
import static voldemort.utils.RemoteTestUtils.stopClusterQuiet;
import static voldemort.utils.RemoteTestUtils.toHostNames;

import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;

import org.apache.log4j.Logger;
import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test;

import voldemort.Attempt;
import voldemort.VoldemortException;
import voldemort.client.ClientConfig;
import voldemort.client.protocol.admin.AdminClient;
import voldemort.client.protocol.admin.AdminClientConfig;
import voldemort.cluster.Cluster;
import voldemort.cluster.Node;
import voldemort.store.metadata.MetadataStore;
import voldemort.versioning.VectorClock;
import voldemort.versioning.Version;
import voldemort.versioning.Versioned;

import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;

/**
*
*/
public class Ec2GossipTest {

    private static Ec2GossipTestConfig ec2GossipTestConfig;
    private static List<HostNamePair> hostNamePairs;
    private static List<String> hostNames;
    private static Map<String, Integer> nodeIds;

    private static final Logger logger = Logger.getLogger(Ec2GossipTest.class);

    @BeforeClass
    public static void setUpClass() throws Exception {
        ec2GossipTestConfig = new Ec2GossipTestConfig();
        hostNamePairs = createInstances(ec2GossipTestConfig);
        hostNames = toHostNames(hostNamePairs);
        nodeIds = generateClusterDescriptor(hostNamePairs, "test", ec2GossipTestConfig, true);

        if(logger.isInfoEnabled())
            logger.info("Sleeping for 30 seconds to give EC2 instances some time to complete startup");

        Thread.sleep(30000);
    }

    @AfterClass
    public static void tearDownClass() throws Exception {
        if(hostNames != null)
            destroyInstances(hostNames, ec2GossipTestConfig);
    }

    @Before
    public void setUp() throws Exception {
        deploy(hostNames, ec2GossipTestConfig);
        startClusterAsync(hostNames, ec2GossipTestConfig, nodeIds);
    }

    @After
    public void tearDown() throws Exception {
        stopClusterQuiet(hostNames, ec2GossipTestConfig);
    }

    @Test
    public void testGossip() throws Exception {
        try {
            Set<String> oldHostnames = new HashSet<String>(hostNames);
            final Set<Integer> oldNodeIdSet = new HashSet<Integer>(nodeIds.values());
            Map<String, Integer> oldNodeIdMap = new HashMap<String, Integer>(nodeIds);

            logger.info("Cluster before expanding: " + nodeIds);

            /**
             * First verify that the cluster had been expanded correctly
             */

            Pair<List<Integer>, List<String>> pair = expandCluster();

            final List<Integer> newNodeIds = pair.getFirst();
            List<String> newHostnames = pair.getSecond();

            assertEquals("correct number of nodes added",
                         newNodeIds.size(),
                         ec2GossipTestConfig.numNewNodes);

            boolean containsOldHostnames = false;
            for(String newHostname: newHostnames) {
                if(oldHostnames.contains(newHostname)) {
                    containsOldHostnames = true;
                    break;
                }
            }

            boolean containsOldNodeIds = false;
            for(Integer newNodeId: newNodeIds) {
                if(oldNodeIdSet.contains(newNodeId)) {
                    containsOldNodeIds = true;
                    break;
                }
            }

            assertFalse("none of the new nodes is an old hostname", containsOldHostnames);
            assertFalse("none of the new nodes is an old node id", containsOldNodeIds);

            for(String oldHostname: oldHostnames) {
                assertEquals("hostname to nodeId mapping preserved for " + oldHostname,
                             oldNodeIdMap.get(oldHostname),
                             nodeIds.get(oldHostname));
            }

            /**
             * Now increment the version on the new nodes, start gossiping
             */
            int peerNodeId = Iterables.find(nodeIds.values(), new Predicate<Integer>() {

                public boolean apply(Integer input) {
                    return !newNodeIds.contains(input);
                }
            });

            logger.info("Select a peer node " + peerNodeId);

            /**
             * So far this only correctly handles a case where a *single* node
             * is added. Present gossip doesn't support more advanced sort of
             * reconcilation.
             */
            for(String hostname: newHostnames) {
                int nodeId = nodeIds.get(hostname);
                AdminClient adminClient = new AdminClient("tcp://" + hostname + ":6666",
                                                          new AdminClientConfig(),
                                                          new ClientConfig());

                Versioned<String> versioned = adminClient.metadataMgmtOps.getRemoteMetadata(nodeId,
                                                                                            MetadataStore.CLUSTER_KEY);
                Version version = versioned.getVersion();

                VectorClock vectorClock = (VectorClock) version;
                vectorClock.incrementVersion(nodeId, System.currentTimeMillis());

                try {
                    adminClient.metadataMgmtOps.updateRemoteMetadata(peerNodeId,
                                                                     MetadataStore.CLUSTER_KEY,
                                                                     versioned);
                    adminClient.metadataMgmtOps.updateRemoteMetadata(nodeId,
                                                                     MetadataStore.CLUSTER_KEY,
                                                                     versioned);
                } catch(VoldemortException e) {
                    logger.error(e);
                }
            }

            /**
             * Finally, verify that all of the nodes have been discovered
             */
            assertWithBackoff(1000, 60000, new Attempt() {

                private int count = 1;
                private AdminClient adminClient = new AdminClient("tcp://" + hostNames.get(0)
                                                                          + ":6666",
                                                                  new AdminClientConfig(),
                                                                  new ClientConfig());

                public void checkCondition() throws Exception, AssertionError {
                    logger.info("Attempt " + count++);

                    for(int testNodeId: oldNodeIdSet) {
                        logger.info("Testing node " + testNodeId);
                        try {
                            Cluster cluster = adminClient.metadataMgmtOps.getRemoteCluster(testNodeId)
                                                                         .getValue();
                            Set<Integer> allNodeIds = new HashSet<Integer>();
                            for(Node node: cluster.getNodes()) {
                                allNodeIds.add(node.getId());
                            }
                            assertTrue("all nodes nodes discovered by node id " + testNodeId,
                                       allNodeIds.containsAll(nodeIds.values()));
                        } catch(VoldemortException e) {
                            fail("caught exception " + e);
                        }

                    }
                }
            });
        } finally {
            stopCluster(hostNames, ec2GossipTestConfig);
        }
    }

    private static Pair<List<Integer>, List<String>> expandCluster() throws Exception {
        List<HostNamePair> newInstances = createInstances(ec2GossipTestConfig.numNewNodes,
                                                          ec2GossipTestConfig);
        List<String> newHostnames = toHostNames(newInstances);

        if(logger.isInfoEnabled())
            logger.info("Sleeping for 15 seconds to let new instances startup");

        Thread.sleep(15000);

        hostNamePairs.addAll(newInstances);
        hostNames = toHostNames(hostNamePairs);
        nodeIds = generateClusterDescriptor(hostNamePairs, "test", ec2GossipTestConfig, true);

        logger.info("Expanded the cluster. New layout: " + nodeIds);

        deploy(newHostnames, ec2GossipTestConfig);
        startClusterAsync(newHostnames, ec2GossipTestConfig, nodeIds);

        if(logger.isInfoEnabled())
            logger.info("Sleeping for 15 seconds to start voldemort on the new nodes");

        Thread.sleep(15000);

        return new Pair<List<Integer>, List<String>>(Lists.transform(newHostnames,
                                                                     new Function<String, Integer>() {

                                                                         public Integer apply(String hostname) {
                                                                             return nodeIds.get(hostname);
                                                                         }
                                                                     }),
                                                     newHostnames);
    }

    private static class Ec2GossipTestConfig extends Ec2RemoteTestConfig {

        private int numNewNodes;

        @Override
        protected void init(Properties properties) {
            super.init(properties);

            numNewNodes = getIntProperty(properties, "gossipNumNewNodes");
        }

        @Override
        protected List<String> getRequiredPropertyNames() {
            List<String> requireds = super.getRequiredPropertyNames();
            requireds.addAll(Arrays.asList("gossipNumNewNodes"));
            return requireds;
        }
    }
}
TOP

Related Classes of voldemort.utils.Ec2GossipTest$Ec2GossipTestConfig

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.