1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18 package org.apache.hadoop.hbase.client;
19
20 import static org.junit.Assert.assertTrue;
21
22 import java.io.IOException;
23
24 import org.apache.commons.logging.Log;
25 import org.apache.commons.logging.LogFactory;
26 import org.apache.commons.logging.impl.Log4JLogger;
27 import org.apache.hadoop.conf.Configuration;
28 import org.apache.hadoop.hbase.HBaseTestingUtility;
29 import org.apache.hadoop.hbase.HConstants;
30 import org.apache.hadoop.hbase.testclassification.MediumTests;
31 import org.apache.hadoop.hbase.MiniHBaseCluster.MiniHBaseClusterRegionServer;
32 import org.apache.hadoop.hbase.TableName;
33 import org.apache.hadoop.hbase.CoordinatedStateManager;
34 import org.apache.hadoop.hbase.ipc.AbstractRpcClient;
35 import org.apache.hadoop.hbase.ipc.RpcServer;
36 import org.apache.hadoop.hbase.protobuf.generated.ClientProtos.ScanRequest;
37 import org.apache.hadoop.hbase.protobuf.generated.ClientProtos.ScanResponse;
38 import org.apache.hadoop.hbase.regionserver.HRegionServer;
39 import org.apache.hadoop.hbase.regionserver.RSRpcServices;
40 import org.apache.hadoop.hbase.util.Bytes;
41 import org.apache.log4j.Level;
42 import org.junit.AfterClass;
43 import org.junit.BeforeClass;
44 import org.junit.Test;
45 import org.junit.experimental.categories.Category;
46
47 import com.google.protobuf.RpcController;
48 import com.google.protobuf.ServiceException;
49
50
51
52
53
54 @Category(MediumTests.class)
55 public class TestClientScannerRPCTimeout {
56 private static final Log LOG = LogFactory.getLog(TestClientScannerRPCTimeout.class);
57 private final static HBaseTestingUtility TEST_UTIL = new HBaseTestingUtility();
58 private static final byte[] FAMILY = Bytes.toBytes("testFamily");
59 private static final byte[] QUALIFIER = Bytes.toBytes("testQualifier");
60 private static final byte[] VALUE = Bytes.toBytes("testValue");
61 private static final int rpcTimeout = 2 * 1000;
62 private static final int CLIENT_RETRIES_NUMBER = 3;
63
64 @BeforeClass
65 public static void setUpBeforeClass() throws Exception {
66 ((Log4JLogger)RpcServer.LOG).getLogger().setLevel(Level.ALL);
67 ((Log4JLogger)AbstractRpcClient.LOG).getLogger().setLevel(Level.ALL);
68 ((Log4JLogger)ScannerCallable.LOG).getLogger().setLevel(Level.ALL);
69 Configuration conf = TEST_UTIL.getConfiguration();
70
71 conf.setInt("hbase.regionserver.msginterval", 3 * 10000);
72 conf.setInt(HConstants.HBASE_RPC_TIMEOUT_KEY, rpcTimeout);
73 conf.setStrings(HConstants.REGION_SERVER_IMPL, RegionServerWithScanTimeout.class.getName());
74 conf.setInt(HConstants.HBASE_CLIENT_RETRIES_NUMBER, CLIENT_RETRIES_NUMBER);
75 conf.setInt(HConstants.HBASE_CLIENT_PAUSE, 1000);
76 TEST_UTIL.startMiniCluster(1);
77 }
78
79 @AfterClass
80 public static void tearDownAfterClass() throws Exception {
81 TEST_UTIL.shutdownMiniCluster();
82 }
83
84 @Test
85 public void testScannerNextRPCTimesout() throws Exception {
86 final TableName TABLE_NAME = TableName.valueOf("testScannerNextRPCTimesout");
87 Table ht = TEST_UTIL.createTable(TABLE_NAME, FAMILY);
88 byte[] r0 = Bytes.toBytes("row-0");
89 byte[] r1 = Bytes.toBytes("row-1");
90 byte[] r2 = Bytes.toBytes("row-2");
91 byte[] r3 = Bytes.toBytes("row-3");
92 putToTable(ht, r0);
93 putToTable(ht, r1);
94 putToTable(ht, r2);
95 putToTable(ht, r3);
96 LOG.info("Wrote our three values");
97 RSRpcServicesWithScanTimeout.seqNoToSleepOn = 1;
98 Scan scan = new Scan();
99 scan.setCaching(1);
100 ResultScanner scanner = ht.getScanner(scan);
101 Result result = scanner.next();
102
103 assertTrue("Expected row: row-0", Bytes.equals(r0, result.getRow()));
104 result = scanner.next();
105 assertTrue("Expected row: row-1", Bytes.equals(r1, result.getRow()));
106 LOG.info("Got expected first row");
107 long t1 = System.currentTimeMillis();
108 result = scanner.next();
109 assertTrue((System.currentTimeMillis() - t1) > rpcTimeout);
110 assertTrue("Expected row: row-2", Bytes.equals(r2, result.getRow()));
111 RSRpcServicesWithScanTimeout.seqNoToSleepOn = -1;
112 result = scanner.next();
113 assertTrue("Expected row: row-3", Bytes.equals(r3, result.getRow()));
114 scanner.close();
115
116
117 scanner = ht.getScanner(scan);
118 RSRpcServicesWithScanTimeout.sleepAlways = true;
119 RSRpcServicesWithScanTimeout.tryNumber = 0;
120 try {
121 result = scanner.next();
122 } catch (IOException ioe) {
123
124 LOG.info("Failed after maximal attempts=" + CLIENT_RETRIES_NUMBER, ioe);
125 }
126 assertTrue("Expected maximal try number=" + CLIENT_RETRIES_NUMBER
127 + ", actual =" + RSRpcServicesWithScanTimeout.tryNumber,
128 RSRpcServicesWithScanTimeout.tryNumber <= CLIENT_RETRIES_NUMBER);
129 }
130
131 private void putToTable(Table ht, byte[] rowkey) throws IOException {
132 Put put = new Put(rowkey);
133 put.add(FAMILY, QUALIFIER, VALUE);
134 ht.put(put);
135 }
136
137 private static class RegionServerWithScanTimeout extends MiniHBaseClusterRegionServer {
138 public RegionServerWithScanTimeout(Configuration conf, CoordinatedStateManager cp)
139 throws IOException, InterruptedException {
140 super(conf, cp);
141 }
142
143 protected RSRpcServices createRpcServices() throws IOException {
144 return new RSRpcServicesWithScanTimeout(this);
145 }
146 }
147
148 private static class RSRpcServicesWithScanTimeout extends RSRpcServices {
149 private long tableScannerId;
150 private boolean slept;
151 private static long seqNoToSleepOn = -1;
152 private static boolean sleepAlways = false;
153 private static int tryNumber = 0;
154
155 public RSRpcServicesWithScanTimeout(HRegionServer rs)
156 throws IOException {
157 super(rs);
158 }
159
160 @Override
161 public ScanResponse scan(final RpcController controller, final ScanRequest request)
162 throws ServiceException {
163 if (request.hasScannerId()) {
164 ScanResponse scanResponse = super.scan(controller, request);
165 if (this.tableScannerId == request.getScannerId() &&
166 (sleepAlways || (!slept && seqNoToSleepOn == request.getNextCallSeq()))) {
167 try {
168 LOG.info("SLEEPING " + (rpcTimeout + 500));
169 Thread.sleep(rpcTimeout + 500);
170 } catch (InterruptedException e) {
171 }
172 slept = true;
173 tryNumber++;
174 if (tryNumber > 2 * CLIENT_RETRIES_NUMBER) {
175 sleepAlways = false;
176 }
177 }
178 return scanResponse;
179 } else {
180 ScanResponse scanRes = super.scan(controller, request);
181 String regionName = Bytes.toString(request.getRegion().getValue().toByteArray());
182 if (!regionName.contains(TableName.META_TABLE_NAME.getNameAsString())) {
183 tableScannerId = scanRes.getScannerId();
184 }
185 return scanRes;
186 }
187 }
188 }
189 }