001/* 002 * Licensed to the Apache Software Foundation (ASF) under one 003 * or more contributor license agreements. See the NOTICE file 004 * distributed with this work for additional information 005 * regarding copyright ownership. The ASF licenses this file 006 * to you under the Apache License, Version 2.0 (the 007 * "License"); you may not use this file except in compliance 008 * with the License. You may obtain a copy of the License at 009 * 010 * http://www.apache.org/licenses/LICENSE-2.0 011 * 012 * Unless required by applicable law or agreed to in writing, software 013 * distributed under the License is distributed on an "AS IS" BASIS, 014 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 015 * See the License for the specific language governing permissions and 016 * limitations under the License. 017 */ 018package org.apache.hadoop.hbase.procedure2; 019 020import static org.junit.Assert.assertEquals; 021import static org.junit.Assert.assertNotEquals; 022import static org.junit.Assert.assertTrue; 023 024import java.io.IOException; 025import org.apache.hadoop.fs.FileSystem; 026import org.apache.hadoop.fs.Path; 027import org.apache.hadoop.hbase.HBaseClassTestRule; 028import org.apache.hadoop.hbase.HBaseCommonTestingUtil; 029import org.apache.hadoop.hbase.procedure2.store.ProcedureStore; 030import org.apache.hadoop.hbase.testclassification.MasterTests; 031import org.apache.hadoop.hbase.testclassification.SmallTests; 032import org.junit.After; 033import org.junit.Before; 034import org.junit.ClassRule; 035import org.junit.Test; 036import org.junit.experimental.categories.Category; 037import org.slf4j.Logger; 038import org.slf4j.LoggerFactory; 039 040@Category({ MasterTests.class, SmallTests.class }) 041public class TestProcedureMetrics { 042 @ClassRule 043 public static final HBaseClassTestRule CLASS_RULE = 044 HBaseClassTestRule.forClass(TestProcedureMetrics.class); 045 046 private static final Logger LOG = LoggerFactory.getLogger(TestProcedureMetrics.class); 047 048 private static final int PROCEDURE_EXECUTOR_SLOTS = 1; 049 050 private TestProcEnv procEnv; 051 private static ProcedureExecutor<TestProcEnv> procExecutor; 052 private ProcedureStore procStore; 053 054 private HBaseCommonTestingUtil htu; 055 private FileSystem fs; 056 private Path testDir; 057 private Path logDir; 058 059 private static int beginCount = 0; 060 private static int successCount = 0; 061 private static int failedCount = 0; 062 063 @Before 064 public void setUp() throws IOException { 065 htu = new HBaseCommonTestingUtil(); 066 testDir = htu.getDataTestDir(); 067 fs = testDir.getFileSystem(htu.getConfiguration()); 068 assertTrue(testDir.depth() > 1); 069 070 logDir = new Path(testDir, "proc-logs"); 071 procEnv = new TestProcEnv(); 072 procStore = ProcedureTestingUtility.createStore(htu.getConfiguration(), logDir); 073 procExecutor = new ProcedureExecutor<TestProcEnv>(htu.getConfiguration(), procEnv, procStore); 074 procExecutor.testing = new ProcedureExecutor.Testing(); 075 procStore.start(PROCEDURE_EXECUTOR_SLOTS); 076 ProcedureTestingUtility.initAndStartWorkers(procExecutor, PROCEDURE_EXECUTOR_SLOTS, true); 077 } 078 079 @After 080 public void tearDown() throws IOException { 081 procExecutor.stop(); 082 procStore.stop(false); 083 fs.delete(logDir, true); 084 } 085 086 @Test 087 public void testMetricForSimpleProcedure() throws Exception { 088 // procedure that executes successfully 089 ProcedureMetrics proc = new ProcedureMetrics(true); 090 long id = ProcedureTestingUtility.submitAndWait(procExecutor, proc); 091 assertNotEquals("ProcId zero!", 0, id); 092 beginCount++; 093 successCount++; 094 ProcedureTestingUtility.waitProcedure(procExecutor, proc); 095 assertEquals("beginCount doesn't match!", beginCount, proc.beginCount); 096 assertEquals("successCount doesn't match!", successCount, proc.successCount); 097 assertEquals("failedCont doesn't match!", failedCount, proc.failedCount); 098 } 099 100 @Test 101 public void testMetricsForFailedProcedure() throws Exception { 102 // procedure that fails 103 ProcedureMetrics proc = new ProcedureMetrics(false); 104 long id = ProcedureTestingUtility.submitAndWait(procExecutor, proc); 105 assertNotEquals("ProcId zero!", 0, id); 106 beginCount++; 107 failedCount++; 108 ProcedureTestingUtility.waitProcedure(procExecutor, proc); 109 assertEquals("beginCount doesn't match!", beginCount, proc.beginCount); 110 assertEquals("successCount doesn't match!", successCount, proc.successCount); 111 assertEquals("failedCont doesn't match!", failedCount, proc.failedCount); 112 } 113 114 @Test 115 public void testMetricForYieldProcedure() throws Exception { 116 // procedure that yields 117 ProcedureMetrics proc = new ProcedureMetrics(true, true); 118 long id = ProcedureTestingUtility.submitAndWait(procExecutor, proc); 119 assertNotEquals("ProcId zero!", 0, id); 120 beginCount++; 121 successCount++; 122 ProcedureTestingUtility.waitProcedure(procExecutor, proc); 123 assertEquals("beginCount doesn't match!", beginCount, proc.beginCount); 124 assertEquals("successCount doesn't match!", successCount, proc.successCount); 125 assertEquals("failedCont doesn't match!", failedCount, proc.failedCount); 126 } 127 128 @Test 129 public void testMetricForFailedYiledProcedure() { 130 // procedure that yields and fails 131 ProcedureMetrics proc = new ProcedureMetrics(false, true); 132 long id = ProcedureTestingUtility.submitAndWait(procExecutor, proc); 133 assertNotEquals("ProcId zero!", 0, id); 134 beginCount++; 135 failedCount++; 136 ProcedureTestingUtility.waitProcedure(procExecutor, proc); 137 assertEquals("beginCount doesn't match!", beginCount, proc.beginCount); 138 assertEquals("successCount doesn't match!", successCount, proc.successCount); 139 assertEquals("failedCont doesn't match!", failedCount, proc.failedCount); 140 } 141 142 @Test 143 public void testMetricForProcedureWithChildren() throws Exception { 144 // Procedure that yileds with one of the sub-procedures that fail 145 int subProcCount = 10; 146 int failChildIndex = 2; 147 int yiledChildIndex = -1; 148 ProcedureMetrics[] subprocs = new ProcedureMetrics[subProcCount]; 149 for (int i = 0; i < subProcCount; ++i) { 150 subprocs[i] = new ProcedureMetrics(failChildIndex != i, yiledChildIndex == i, 3); 151 } 152 153 ProcedureMetrics proc = new ProcedureMetrics(true, true, 3, subprocs); 154 long id = ProcedureTestingUtility.submitAndWait(procExecutor, proc); 155 assertNotEquals("ProcId zero!", 0, id); 156 beginCount += subProcCount + 1; 157 successCount += subProcCount - (failChildIndex + 1); 158 if (failChildIndex >= 0) { 159 failedCount += subProcCount + 1; 160 } else { 161 successCount++; 162 } 163 ProcedureTestingUtility.waitProcedure(procExecutor, proc); 164 assertEquals("beginCount doesn't match!", beginCount, proc.beginCount); 165 assertEquals("successCount doesn't match!", successCount, proc.successCount); 166 assertEquals("failedCont doesn't match!", failedCount, proc.failedCount); 167 } 168 169 private static class TestProcEnv { 170 public boolean toggleKillBeforeStoreUpdate = false; 171 public boolean triggerRollbackOnChild = false; 172 } 173 174 public static class ProcedureMetrics extends SequentialProcedure<TestProcEnv> { 175 public static long beginCount = 0; 176 public static long successCount = 0; 177 public static long failedCount = 0; 178 179 private boolean success; 180 private boolean yield; 181 private int yieldCount; 182 private int yieldNum; 183 184 private ProcedureMetrics[] subprocs = null; 185 186 public ProcedureMetrics() { 187 this(true); 188 } 189 190 public ProcedureMetrics(boolean success) { 191 this(success, true); 192 } 193 194 public ProcedureMetrics(boolean success, boolean yield) { 195 this(success, yield, 1); 196 } 197 198 public ProcedureMetrics(boolean success, boolean yield, int yieldCount) { 199 this(success, yield, yieldCount, null); 200 } 201 202 public ProcedureMetrics(boolean success, ProcedureMetrics[] subprocs) { 203 this(success, false, 1, subprocs); 204 } 205 206 public ProcedureMetrics(boolean success, boolean yield, int yieldCount, 207 ProcedureMetrics[] subprocs) { 208 this.success = success; 209 this.yield = yield; 210 this.yieldCount = yieldCount; 211 this.subprocs = subprocs; 212 yieldNum = 0; 213 } 214 215 @Override 216 protected void updateMetricsOnSubmit(TestProcEnv env) { 217 beginCount++; 218 } 219 220 @Override 221 protected Procedure[] execute(TestProcEnv env) 222 throws ProcedureYieldException, ProcedureSuspendedException, InterruptedException { 223 if (this.yield) { 224 if (yieldNum < yieldCount) { 225 yieldNum++; 226 throw new ProcedureYieldException(); 227 } 228 } 229 if (!this.success) { 230 setFailure("Failed", new InterruptedException("Failed")); 231 return null; 232 } 233 return subprocs; 234 } 235 236 @Override 237 protected void rollback(TestProcEnv env) throws IOException, InterruptedException { 238 } 239 240 @Override 241 protected boolean abort(TestProcEnv env) { 242 return false; 243 } 244 245 @Override 246 protected void updateMetricsOnFinish(final TestProcEnv env, final long time, boolean success) { 247 if (success) { 248 successCount++; 249 } else { 250 failedCount++; 251 } 252 } 253 } 254}