package org.jgroups.tests; import org.jgroups.Global; import org.jgroups.JChannel; import org.jgroups.ViewId; import org.jgroups.protocols.*; import org.jgroups.protocols.pbcast.GMS; import org.jgroups.protocols.pbcast.NAKACK2; import org.jgroups.protocols.pbcast.STABLE; import org.jgroups.stack.ProtocolStack; import org.jgroups.util.Util; import org.testng.annotations.AfterMethod; import org.testng.annotations.BeforeMethod; import org.testng.annotations.Test; import java.util.HashMap; import java.util.Map; /** * Tests merging with a large number of members * @author Bela Ban */ @Test(groups=Global.FUNCTIONAL,singleThreaded=true) public class LargeMergeTest { static final int NUM=50; // number of members protected final JChannel[] channels=new JChannel[NUM]; @BeforeMethod void setUp() throws Exception { System.out.print("Connecting channels: "); for(int i=0; i < NUM; i++) { channels[i]=new JChannel(new SHARED_LOOPBACK(), new DISCARD().setValue("discard_all",true), new SHARED_LOOPBACK_PING(), new MERGE3().setValue("min_interval",1000) .setValue("max_interval",3000) .setValue("check_interval", 6000) .setValue("max_participants_in_merge", NUM), new NAKACK2().setValue("use_mcast_xmit",false) .setValue("discard_delivered_msgs",true) .setValue("log_discard_msgs",false).setValue("log_not_found_msgs",false) .setValue("xmit_table_num_rows",5) .setValue("xmit_table_msgs_per_row",10), new UNICAST3().setValue("xmit_table_num_rows",5) .setValue("xmit_table_msgs_per_row",10) .setValue("conn_expiry_timeout", 10000), new STABLE().setValue("max_bytes",500000), new GMS().setValue("print_local_addr",false) .setValue("join_timeout", 1) .setValue("leave_timeout",100) .setValue("log_view_warnings",false) .setValue("view_ack_collection_timeout",2000) .setValue("log_collect_msgs",false)); channels[i].setName(String.valueOf((i + 1))); channels[i].connect("LargeMergeTest"); System.out.print(i + 1 + " "); } System.out.println(""); } @AfterMethod void tearDown() throws Exception { for(int i=NUM-1; i >= 0; i--) { ProtocolStack stack=channels[i].getProtocolStack(); String cluster_name=channels[i].getClusterName(); stack.stopStack(cluster_name); stack.destroy(); } } public void testClusterFormationAfterMerge() { System.out.println("\nEnabling message traffic between members to start the merge"); for(JChannel ch: channels) { DISCARD discard=(DISCARD)ch.getProtocolStack().findProtocol(DISCARD.class); discard.setDiscardAll(false); } boolean merge_completed=true; for(int i=0; i < NUM; i++) { merge_completed=true; System.out.println(); Map<ViewId,Integer> views=new HashMap<>(); for(JChannel ch: channels) { ViewId view_id=ch.getView().getViewId(); Integer val=views.get(view_id); views.put(view_id, val == null? 1 : val+1); int size=ch.getView().size(); if(size != NUM) merge_completed=false; } if(i++ > 0) { int num_singleton_views=0; for(Map.Entry<ViewId,Integer> entry: views.entrySet()) { if(entry.getValue() == 1) num_singleton_views++; else { System.out.println("==> " + entry.getKey() + ": " + entry.getValue() + " members"); } } if(num_singleton_views > 0) System.out.println("==> " + num_singleton_views + " singleton views"); System.out.println("------------------\n" + getStats()); } if(merge_completed) break; Util.sleep(5000); } if(!merge_completed) { System.out.println("\nFinal cluster:"); for(JChannel ch: channels) { int size=ch.getView().size(); System.out.println(ch.getAddress() + ": " + size + " members - " + (size == NUM? "OK" : "FAIL")); } } for(JChannel ch: channels) { int size=ch.getView().size(); assert size == NUM : "Channel has " + size + " members, but should have " + NUM; } } protected String getStats() { StringBuilder sb=new StringBuilder(); int merge_task_running=0; int merge_canceller_running=0; int merge_in_progress=0; int gms_merge_task_running=0; for(JChannel ch: channels) { MERGE3 merge3=(MERGE3)ch.getProtocolStack().findProtocol(MERGE3.class); if(merge3 != null && merge3.isMergeTaskRunning()) merge_task_running++; GMS gms=(GMS)ch.getProtocolStack().findProtocol(GMS.class); if(gms.isMergeKillerRunning()) merge_canceller_running++; if(gms.isMergeInProgress()) merge_in_progress++; if(gms.isMergeTaskRunning()) gms_merge_task_running++; } sb.append("merge tasks running: " + merge_task_running).append("\n"); sb.append("merge killers running: " + merge_canceller_running).append("\n"); sb.append("merge in progress: " + merge_in_progress).append("\n"); sb.append("gms.merge tasks running: " + gms_merge_task_running).append("\n"); return sb.toString(); } @Test(enabled=false) public static void main(String args[]) throws Exception { LargeMergeTest test=new LargeMergeTest(); test.setUp(); test.testClusterFormationAfterMerge(); test.tearDown(); } }