HDFS HA配置的完整步骤


部分内容参考了这篇文章: 。其中配置文件时基于我的集群信息做的,还增加了配置中遇到的问题和解决办法。

最近了解了Hadoop后,又开始涉及Hadoop 2.0相关的知识,所以后续我会陆陆续续把2.0相关的一些知识和学习总结整理上来。
这里是QJM方式的HA配置,众所周知,HDFS HA一般是基于NFS公共存储的,这里不采用NFS。
主要步骤:
1. zookeeper集群配置
2. HDFS HA参数配置
3. HDFS HA启动
4. 验证测试


1. zookeeper集群配置

说明: 我是在2台机器上做的配置,貌似zookeeper推荐用奇数台机器进行搭建,一般选3台。所以两台不推荐啊不推荐。
下载zookeeper的压缩包,并且解压,我的安装包目录是/search/zookeeper/zookeeper-3.4.5/,因为是HDFS 2.0,所以相应的zookeeper的版本也不能太低,免得出现兼容性问题。

(1) 首先建立zookeeper的数据目录,比如:


mkdir /search/zookeeper/dataDir

(2) 同时建立日志存放目录:

mkdir /search/zookeeper/dataDir/logs/

(3) 在zookeeper安装包中,/conf下的zoo_sample.cfg拷贝一份,命名为zoo.cfg,这是zookeeper的配置文件:

cp zoo_sample.cfg zoo.cfg

(4) 在zoo.cfg文件是进行配置:

tickTime=2000

initLimit=10

syncLimit=5

dataDir=/search/zookeeper/dataDir/

dataLogDir=/search/zookeeper/dataDir/logs/

clientPort=21810

#server.id in cluster = ip:port used in communicating with leader : port used when choosing a new leader

server.1=10.14.131.47:28880:38880

server.2=10.14.135.11:28880:38880

(5) 在每台服务器的dataDir目录下分别创建一个叫myid的文件,内容分别是1,2,如:

#在10.14.131.47上执行如下命令

echo 1 >/search/zookeeper/dataDir/myid

#在10.14.135.11上执行如下命令

echo 2 >/search/zookeeper/dataDir/myid

(6) 最后就是分别启动zookeeper服务了:

./bin/zkServer.sh start

(7) 通过jps命令可以检查是否启动成功:

1239 QuorumPeerMain

#看到QuorumPeerMain进程就表示zookeeper启动成功了。

(8) 测试zookeeper集群是否建立成功,在安装包目录下执行以下命令即可,如无报错表示集群创建成功:

./bin/zkCli.sh -server localhost:21810


2. HDFS2.0 HA参数配置

HA相关的配置是在core-site.xml和hdfs-site.xml中,当把这两个配置文件修改完成后,需要停止hadoop服务,然后同步修改后的配置信息到每个节点上。

说明: 这里的配置,都是基于已经成功配置了Hadoop 2.0集群的前提下,所以其他的一些配置信息没有列出。


(1) core-site.xml


<property>
			       <name>fs.defaultFS</name>
			       <value>hdfs://mycluster</value>
			  </property>
			
			  <property>
			       <name>ha.zookeeper.quorum</name>
			       <value>10.14.131.47:21810,10.14.135.11:21810</value>
			  </property>
			
			  <property>
			       <name>ha.zookeeper.session-timeout.ms</name>
			       <value>1000</value>
			       <description>ms</description>
			  </property>


  


(2) hdfs-site.xml


<!-- HA config start -->
			  <property>
			      <name>dfs.webhdfs.enabled</name>
			      <value>true</value>
			  </property>
			
			  <!-- the name service -->
			  <property>
			      <name>dfs.nameservices</name>
			      <value>mycluster</value>
			      <description>Logical name for this newnameservice</description>
			  </property>
			
			  <!-- the node in this name service -->
			  <property>
			      <name>dfs.ha.namenodes.mycluster</name>
			      <value>nn1,nn2</value>
			      <description>Unique identifiers for each NameNode in thenameservice</description>
			  </property>
			
			  <!-- the rpc port of each node -->
			  <property>
			      <name>dfs.namenode.rpc-address.mycluster.nn1</name>
			      <value>10.14.143.83:8022</value>
			  </property>
			  <property>
			      <name>dfs.namenode.rpc-address.mycluster.nn2</name>
			      <value>10.14.131.47:8022</value>
			  </property>
			
			  <!-- the http address of each namenode -->
			  <property>
			      <name>dfs.namenode.http-address.mycluster.nn1</name>
			      <value>10.14.143.83:50070</value>
			  </property>
			  <property>
			      <name>dfs.namenode.http-address.mycluster.nn2</name>
			      <value>10.14.131.47:50070</value>
			  </property>
			
			  <!-- shared storage -->
			  <property>
			      <name>dfs.namenode.shared.edits.dir</name>
			      <value>qjournal://10.14.131.47:8485;10.14.135.11:8485/mycluster</value>
			  </property>
			
			  <!-- the agent for HA auto switch -->
			  <property>
			      <name>dfs.client.failover.proxy.provider.mycluster</name>
			      <value>org.apache.hadoop.hdfs.server.namenode.ha.ConfiguredFailoverProxyProvider</value>
			  </property>
			
			  <!-- the fencing mathod -->
			  <property>
			      <name>dfs.ha.fencing.methods</name>
			      <value>sshfence</value>
			  </property>
			  <property>
			      <name>dfs.ha.fencing.ssh.private-key-files</name>
			      <value>/home/hadoop/.ssh/id_rsa</value>
			  </property>
			  <property>
			      <name>dfs.ha.fencing.ssh.connect-timeout</name>
			      <value>30000</value>
			  </property>
			
			  <!-- journalnode config -->
			  <property>
			      <name>dfs.journalnode.edits.dir</name>
			      <value>/search/hadoop01/journalnode/</value>
			  </property>
			  <!-- <property>
			      <name>dfs.journalnode.rpc-address</name>
			      <value>0.0.0.0:20022</value>
			  </property> -->
			
			  <!-- must be the HA auto switch or not -->
			  <property>
			      <name>dfs.ha.automatic-failover.enabled</name>
			      <value>true</value>
			  </property>
			  			
			  <!-- HA config end -->



