HBASE-2221 MR to copy a table
git-svn-id: https://svn.apache.org/repos/asf/hadoop/hbase/trunk@911621 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
4c7414ec20
commit
96c80b48de
|
@ -395,6 +395,7 @@ Release 0.21.0 - Unreleased
|
||||||
repo, etc. (Kay Kay via Stack)
|
repo, etc. (Kay Kay via Stack)
|
||||||
HBASE-2129 Simple Master/Slave replication
|
HBASE-2129 Simple Master/Slave replication
|
||||||
HBASE-2070 Collect HLogs and delete them after a period of time
|
HBASE-2070 Collect HLogs and delete them after a period of time
|
||||||
|
HBASE-2221 MR to copy a table
|
||||||
|
|
||||||
OPTIMIZATIONS
|
OPTIMIZATIONS
|
||||||
HBASE-410 [testing] Speed up the test suite
|
HBASE-410 [testing] Speed up the test suite
|
||||||
|
|
|
@ -86,6 +86,10 @@
|
||||||
rev="${hadoop-hdfs.version}" conf="common->default" changing="true" >
|
rev="${hadoop-hdfs.version}" conf="common->default" changing="true" >
|
||||||
<exclude conf="test"/>
|
<exclude conf="test"/>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency org="org.apache.hadoop" name="hadoop-mapred"
|
||||||
|
rev="${hadoop-mapred.version}" conf="common->default" changing="true" >
|
||||||
|
<exclude conf="test"/>
|
||||||
|
</dependency>
|
||||||
<dependency org="org.jruby" name="jruby-complete"
|
<dependency org="org.jruby" name="jruby-complete"
|
||||||
rev="${jruby.version}" conf="common->default" />
|
rev="${jruby.version}" conf="common->default" />
|
||||||
|
|
||||||
|
@ -99,6 +103,8 @@
|
||||||
rev="${hadoop-core.version}" conf="test->default" transitive="false" changing="true" />
|
rev="${hadoop-core.version}" conf="test->default" transitive="false" changing="true" />
|
||||||
<dependency org="org.apache.hadoop" name="hadoop-hdfs-test"
|
<dependency org="org.apache.hadoop" name="hadoop-hdfs-test"
|
||||||
rev="${hadoop-hdfs.version}" conf="test->default" transitive="false" changing="true"/>
|
rev="${hadoop-hdfs.version}" conf="test->default" transitive="false" changing="true"/>
|
||||||
|
<dependency org="org.apache.hadoop" name="hadoop-mapred-test"
|
||||||
|
rev="${hadoop-mapred.version}" conf="test->default" transitive="false" changing="true"/>
|
||||||
<dependency org="log4j" name="log4j"
|
<dependency org="log4j" name="log4j"
|
||||||
rev="${log4j.version}" conf="test->master">
|
rev="${log4j.version}" conf="test->master">
|
||||||
<exclude conf="jmx,mail,jms"/>
|
<exclude conf="jmx,mail,jms"/>
|
||||||
|
|
|
@ -19,47 +19,131 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.hadoop.hbase.replication;
|
package org.apache.hadoop.hbase.replication;
|
||||||
|
|
||||||
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 org.junit.Assert.assertArrayEquals;
|
|
||||||
|
|
||||||
import org.junit.*;
|
|
||||||
import org.apache.hadoop.hbase.*;
|
|
||||||
import org.apache.hadoop.hbase.client.*;
|
|
||||||
import org.apache.hadoop.hbase.util.Bytes;
|
|
||||||
import org.apache.hadoop.hbase.regionserver.replication.ReplicationRegionServer;
|
|
||||||
import org.apache.hadoop.hbase.zookeeper.ZooKeeperWrapper;
|
|
||||||
import org.apache.hadoop.hbase.ipc.ReplicationRegionInterface;
|
|
||||||
import org.apache.hadoop.conf.Configuration;
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.hbase.EmptyWatcher;
|
||||||
|
import org.apache.hadoop.hbase.HBaseConfiguration;
|
||||||
|
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
||||||
|
import org.apache.hadoop.hbase.HColumnDescriptor;
|
||||||
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
|
import org.apache.hadoop.hbase.HTableDescriptor;
|
||||||
|
import org.apache.hadoop.hbase.MiniZooKeeperCluster;
|
||||||
|
import org.apache.hadoop.hbase.client.Delete;
|
||||||
|
import org.apache.hadoop.hbase.client.Get;
|
||||||
|
import org.apache.hadoop.hbase.client.HBaseAdmin;
|
||||||
|
import org.apache.hadoop.hbase.client.HTable;
|
||||||
|
import org.apache.hadoop.hbase.client.Put;
|
||||||
|
import org.apache.hadoop.hbase.client.Result;
|
||||||
|
import org.apache.hadoop.hbase.client.ResultScanner;
|
||||||
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
|
import org.apache.hadoop.hbase.ipc.ReplicationRegionInterface;
|
||||||
|
import org.apache.hadoop.hbase.mapreduce.CopyTable;
|
||||||
|
import org.apache.hadoop.hbase.regionserver.replication.ReplicationRegionServer;
|
||||||
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
|
import org.apache.hadoop.hbase.zookeeper.ZooKeeperWrapper;
|
||||||
|
import org.apache.hadoop.mapreduce.Job;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import static org.junit.Assert.assertArrayEquals;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
public class TestReplication implements HConstants{
|
public class TestReplication implements HConstants{
|
||||||
|
|
||||||
protected static final Log LOG = LogFactory.getLog(TestReplication.class);
|
protected static final Log LOG = LogFactory.getLog(TestReplication.class);
|
||||||
|
|
||||||
private Configuration conf1;
|
private static Configuration conf1;
|
||||||
private Configuration conf2;
|
private static Configuration conf2;
|
||||||
|
|
||||||
private ZooKeeperWrapper zkw1;
|
private static ZooKeeperWrapper zkw1;
|
||||||
private ZooKeeperWrapper zkw2;
|
private static ZooKeeperWrapper zkw2;
|
||||||
|
|
||||||
private HBaseTestingUtility utility1;
|
private static HBaseTestingUtility utility1;
|
||||||
private HBaseTestingUtility utility2;
|
private static HBaseTestingUtility utility2;
|
||||||
|
private static final int NB_ROWS_IN_BATCH = 100;
|
||||||
private final int NB_ROWS_IN_BATCH = 100;
|
private static final long SLEEP_TIME = 500; //ms
|
||||||
private final long SLEEP_TIME = 500;
|
private static final int NB_RETRIES = 10;
|
||||||
private final int NB_RETRIES = 5;
|
|
||||||
|
|
||||||
|
private static final byte[] tableName = Bytes.toBytes("test");
|
||||||
|
private static final byte[] famName = Bytes.toBytes("f");
|
||||||
|
private static final byte[] row = Bytes.toBytes("row");
|
||||||
|
private static final byte[] noRepfamName = Bytes.toBytes("norep");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws java.lang.Exception
|
* @throws java.lang.Exception
|
||||||
*/
|
*/
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void setUpBeforeClass() throws Exception {
|
public static void setUpBeforeClass() throws Exception {
|
||||||
|
try {
|
||||||
|
conf1 = HBaseConfiguration.create();
|
||||||
|
conf1.set(REGION_SERVER_CLASS, ReplicationRegionInterface.class
|
||||||
|
.getName());
|
||||||
|
conf1.set(REGION_SERVER_IMPL, ReplicationRegionServer.class
|
||||||
|
.getName());
|
||||||
|
conf1.set(ZOOKEEPER_ZNODE_PARENT, "/1");
|
||||||
|
|
||||||
|
utility1 = new HBaseTestingUtility(conf1);
|
||||||
|
utility1.startMiniZKCluster();
|
||||||
|
MiniZooKeeperCluster miniZK = utility1.getZkCluster();
|
||||||
|
zkw1 = new ZooKeeperWrapper(conf1, EmptyWatcher.instance);
|
||||||
|
zkw1.writeZNode("/1", "replication", "");
|
||||||
|
zkw1.writeZNode("/1/replication", "master",
|
||||||
|
conf1.get(ZOOKEEPER_QUORUM)+":" +
|
||||||
|
conf1.get("hbase.zookeeper.property.clientPort")+":/1");
|
||||||
|
setIsReplication("true");
|
||||||
|
|
||||||
|
LOG.info("Setup first Zk");
|
||||||
|
|
||||||
|
conf2 = HBaseConfiguration.create();
|
||||||
|
conf2.set(REGION_SERVER_CLASS, ReplicationRegionInterface.class
|
||||||
|
.getName());
|
||||||
|
conf2.set(REGION_SERVER_IMPL, ReplicationRegionServer.class
|
||||||
|
.getName());
|
||||||
|
conf2.set(ZOOKEEPER_ZNODE_PARENT, "/2");
|
||||||
|
|
||||||
|
utility2 = new HBaseTestingUtility(conf2);
|
||||||
|
utility2.setZkCluster(miniZK);
|
||||||
|
zkw2 = new ZooKeeperWrapper(conf2, EmptyWatcher.instance);
|
||||||
|
zkw2.writeZNode("/2", "replication", "");
|
||||||
|
zkw2.writeZNode("/2/replication", "master",
|
||||||
|
conf1.get(ZOOKEEPER_QUORUM)+":" +
|
||||||
|
conf1.get("hbase.zookeeper.property.clientPort")+":/1");
|
||||||
|
|
||||||
|
zkw1.writeZNode("/1/replication/peers", "test",
|
||||||
|
conf2.get(ZOOKEEPER_QUORUM)+":" +
|
||||||
|
conf2.get("hbase.zookeeper.property.clientPort")+":/2");
|
||||||
|
|
||||||
|
LOG.info("Setup second Zk");
|
||||||
|
|
||||||
|
utility1.startMiniCluster();
|
||||||
|
utility2.startMiniCluster();
|
||||||
|
|
||||||
|
utility1.startMiniMapReduceCluster();
|
||||||
|
|
||||||
|
HTableDescriptor table = new HTableDescriptor(tableName);
|
||||||
|
HColumnDescriptor fam = new HColumnDescriptor(famName);
|
||||||
|
fam.setScope(REPLICATION_SCOPE_GLOBAL);
|
||||||
|
table.addFamily(fam);
|
||||||
|
fam = new HColumnDescriptor(noRepfamName);
|
||||||
|
table.addFamily(fam);
|
||||||
|
|
||||||
|
HBaseAdmin admin1 = new HBaseAdmin(conf1);
|
||||||
|
HBaseAdmin admin2 = new HBaseAdmin(conf2);
|
||||||
|
admin1.createTable(table);
|
||||||
|
admin2.createTable(table);
|
||||||
|
|
||||||
|
} catch (Exception ex) { ex.printStackTrace(); throw ex; }
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void setIsReplication(String bool) throws Exception {
|
||||||
|
zkw1.writeZNode("/1/replication", "state", bool);
|
||||||
|
// Takes some ms for ZK to fire the watcher
|
||||||
|
Thread.sleep(100);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -67,88 +151,29 @@ public class TestReplication implements HConstants{
|
||||||
*/
|
*/
|
||||||
@AfterClass
|
@AfterClass
|
||||||
public static void tearDownAfterClass() throws Exception {
|
public static void tearDownAfterClass() throws Exception {
|
||||||
|
utility1.shutdownMiniCluster();
|
||||||
|
utility2.shutdownMiniCluster();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws java.lang.Exception
|
* @throws java.lang.Exception
|
||||||
*/
|
*/
|
||||||
@Before
|
@Before
|
||||||
public void setUp() throws Exception {
|
public void setUp() throws Exception {}
|
||||||
try {
|
|
||||||
conf1 = HBaseConfiguration.create();
|
|
||||||
conf1.set(REGION_SERVER_CLASS, ReplicationRegionInterface.class
|
|
||||||
.getName());
|
|
||||||
conf1.set(REGION_SERVER_IMPL, ReplicationRegionServer.class
|
|
||||||
.getName());
|
|
||||||
conf1.set(ZOOKEEPER_ZNODE_PARENT, "/1");
|
|
||||||
|
|
||||||
utility1 = new HBaseTestingUtility(conf1);
|
|
||||||
utility1.startMiniZKCluster();
|
|
||||||
MiniZooKeeperCluster miniZK = utility1.getZkCluster();
|
|
||||||
zkw1 = new ZooKeeperWrapper(conf1, EmptyWatcher.instance);
|
|
||||||
zkw1.writeZNode("/1", "replication", "");
|
|
||||||
zkw1.writeZNode("/1/replication", "master",
|
|
||||||
conf1.get(ZOOKEEPER_QUORUM)+":" +
|
|
||||||
conf1.get("hbase.zookeeper.property.clientPort")+":/1");
|
|
||||||
setIsReplication("true");
|
|
||||||
|
|
||||||
|
|
||||||
LOG.info("Setup first Zk");
|
|
||||||
|
|
||||||
conf2 = HBaseConfiguration.create();
|
|
||||||
conf2.set(REGION_SERVER_CLASS, ReplicationRegionInterface.class
|
|
||||||
.getName());
|
|
||||||
conf2.set(REGION_SERVER_IMPL, ReplicationRegionServer.class
|
|
||||||
.getName());
|
|
||||||
conf2.set(ZOOKEEPER_ZNODE_PARENT, "/2");
|
|
||||||
|
|
||||||
utility2 = new HBaseTestingUtility(conf2);
|
|
||||||
utility2.setZkCluster(miniZK);
|
|
||||||
zkw2 = new ZooKeeperWrapper(conf2, EmptyWatcher.instance);
|
|
||||||
zkw2.writeZNode("/2", "replication", "");
|
|
||||||
zkw2.writeZNode("/2/replication", "master",
|
|
||||||
conf1.get(ZOOKEEPER_QUORUM)+":" +
|
|
||||||
conf1.get("hbase.zookeeper.property.clientPort")+":/1");
|
|
||||||
|
|
||||||
zkw1.writeZNode("/1/replication/peers", "test",
|
|
||||||
conf2.get(ZOOKEEPER_QUORUM)+":" +
|
|
||||||
conf2.get("hbase.zookeeper.property.clientPort")+":/2");
|
|
||||||
|
|
||||||
LOG.info("Setup second Zk");
|
|
||||||
} catch (Exception ex) { ex.printStackTrace(); throw ex; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws java.lang.Exception
|
* @throws java.lang.Exception
|
||||||
*/
|
*/
|
||||||
@After
|
@After
|
||||||
public void tearDown() throws Exception {
|
public void tearDown() throws Exception {
|
||||||
utility1.shutdownMiniCluster();
|
setIsReplication("false");
|
||||||
utility2.shutdownMiniCluster();
|
utility1.truncateTable(tableName);
|
||||||
|
utility2.truncateTable(tableName);
|
||||||
|
setIsReplication("true");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testReplication() throws Exception {
|
public void testReplication() throws Exception {
|
||||||
utility1.startMiniCluster();
|
|
||||||
utility2.startMiniCluster();
|
|
||||||
|
|
||||||
byte[] tableName = Bytes.toBytes("test");
|
|
||||||
byte[] famName = Bytes.toBytes("f");
|
|
||||||
byte[] noRepfamName = Bytes.toBytes("norep");
|
|
||||||
byte[] row = Bytes.toBytes("row");
|
|
||||||
|
|
||||||
HTableDescriptor table = new HTableDescriptor(tableName);
|
|
||||||
HColumnDescriptor fam = new HColumnDescriptor(famName);
|
|
||||||
fam.setScope(REPLICATION_SCOPE_GLOBAL);
|
|
||||||
table.addFamily(fam);
|
|
||||||
fam = new HColumnDescriptor(noRepfamName);
|
|
||||||
table.addFamily(fam);
|
|
||||||
|
|
||||||
HBaseAdmin admin1 = new HBaseAdmin(conf1);
|
|
||||||
HBaseAdmin admin2 = new HBaseAdmin(conf2);
|
|
||||||
admin1.createTable(table);
|
|
||||||
admin2.createTable(table);
|
|
||||||
|
|
||||||
Put put = new Put(row);
|
Put put = new Put(row);
|
||||||
put.add(famName, row, row);
|
put.add(famName, row, row);
|
||||||
|
|
||||||
|
@ -157,12 +182,12 @@ public class TestReplication implements HConstants{
|
||||||
|
|
||||||
HTable table2 = new HTable(conf2, tableName);
|
HTable table2 = new HTable(conf2, tableName);
|
||||||
Get get = new Get(row);
|
Get get = new Get(row);
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i==NB_RETRIES-1) {
|
if (i==NB_RETRIES-1) {
|
||||||
fail("Waited too much time for put replication");
|
fail("Waited too much time for put replication");
|
||||||
}
|
}
|
||||||
Result res = table2.get(get);
|
Result res = table2.get(get);
|
||||||
if(res.size() == 0) {
|
if (res.size() == 0) {
|
||||||
LOG.info("Row not available");
|
LOG.info("Row not available");
|
||||||
Thread.sleep(SLEEP_TIME);
|
Thread.sleep(SLEEP_TIME);
|
||||||
} else {
|
} else {
|
||||||
|
@ -176,12 +201,12 @@ public class TestReplication implements HConstants{
|
||||||
|
|
||||||
table2 = new HTable(conf2, tableName);
|
table2 = new HTable(conf2, tableName);
|
||||||
get = new Get(row);
|
get = new Get(row);
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i==NB_RETRIES-1) {
|
if (i==NB_RETRIES-1) {
|
||||||
fail("Waited too much time for del replication");
|
fail("Waited too much time for del replication");
|
||||||
}
|
}
|
||||||
Result res = table2.get(get);
|
Result res = table2.get(get);
|
||||||
if(res.size() >= 1) {
|
if (res.size() >= 1) {
|
||||||
LOG.info("Row not deleted");
|
LOG.info("Row not deleted");
|
||||||
Thread.sleep(SLEEP_TIME);
|
Thread.sleep(SLEEP_TIME);
|
||||||
} else {
|
} else {
|
||||||
|
@ -200,14 +225,14 @@ public class TestReplication implements HConstants{
|
||||||
|
|
||||||
Scan scan = new Scan();
|
Scan scan = new Scan();
|
||||||
|
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i==NB_RETRIES-1) {
|
if (i==NB_RETRIES-1) {
|
||||||
fail("Waited too much time for normal batch replication");
|
fail("Waited too much time for normal batch replication");
|
||||||
}
|
}
|
||||||
ResultScanner scanner = table2.getScanner(scan);
|
ResultScanner scanner = table2.getScanner(scan);
|
||||||
Result[] res = scanner.next(NB_ROWS_IN_BATCH);
|
Result[] res = scanner.next(NB_ROWS_IN_BATCH);
|
||||||
scanner.close();
|
scanner.close();
|
||||||
if(res.length != NB_ROWS_IN_BATCH) {
|
if (res.length != NB_ROWS_IN_BATCH) {
|
||||||
LOG.info("Only got " + res.length + " rows");
|
LOG.info("Only got " + res.length + " rows");
|
||||||
Thread.sleep(SLEEP_TIME);
|
Thread.sleep(SLEEP_TIME);
|
||||||
} else {
|
} else {
|
||||||
|
@ -220,17 +245,13 @@ public class TestReplication implements HConstants{
|
||||||
// Test stopping replication
|
// Test stopping replication
|
||||||
setIsReplication("false");
|
setIsReplication("false");
|
||||||
|
|
||||||
// Takes some ms for ZK to fire the watcher
|
|
||||||
Thread.sleep(100);
|
|
||||||
|
|
||||||
|
|
||||||
put = new Put(Bytes.toBytes("stop start"));
|
put = new Put(Bytes.toBytes("stop start"));
|
||||||
put.add(famName, row, row);
|
put.add(famName, row, row);
|
||||||
table1.put(put);
|
table1.put(put);
|
||||||
|
|
||||||
get = new Get(Bytes.toBytes("stop start"));
|
get = new Get(Bytes.toBytes("stop start"));
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i==NB_RETRIES-1) {
|
if (i==NB_RETRIES-1) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Result res = table2.get(get);
|
Result res = table2.get(get);
|
||||||
|
@ -244,15 +265,12 @@ public class TestReplication implements HConstants{
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test restart replication
|
// Test restart replication
|
||||||
|
|
||||||
setIsReplication("true");
|
setIsReplication("true");
|
||||||
|
|
||||||
Thread.sleep(100);
|
|
||||||
|
|
||||||
table1.put(put);
|
table1.put(put);
|
||||||
|
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i==NB_RETRIES-1) {
|
if (i==NB_RETRIES-1) {
|
||||||
fail("Waited too much time for put replication");
|
fail("Waited too much time for put replication");
|
||||||
}
|
}
|
||||||
Result res = table2.get(get);
|
Result res = table2.get(get);
|
||||||
|
@ -270,12 +288,12 @@ public class TestReplication implements HConstants{
|
||||||
table1.put(put);
|
table1.put(put);
|
||||||
|
|
||||||
get = new Get(Bytes.toBytes("do not rep"));
|
get = new Get(Bytes.toBytes("do not rep"));
|
||||||
for(int i = 0; i < NB_RETRIES; i++) {
|
for (int i = 0; i < NB_RETRIES; i++) {
|
||||||
if(i == NB_RETRIES-1) {
|
if (i == NB_RETRIES-1) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Result res = table2.get(get);
|
Result res = table2.get(get);
|
||||||
if(res.size() >= 1) {
|
if (res.size() >= 1) {
|
||||||
fail("Not supposed to be replicated");
|
fail("Not supposed to be replicated");
|
||||||
} else {
|
} else {
|
||||||
LOG.info("Row not replicated, let's wait a bit more...");
|
LOG.info("Row not replicated, let's wait a bit more...");
|
||||||
|
@ -285,7 +303,29 @@ public class TestReplication implements HConstants{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setIsReplication(String bool) throws Exception{
|
@Test
|
||||||
zkw1.writeZNode("/1/replication", "state", bool);
|
public void testMRCopy() throws Exception {
|
||||||
|
setIsReplication("false");
|
||||||
|
HTable table1 = new HTable(conf1, tableName);
|
||||||
|
for (int i = 0; i < NB_ROWS_IN_BATCH; i++) {
|
||||||
|
Put put = new Put(Bytes.toBytes(i));
|
||||||
|
put.add(famName, row, row);
|
||||||
|
table1.put(put);
|
||||||
|
}
|
||||||
|
String[] args = new String[] {
|
||||||
|
"--rs.class="+ReplicationRegionInterface.class.getName(),
|
||||||
|
"--rs.impl="+ReplicationRegionServer.class.getName(),
|
||||||
|
"--peer.adr="+conf2.get(ZOOKEEPER_QUORUM)+":/2",
|
||||||
|
"--families=f", "test"};
|
||||||
|
Job job = CopyTable.createSubmittableJob(conf1, args);
|
||||||
|
assertTrue(job.waitForCompletion(true));
|
||||||
|
|
||||||
|
HTable table2 = new HTable(conf2, tableName);
|
||||||
|
Scan scan = new Scan();
|
||||||
|
ResultScanner scanner = table2.getScanner(scan);
|
||||||
|
Result[] res = scanner.next(NB_ROWS_IN_BATCH);
|
||||||
|
scanner.close();
|
||||||
|
assertEquals(NB_ROWS_IN_BATCH, res.length);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,207 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2010 The Apache Software Foundation
|
||||||
|
*
|
||||||
|
* 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.hadoop.hbase.mapreduce;
|
||||||
|
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.hbase.HBaseConfiguration;
|
||||||
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
|
import org.apache.hadoop.mapreduce.Cluster;
|
||||||
|
import org.apache.hadoop.mapreduce.Job;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tool used to copy a table to another one which can be on a different setup.
|
||||||
|
* It is also configurable with a start and time as well as a specification
|
||||||
|
* of the region server implementation if different from the local cluster.
|
||||||
|
*/
|
||||||
|
public class CopyTable {
|
||||||
|
|
||||||
|
final static String NAME = "Copy Table";
|
||||||
|
static String rsClass = null;
|
||||||
|
static String rsImpl = null;
|
||||||
|
static long startTime = 0;
|
||||||
|
static long endTime = 0;
|
||||||
|
static String tableName = null;
|
||||||
|
static String newTableName = null;
|
||||||
|
static String peerAddress = null;
|
||||||
|
static String families = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets up the actual job.
|
||||||
|
*
|
||||||
|
* @param conf The current configuration.
|
||||||
|
* @param args The command line parameters.
|
||||||
|
* @return The newly created job.
|
||||||
|
* @throws IOException When setting up the job fails.
|
||||||
|
*/
|
||||||
|
public static Job createSubmittableJob(Configuration conf, String[] args)
|
||||||
|
throws IOException {
|
||||||
|
if (!doCommandLine(args)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Cluster mrCluster = new Cluster(conf);
|
||||||
|
Job job = Job.getInstance(mrCluster, conf);
|
||||||
|
job.setJobName(NAME + "_" + tableName);
|
||||||
|
job.setJarByClass(CopyTable.class);
|
||||||
|
Scan scan = new Scan();
|
||||||
|
if (startTime != 0) {
|
||||||
|
scan.setTimeRange(startTime,
|
||||||
|
endTime == 0 ? HConstants.LATEST_TIMESTAMP : endTime);
|
||||||
|
}
|
||||||
|
if(families != null) {
|
||||||
|
String[] fams = families.split(",");
|
||||||
|
for(String fam : fams) {
|
||||||
|
scan.addFamily(Bytes.toBytes(fam));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
TableMapReduceUtil.initTableMapperJob(tableName, scan,
|
||||||
|
Import.Importer.class, null, null, job);
|
||||||
|
TableMapReduceUtil.initTableReducerJob(
|
||||||
|
newTableName == null ? tableName : newTableName, null, job,
|
||||||
|
null, peerAddress, rsClass, rsImpl);
|
||||||
|
job.setNumReduceTasks(0);
|
||||||
|
return job;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @param errorMsg Error message. Can be null.
|
||||||
|
*/
|
||||||
|
private static void printUsage(final String errorMsg) {
|
||||||
|
if (errorMsg != null && errorMsg.length() > 0) {
|
||||||
|
System.err.println("ERROR: " + errorMsg);
|
||||||
|
}
|
||||||
|
System.err.println("Usage: CopyTable [--rs.class=CLASS] " +
|
||||||
|
"[--rs.impl=IMPL] [--starttime=X] [--endtime=Y] " +
|
||||||
|
"[--new.name=NEW] [--peer.adr=ADR] <tablename>");
|
||||||
|
System.err.println();
|
||||||
|
System.err.println("Options:");
|
||||||
|
System.err.println(" rs.class hbase.regionserver.class of the peer cluster");
|
||||||
|
System.err.println(" specify if different from current cluster");
|
||||||
|
System.err.println(" rs.impl hbase.regionserver.impl of the peer cluster");
|
||||||
|
System.err.println(" starttime beginning of the time range");
|
||||||
|
System.err.println(" without endtime means from starttime to forever");
|
||||||
|
System.err.println(" endtime end of the time range");
|
||||||
|
System.err.println(" new.name new table's name");
|
||||||
|
System.err.println(" peer.adr Address of the peer cluster given in the format");
|
||||||
|
System.err.println(" hbase.zookeeer.quorum:zookeeper.znode.parent");
|
||||||
|
System.err.println(" families comma-seperated list of families to copy");
|
||||||
|
System.err.println();
|
||||||
|
System.err.println("Args:");
|
||||||
|
System.err.println(" tablename Name of the table to copy");
|
||||||
|
System.err.println();
|
||||||
|
System.err.println("Examples:");
|
||||||
|
System.err.println(" To copy 'TestTable' to a cluster that uses replication for a 1 hour window:");
|
||||||
|
System.err.println(" $ bin/hbase " +
|
||||||
|
"org.apache.hadoop.hbase.mapreduce.CopyTable --rs.class=org.apache.hadoop.hbase.ipc.ReplicationRegionInterface " +
|
||||||
|
"--rs.impl=org.apache.hadoop.hbase.regionserver.replication.ReplicationRegionServer --starttime=1265875194289 --endtime=1265878794289 " +
|
||||||
|
"--peer.adr=server1,server2,server3:/hbase TestTable ");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean doCommandLine(final String[] args) {
|
||||||
|
// Process command-line args. TODO: Better cmd-line processing
|
||||||
|
// (but hopefully something not as painful as cli options).
|
||||||
|
if (args.length < 1) {
|
||||||
|
printUsage(null);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
for (int i = 0; i < args.length; i++) {
|
||||||
|
String cmd = args[i];
|
||||||
|
if (cmd.equals("-h") || cmd.startsWith("--h")) {
|
||||||
|
printUsage(null);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String rsClassArgKey = "--rs.class=";
|
||||||
|
if (cmd.startsWith(rsClassArgKey)) {
|
||||||
|
rsClass = cmd.substring(rsClassArgKey.length());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String rsImplArgKey = "--rs.impl=";
|
||||||
|
if (cmd.startsWith(rsImplArgKey)) {
|
||||||
|
rsImpl = cmd.substring(rsImplArgKey.length());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String startTimeArgKey = "--starttime=";
|
||||||
|
if (cmd.startsWith(startTimeArgKey)) {
|
||||||
|
startTime = Long.parseLong(cmd.substring(startTimeArgKey.length()));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String endTimeArgKey = "--endtime=";
|
||||||
|
if (cmd.startsWith(endTimeArgKey)) {
|
||||||
|
endTime = Long.parseLong(cmd.substring(endTimeArgKey.length()));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String newNameArgKey = "--new.name=";
|
||||||
|
if (cmd.startsWith(rsClassArgKey)) {
|
||||||
|
newTableName = cmd.substring(newNameArgKey.length());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String peerAdrArgKey = "--peer.adr=";
|
||||||
|
if (cmd.startsWith(peerAdrArgKey)) {
|
||||||
|
peerAddress = cmd.substring(peerAdrArgKey.length());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final String familiesArgKey = "--families=";
|
||||||
|
if (cmd.startsWith(familiesArgKey)) {
|
||||||
|
families = cmd.substring(familiesArgKey.length());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i == args.length-1) {
|
||||||
|
tableName = cmd;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (newTableName == null && peerAddress == null) {
|
||||||
|
printUsage("At least a new table name or a " +
|
||||||
|
"peer address must be specified");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
printUsage("Can't start because " + e.getMessage());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main entry point.
|
||||||
|
*
|
||||||
|
* @param args The command line parameters.
|
||||||
|
* @throws Exception When running the job fails.
|
||||||
|
*/
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
Configuration conf = HBaseConfiguration.create();
|
||||||
|
Job job = createSubmittableJob(conf, args);
|
||||||
|
if (job != null) {
|
||||||
|
System.exit(job.waitForCompletion(true) ? 0 : 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -36,6 +36,8 @@ public class Driver {
|
||||||
"Count rows in HBase table");
|
"Count rows in HBase table");
|
||||||
pgd.addClass(Export.NAME, Export.class, "Write table data to HDFS.");
|
pgd.addClass(Export.NAME, Export.class, "Write table data to HDFS.");
|
||||||
pgd.addClass(Import.NAME, Import.class, "Import data written by Export.");
|
pgd.addClass(Import.NAME, Import.class, "Import data written by Export.");
|
||||||
|
pgd.addClass(CopyTable.NAME, CopyTable.class,
|
||||||
|
"Export a table from local cluster to peer cluster");
|
||||||
pgd.driver(args);
|
pgd.driver(args);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ import java.io.DataOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
import org.apache.hadoop.hbase.HBaseConfiguration;
|
import org.apache.hadoop.hbase.HBaseConfiguration;
|
||||||
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
import org.apache.hadoop.hbase.client.HTable;
|
import org.apache.hadoop.hbase.client.HTable;
|
||||||
import org.apache.hadoop.hbase.client.Scan;
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
import org.apache.hadoop.hbase.io.ImmutableBytesWritable;
|
import org.apache.hadoop.hbase.io.ImmutableBytesWritable;
|
||||||
|
@ -33,6 +34,7 @@ import org.apache.hadoop.hbase.util.Base64;
|
||||||
import org.apache.hadoop.io.Writable;
|
import org.apache.hadoop.io.Writable;
|
||||||
import org.apache.hadoop.io.WritableComparable;
|
import org.apache.hadoop.io.WritableComparable;
|
||||||
import org.apache.hadoop.mapreduce.Job;
|
import org.apache.hadoop.mapreduce.Job;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility for {@link TableMapper} and {@link TableReducer}
|
* Utility for {@link TableMapper} and {@link TableReducer}
|
||||||
|
@ -109,6 +111,23 @@ public class TableMapReduceUtil {
|
||||||
initTableReducerJob(table, reducer, job, null);
|
initTableReducerJob(table, reducer, job, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use this before submitting a TableReduce job. It will
|
||||||
|
* appropriately set up the JobConf.
|
||||||
|
*
|
||||||
|
* @param table The output table.
|
||||||
|
* @param reducer The reducer class to use.
|
||||||
|
* @param job The current job to adjust.
|
||||||
|
* @param partitioner Partitioner to use. Pass <code>null</code> to use
|
||||||
|
* default partitioner.
|
||||||
|
* @throws IOException When determining the region count fails.
|
||||||
|
*/
|
||||||
|
public static void initTableReducerJob(String table,
|
||||||
|
Class<? extends TableReducer> reducer, Job job,
|
||||||
|
Class partitioner) throws IOException {
|
||||||
|
initTableReducerJob(table, reducer, job, null, null, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Use this before submitting a TableReduce job. It will
|
* Use this before submitting a TableReduce job. It will
|
||||||
* appropriately set up the JobConf.
|
* appropriately set up the JobConf.
|
||||||
|
@ -118,20 +137,38 @@ public class TableMapReduceUtil {
|
||||||
* @param job The current job to adjust.
|
* @param job The current job to adjust.
|
||||||
* @param partitioner Partitioner to use. Pass <code>null</code> to use
|
* @param partitioner Partitioner to use. Pass <code>null</code> to use
|
||||||
* default partitioner.
|
* default partitioner.
|
||||||
|
* @param quorumAddress Distant cluster to write to
|
||||||
|
* @param serverClass redefined hbase.regionserver.class
|
||||||
|
* @param serverImpl redefined hbase.regionserver.impl
|
||||||
* @throws IOException When determining the region count fails.
|
* @throws IOException When determining the region count fails.
|
||||||
*/
|
*/
|
||||||
public static void initTableReducerJob(String table,
|
public static void initTableReducerJob(String table,
|
||||||
Class<? extends TableReducer> reducer, Job job, Class partitioner)
|
Class<? extends TableReducer> reducer, Job job,
|
||||||
throws IOException {
|
Class partitioner, String quorumAddress, String serverClass,
|
||||||
|
String serverImpl) throws IOException {
|
||||||
|
|
||||||
|
Configuration conf = job.getConfiguration();
|
||||||
job.setOutputFormatClass(TableOutputFormat.class);
|
job.setOutputFormatClass(TableOutputFormat.class);
|
||||||
if (reducer != null) job.setReducerClass(reducer);
|
if (reducer != null) job.setReducerClass(reducer);
|
||||||
job.getConfiguration().set(TableOutputFormat.OUTPUT_TABLE, table);
|
conf.set(TableOutputFormat.OUTPUT_TABLE, table);
|
||||||
|
if (quorumAddress != null) {
|
||||||
|
if (quorumAddress.split(":").length == 2) {
|
||||||
|
conf.set(TableOutputFormat.QUORUM_ADDRESS, quorumAddress);
|
||||||
|
} else {
|
||||||
|
throw new IOException("Please specify the peer cluster as " +
|
||||||
|
HConstants.ZOOKEEPER_QUORUM+":"+HConstants.ZOOKEEPER_ZNODE_PARENT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (serverClass != null && serverImpl != null) {
|
||||||
|
conf.set(TableOutputFormat.REGION_SERVER_CLASS, serverClass);
|
||||||
|
conf.set(TableOutputFormat.REGION_SERVER_IMPL, serverImpl);
|
||||||
|
}
|
||||||
job.setOutputKeyClass(ImmutableBytesWritable.class);
|
job.setOutputKeyClass(ImmutableBytesWritable.class);
|
||||||
job.setOutputValueClass(Writable.class);
|
job.setOutputValueClass(Writable.class);
|
||||||
if (partitioner == HRegionPartitioner.class) {
|
if (partitioner == HRegionPartitioner.class) {
|
||||||
HBaseConfiguration.addHbaseResources(job.getConfiguration());
|
HBaseConfiguration.addHbaseResources(conf);
|
||||||
job.setPartitionerClass(HRegionPartitioner.class);
|
job.setPartitionerClass(HRegionPartitioner.class);
|
||||||
HTable outputTable = new HTable(job.getConfiguration(), table);
|
HTable outputTable = new HTable(conf, table);
|
||||||
int regions = outputTable.getRegionsInfo().size();
|
int regions = outputTable.getRegionsInfo().size();
|
||||||
if (job.getNumReduceTasks() > regions) {
|
if (job.getNumReduceTasks() > regions) {
|
||||||
job.setNumReduceTasks(outputTable.getRegionsInfo().size());
|
job.setNumReduceTasks(outputTable.getRegionsInfo().size());
|
||||||
|
|
|
@ -24,6 +24,7 @@ import java.io.IOException;
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.apache.hadoop.hbase.HBaseConfiguration;
|
import org.apache.hadoop.hbase.HBaseConfiguration;
|
||||||
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
import org.apache.hadoop.hbase.client.Delete;
|
import org.apache.hadoop.hbase.client.Delete;
|
||||||
import org.apache.hadoop.hbase.client.HTable;
|
import org.apache.hadoop.hbase.client.HTable;
|
||||||
import org.apache.hadoop.hbase.client.Put;
|
import org.apache.hadoop.hbase.client.Put;
|
||||||
|
@ -33,6 +34,7 @@ import org.apache.hadoop.mapreduce.OutputCommitter;
|
||||||
import org.apache.hadoop.mapreduce.OutputFormat;
|
import org.apache.hadoop.mapreduce.OutputFormat;
|
||||||
import org.apache.hadoop.mapreduce.RecordWriter;
|
import org.apache.hadoop.mapreduce.RecordWriter;
|
||||||
import org.apache.hadoop.mapreduce.TaskAttemptContext;
|
import org.apache.hadoop.mapreduce.TaskAttemptContext;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert Map/Reduce output and write it to an HBase table. The KEY is ignored
|
* Convert Map/Reduce output and write it to an HBase table. The KEY is ignored
|
||||||
|
@ -46,6 +48,14 @@ public class TableOutputFormat<KEY> extends OutputFormat<KEY, Writable> {
|
||||||
private final Log LOG = LogFactory.getLog(TableOutputFormat.class);
|
private final Log LOG = LogFactory.getLog(TableOutputFormat.class);
|
||||||
/** Job parameter that specifies the output table. */
|
/** Job parameter that specifies the output table. */
|
||||||
public static final String OUTPUT_TABLE = "hbase.mapred.outputtable";
|
public static final String OUTPUT_TABLE = "hbase.mapred.outputtable";
|
||||||
|
/** Optional job parameter to specify a peer cluster */
|
||||||
|
public static final String QUORUM_ADDRESS = "hbase.mapred.output.quorum";
|
||||||
|
/** Optional specification of the rs class name of the peer cluster */
|
||||||
|
public static final String
|
||||||
|
REGION_SERVER_CLASS = "hbase.mapred.output.rs.class";
|
||||||
|
/** Optional specification of the rs impl name of the peer cluster */
|
||||||
|
public static final String
|
||||||
|
REGION_SERVER_IMPL = "hbase.mapred.output.rs.impl";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes the reducer output to an HBase table.
|
* Writes the reducer output to an HBase table.
|
||||||
|
@ -111,12 +121,25 @@ public class TableOutputFormat<KEY> extends OutputFormat<KEY, Writable> {
|
||||||
TaskAttemptContext context)
|
TaskAttemptContext context)
|
||||||
throws IOException, InterruptedException {
|
throws IOException, InterruptedException {
|
||||||
// expecting exactly one path
|
// expecting exactly one path
|
||||||
String tableName = context.getConfiguration().get(OUTPUT_TABLE);
|
Configuration conf = new Configuration(context.getConfiguration());
|
||||||
|
String tableName = conf.get(OUTPUT_TABLE);
|
||||||
|
String address = conf.get(QUORUM_ADDRESS);
|
||||||
|
String serverClass = conf.get(REGION_SERVER_CLASS);
|
||||||
|
String serverImpl = conf.get(REGION_SERVER_IMPL);
|
||||||
HTable table = null;
|
HTable table = null;
|
||||||
try {
|
try {
|
||||||
HBaseConfiguration.addHbaseResources(context.getConfiguration());
|
HBaseConfiguration.addHbaseResources(conf);
|
||||||
table = new HTable(context.getConfiguration(),
|
if (address != null) {
|
||||||
tableName);
|
// Check is done in TMRU
|
||||||
|
String[] parts = address.split(":");
|
||||||
|
conf.set(HConstants.ZOOKEEPER_QUORUM, parts[0]);
|
||||||
|
conf.set(HConstants.ZOOKEEPER_ZNODE_PARENT, parts[1]);
|
||||||
|
}
|
||||||
|
if (serverClass != null) {
|
||||||
|
conf.set(HConstants.REGION_SERVER_CLASS, serverClass);
|
||||||
|
conf.set(HConstants.REGION_SERVER_IMPL, serverImpl);
|
||||||
|
}
|
||||||
|
table = new HTable(conf, tableName);
|
||||||
} catch(IOException e) {
|
} catch(IOException e) {
|
||||||
LOG.error(e);
|
LOG.error(e);
|
||||||
throw e;
|
throw e;
|
||||||
|
|
Loading…
Reference in New Issue