QuorumOracleMajTest.java

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.zookeeper.test;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertTrue;
import java.util.ArrayList;
import org.apache.zookeeper.jmx.MBeanRegistry;
import org.apache.zookeeper.server.quorum.Leader;
import org.apache.zookeeper.server.quorum.LearnerHandler;
import org.apache.zookeeper.server.quorum.QuorumPeer;
import org.junit.jupiter.api.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;


public class QuorumOracleMajTest extends QuorumBaseOracle_2Nodes {

    protected static final Logger LOG = LoggerFactory.getLogger(QuorumMajorityTest.class);
    public static final long CONNECTION_TIMEOUT = ClientTest.CONNECTION_TIMEOUT;

    /***************************************************************/
    /* Test that the majority quorum verifier only counts votes from */
    /* followers in its view                                    */
    /***************************************************************/
    @Test
    public void testMajQuorums() throws Throwable {
        LOG.info("Verify QuorumPeer#electionTimeTaken jmx bean attribute");

        ArrayList<QuorumPeer> peers = getPeerList();
        for (int i = 1; i <= peers.size(); i++) {
            QuorumPeer qp = peers.get(i - 1);
            Long electionTimeTaken = -1L;
            String bean = "";
            if (qp.getPeerState() == QuorumPeer.ServerState.FOLLOWING) {
                bean = String.format("%s:name0=ReplicatedServer_id%d,name1=replica.%d,name2=Follower", MBeanRegistry.DOMAIN, i, i);
            } else if (qp.getPeerState() == QuorumPeer.ServerState.LEADING) {
                bean = String.format("%s:name0=ReplicatedServer_id%d,name1=replica.%d,name2=Leader", MBeanRegistry.DOMAIN, i, i);
            }
            electionTimeTaken = (Long) JMXEnv.ensureBeanAttribute(bean, "ElectionTimeTaken");
            assertTrue(electionTimeTaken >= 0, "Wrong electionTimeTaken value!");
        }

        tearDown();
        //setup servers 1-2 to be followers
        // id=1, oracle is false; id=2, oracle is true
        setUp();

        QuorumPeer s;
        int leader;
        if ((leader = getLeaderIndex()) == 1) {
            s = s1;
        } else {
            s = s2;
        }

        noDropConnectionTest(s);

        dropConnectionTest(s, leader);

    }

    private void noDropConnectionTest(QuorumPeer s) {
        Leader.Proposal p = new Leader.Proposal();


        p.addQuorumVerifier(s.getQuorumVerifier());

        // 1 followers out of 2 is not a majority
        p.addAck(Long.valueOf(1));
        assertEquals(false, p.hasAllQuorums());

        // 6 is not in the view - its vote shouldn't count
        p.addAck(Long.valueOf(6));
        assertEquals(false, p.hasAllQuorums());

        // 2 followers out of 2 is good
        p.addAck(Long.valueOf(2));
        assertEquals(true, p.hasAllQuorums());

    }


    private void dropConnectionTest(QuorumPeer s, int leader) {
        Leader.Proposal p = new Leader.Proposal();
        p.addQuorumVerifier(s.getQuorumVerifier());

        ArrayList<LearnerHandler> fake = new ArrayList<>();

        LearnerHandler f = null;
        fake.add(f);

        s.getQuorumVerifier().updateNeedOracle(fake);
        // still have valid followers, the oracle should not take place
        assertEquals(false, s.getQuorumVerifier().getNeedOracle());

        fake.remove(0);
        s.getQuorumVerifier().updateNeedOracle(fake);
        // lose all of followers, the oracle should take place
        assertEquals(true, s.getQuorumVerifier().getNeedOracle());


        // when leader is 1, we expect false.
        // when leader is 2, we expect true.
        assertEquals(leader != 1, p.hasAllQuorums());
    }
}