3. HDFS HA启动

(1) 启动zookeeper集群: 在每个zookeeper的机器上执行:

./bin/zkServer.sh start

#据说集群会自动把第一个启动的机器作为leader。who knows ~~

  (2) 在某一个namenode节点执行如下命令,创建命名空间

./bin/hdfs zkfc -formatZK

  (3) 在各个节点用如下命令启日志程序

./sbin/hadoop-daemon.sh start journalnode

#如果Hadoop集群比较大的话,这真是个吃力不讨好的工作。所以我写了个脚本来实现它。

  (4) 在主namenode节点用./bin/hadoopnamenode -format格式化namenode和journalnode目录

./bin/hadoop namenode -format mycluster

#namenode节点在初始化时,也将自己的edits文件共享到了journalnode上。

(5) 在主namenode节点启动namenode进程

./sbin/hadoop-daemon.sh start namenode

  (6) 在备节点启动备namenode进程

a) ./bin/hdfs namenode –bootstrapStandby

#这个命令是把备namenode节点的目录格式化,并且把主namenode的元数据和journalnode上的edits文件同步过来,从而使元数据和主namenode保持一致,并且这个命令不会把journalnode目录再格式化。

ERROR: 不过我在执行bootstrapStandby时失败了,具体失败的体现是下一步启动备namenode失败,报错信息是缺少$dfs.tmp.dir/data/下的name目录。因为name目录下保存的都是namenode的基本元数据,所以说明这一步的格式化没有成功。

SOLUTION:  手工把主namenode上的元数据($dfs.namenode.name.dir下的所有数据)拷贝到备namenode下的同目录下,然后再执行bootstrapStandby.

b) ./sbin/hadoop-daemon.sh start namenode

#启动备namenode。

(7) 在两个namenode节点都启动ZKFC:

./sbin/hadoop-daemon.sh start zkfc

#启动DFSZKFailoverController,也就是用于监控namenode并且保持HA状态的那个watchdog。

(8) 在所有datanode节点都执行以下命令启动datanode

./sbin/hadoop-daemon.sh start datanode

#这个步骤略坑爹,因为节点很多。最好直接在namenode节点上执行 sbin/hadoop-daemons.sh start datanode  启动。

ERROR: 启动Datanode可能失败,失败报错信息为

2014-03-13 11:26:30,788 FATAL org.apache.hadoop.hdfs.server.datanode.DataNode: Initialization failed for block pool Block pool BP-1257313099-10.10.208.38-1394679083528 (storage id DS-743638901-127.0.0.1-50010-1394616048958) service to Linux-hadoop-38/10.10.208.38:9000

java.io.IOException: Incompatible clusterIDs in /usr/local/hadoop/tmp/dfs/data: namenode clusterID = CID-8e201022-6faa-440a-b61c-290e4ccfb006; datanode clusterID = clustername

错误提示很明显,Datanode节点初始化失败,和hadoop.tmp.dir目录有关系。一般是因为,我们的Hadoop集群不是首次启动,所以版本信息产生冲突。


SOLUTION: 对比master节点上dfs.namenode.name.dir目录下,current文件夹中的VERSION文件的内容,和slave节点上hadoop.tmp.dir目录下,data/current/目录下的VERSION文件的内容,会发现他们的内容不一致,很有可能是这个VERSION是Datanode在没有配置HA之前的集群中使用的VERSION,所以Datanode无法正常启动。删除slave上的VERSION文件,再启动就成功了。


(9) 通过web界面查看两个namenode的状态,如果都处在standby状态的话,就把其中一个namenode切换成Active状态。

hdfs haadmin -failover --forcefence --forceactive <serviceId> <serviceId>  

#这个命令的作用我不是很清楚,到底哪个serviceId会被切换成Active。或者用下面的命令:

bin/hdfs haadmin -transitionToActive nn1

此时进行检查,HDFS集群应该已经全部正常启动了。下次再启动时,直接执行./sbin/start-dfs.sh就好。关闭直接执行./sbin/stop-dfs.sh就好。

 

4. 测试HA的功能

(1) 在Active namenode节点上先jps找到namenode的进程号,然后kill -9杀死namenode进程。观察standby namenode节点的状态变化。

jps

kill -9 pid_namenode

(2) 观察standby节点是否自动切换成了Active:

通过web界面查看,括号内显示的状态从standby变成了active;

观察zkfc日志(logs/.....-zkfc.......log),最后一行出现如下日志,表示切换成功:

INFOorg.apache.hadoop.ha.ZKFailoverController: Successfully transitioned NameNodeat hd0/192.168.111.130:53310 to active state

(3) 启动被kill掉的namenode

./sbin/hadoop-daemon.sh start namenode

(4) 观察该namenode节点的状态:

通过web界面查看,括号内显示的状态为standby;

观察zkfc日志(logs/.....-zkfc.......log),最后一行出现如下日志,表示该节点作为standby节点已经正常启动:

INFOorg.apache.hadoop.ha.ZKFailoverController: Successfully transitioned NameNodeat hd2/192.168.111.132:53310 to standby state