HDFS源码分析之FSImage文件内容(一)总体格式
FSImage文件是HDFS中名字节点NameNode上文件/目录元数据在特定某一时刻的持久化存储文件。它的作用不言而喻,在HA出现之前,NameNode因为各种原因宕机后,若要恢复或在其他机器上重启NameNode,重新组织元数据,就需要加载对应的FSImage文件、FSEditLog文件,并在内存中重做FSEditLog文件中的事务条目。本节,我们先来看下FSImage文件格式,及其内部数据是如何组织的。
通过翻看HDFS中加载FSImage文件的代码,从FSNamesystem的loadFSImage()方法开始,我将HDFS集群上的一个FSImage文件放到本地Windows系统中的F盘下,并写了如下方法解析文件,并打印关键内容,如下:
- import java.io.IOException;
- import java.io.File;
- import java.util.List;
- import org.junit.Test;
- import java.io.ByteArrayInputStream;
- import java.io.RandomAccessFile;
- import org.apache.hadoop.hdfs.server.namenode.FsImageProto.FileSummary;
- import org.apache.hadoop.hdfs.server.namenode.FsImageProto.FileSummary.Section;
- public class TestImageUtil {
- @Test
- public void testImage() {
- // 文件头字符串HDFSIMG1对应byte[]
- byte[] fileHead = "HDFSIMG1".getBytes();
- RandomAccessFile raFile = null;
- try {
- // 创建文件file,对应为f盘下FSImage文件fsimage_0000000000002311798
- File file = new File("f:/fsimage_0000000000002311798");
- raFile = new RandomAccessFile(file, "r");
- // 文件summary长度域所占大小为4
- final int FILE_LENGTH_FIELD_SIZE = 4;
- System.out.println("文件summary长度域大小:FILE_LENGTH_FIELD_SIZE=" + FILE_LENGTH_FIELD_SIZE);
- // 获取FSImage文件长度
- long fileLength = raFile.length();
- System.out.println("获取FSImage文件长度:fileLength=" + fileLength);
- // 创建文件头byte[]数组fileHeadTmp,用于存储文件头byte[]数组,大小为上述fileHead数组大小
- byte[] fileHeadTmp = new byte[fileHead.length];
- // 读入文件头至byte[]数组fileHeadTmp
- System.out.println("文件从头开始读取" + fileHeadTmp.length + "个byte至byte[]数组fileHeadTmp");
- raFile.readFully(fileHeadTmp);
- // 获取文件头长度
- System.out.println("获取文件头长度:fileHeadLength=" + fileHead.length);
- // 将byte[]数组fileHeadTmp转换成字符串fileHeadString
- String fileHeadString = new String(fileHeadTmp);
- // 验证文件头字符串
- System.out.println("fileHeadString=" + fileHeadString);
- // 文件file通过raFile.seek()方法定位到文件summary长度字段起始处,即文件大小减去文件summary长度域所占字节数4
- raFile.seek(fileLength - FILE_LENGTH_FIELD_SIZE);
- System.out.println("文件定位到文件summary长度开始处:" + (fileLength - FILE_LENGTH_FIELD_SIZE));
- // 读入一个int,即文件长度summaryLength
- int summaryLength = raFile.readInt();
- System.out.println("获取文件summary部分长度:summaryLength=" + summaryLength);
- // 文件file通过raFile.seek()方法定位到文件summary部分开始处,即文件大小减去文件长度所占字节数4,再减去文件内容总长度
- raFile.seek(fileLength - FILE_LENGTH_FIELD_SIZE - summaryLength);
- System.out.println("文件定位到文件summary部分开始处:" + (fileLength - FILE_LENGTH_FIELD_SIZE - summaryLength));
- // 再从当前位置开始读入文件summary部分内容
- // 构造文件长度summaryLength大小的byte[]数组
- byte[] summaryBytes = new byte[summaryLength];
- // 读取文件内容至数组summaryBytes
- raFile.readFully(summaryBytes);
- System.out.println("从当前位置开始读入文件summary部分内容至summaryBytes数组");
- FileSummary summary = FileSummary
- .parseDelimitedFrom(new ByteArrayInputStream(summaryBytes));
- System.out.println("解析文件summary部分内容如下:");
- System.out.println("1、ondiskVersion=" + summary.getOndiskVersion());
- System.out.println("2、layoutVersion=" + summary.getLayoutVersion());
- System.out.println("3、codec=" + summary.getCodec());
- System.out.println("4、section");
- List<Section> sectionsList = summary.getSectionsList();
- for (Section section : sectionsList) {
- System.out.println(" ");
- System.out.println("name=" + section.getName());
- System.out.println("length=" + section.getLength());
- System.out.println("offset=" + section.getOffset());
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- if (raFile != null) {
- try {
- raFile.close();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
- }
- /**
- * Supported section name. The order of the enum determines the order of
- * loading.
- */
- public enum SectionName {
- NS_INFO("NS_INFO"), STRING_TABLE("STRING_TABLE"), EXTENDED_ACL(
- "EXTENDED_ACL"), INODE("INODE"), INODE_REFERENCE(
- "INODE_REFERENCE"), SNAPSHOT("SNAPSHOT"), INODE_DIR("INODE_DIR"), FILES_UNDERCONSTRUCTION(
- "FILES_UNDERCONSTRUCTION"), SNAPSHOT_DIFF("SNAPSHOT_DIFF"), SECRET_MANAGER(
- "SECRET_MANAGER"), CACHE_MANAGER("CACHE_MANAGER");
- private static final SectionName[] values = SectionName.values();
- public static SectionName fromString(String name) {
- for (SectionName n : values) {
- if (n.name.equals(name))
- return n;
- }
- return null;
- }
- private final String name;
- private SectionName(String name) {
- this.name = name;
- }
- }
- }
关于代码解释,我们会在专门的FSImage文件加载源码分析相关文章中进行详细介绍,本文只关注FSImage文件的总体格式。
执行上述方法,打印内容输出如下:
- 文件summary长度域大小:FILE_LENGTH_FIELD_SIZE=4
- 获取FSImage文件长度:fileLength=1154156
- 文件从头开始读取8个byte至byte[]数组fileHeadTmp
- 获取文件头长度:fileHeadLength=8
- fileHeadString=HDFSIMG1
- 文件定位到文件summary长度开始处:1154152
- 获取文件summary部分长度:summaryLength=231
- 文件定位到文件summary部分开始处:1153921
- 从当前位置开始读入文件summary部分内容至summaryBytes数组
- 解析文件summary部分内容如下:
- 1、ondiskVersion=1
- 2、layoutVersion=-60
- 3、codec=
- 4、section
- name=NS_INFO
- length=27
- offset=8
- name=INODE
- length=1093067
- offset=35
- name=INODE_DIR
- length=60225
- offset=1093102
- name=FILES_UNDERCONSTRUCTION
- length=345
- offset=1153327
- name=SNAPSHOT
- length=68
- offset=1153672
- name=SNAPSHOT_DIFF
- length=36
- offset=1153740
- name=INODE_REFERENCE
- length=0
- offset=1153776
- name=SECRET_MANAGER
- length=9
- offset=1153776
- name=CACHE_MANAGER
- length=7
- offset=1153785
- name=STRING_TABLE
- length=129
- offset=1153792
不难看出,文件的总长度为1154156,这与我通过windows系统下右击-属性的方式查看结果是一致的,如下:
(一)文件的起始位置(下标我们从0开始),0-7处为文件头信息,占8个byte的"HDFSIMG1";
(二)然后是接下来是10个section区域,这部分在FSImage文件中所占起止位置为8-1153920,这些是根据下面的summary区域的分析得到的结论,section分别如下:
1、8-34:占27个byte的section--NS_INFO,命名系统NameSystem信息section区域,具体内容后续文章再讲;
2、35-1093101:占1093067个byte的section--INODE,HDFS中INODE节点section区域,具体内容后续文章再讲;
3、1093102-1153326:占60225个byte的section--INODE_DIR,HDFS中INODE目录节点section区域,具体内容后续文章再讲;
4、1153327-1153671:占345个byte的section--FILES_UNDERCONSTRUCTION,HDFS中FILES_UNDERCONSTRUCTION处于构建状态文件部分section区域,具体内容后续文章再讲;
5、1153672-1153739:占68个byte的section--SNAPSHOT,HDFS中SNAPSHOT快照部分section区域,具体内容后续文章再讲;
6、1153740-1153775:占36个byte的section--SNAPSHOT_DIFF,HDFS中SNAPSHOT_DIFF部分section区域,具体内容后续文章再讲;
7、1153776-?:占0个byte的section--INODE_REFERENCE,HDFS中INODE_REFERENCE节点引用部分section区域,具体内容后续文章再讲,实际上本文件中没有这部分,为了体现FSImage文件的完整性,还是增加这部分的描述;
8、1153776-1153784:占9个byte的section--SECRET_MANAGER,HDFS中SECRET_MANAGER部分section区域,具体内容后续文章再讲;
9、1153785-1153791:占7个byte的section--CACHE_MANAGER,HDFS中CACHE_MANAGER部分section区域,具体内容后续文章再讲;
10、1153792-1153920:占129个byte的section--STRING_TABLE,HDFS中STRING_TABLE部分section区域,具体内容后续文章再讲;
(三)再接下来是文件summary区域,这部分在FSImage文件中所占起止位置为1153921-1154151,长度为231,它主要标识了上述各section区域的区域名name、在FSImage文件所占长度length及其起始位置offset,另外还有三个十分总要的变量,FSImage文件在磁盘上的版本号ondiskVersion、布局layout版本号layoutVersion及其解压/压缩器codec,前面两个会在load文件时与HDFS中NameNode进程内存中的版本号分别进行校验,防止错误版本的FSImage文件被加载,而codec则用于如何加载各个section区域,为空默认不做任何解压/压缩处理;
(四)最后为文件summary部分所占长度区域,这部分在FSImage文件中所占起止位置为1154152-1154155,正好是文件的最后一部分内容。
或许通过图的方式你会看的更直观,但是请原谅我拙劣的画图技巧:
实际上,FSImage文件中各个区域包含的内容,采用的是Google的protobuf编码格式,而protobuf不单单是一种消息传输格式,你也可以把它理解为一种数据编码格式,所以各个区域数据格式,在HDFS内的fsimage.proto文件中也有所阐述,比如FileSummary:
- message FileSummary {
- // The version of the above EBNF grammars.
- required uint32 ondiskVersion = 1;
- // layoutVersion describes which features are available in the
- // FSImage.
- required uint32 layoutVersion = 2;
- optional string codec = 3;
- // index for each section
- message Section {
- optional string name = 1;
- optional uint64 length = 2;
- optional uint64 offset = 3;
- }
- repeated Section sections = 4;
- }
它就包含我们上面所描述的ondiskVersion、layoutVersion、codec、sections五部分,最后的sections是可以重复的,即repeated,而每个section又是一个message,包含name、length、offset三部分,正和我们上面解析的结果一致。
又如StringTableSection:
- /**
- * This section maps string to id
- * NAME: STRING_TABLE
- */
- message StringTableSection {
- message Entry {
- optional uint32 id = 1;
- optional string str = 2;
- }
- optional uint32 numEntry = 1;
- // repeated Entry
- }
包含两部分,Entry数量:numEntry,和重复的Entry,每个Entry又是一个Message,包含id和str两部分。
以上就是FSImage文件的主体信息,至于文件中的详细内容,特别是每个不同section区域中都有哪些内容,尤其是复杂的INodeSection等,我们后续再讲!
HDFS源码分析之FSImage文件内容(一)总体格式的更多相关文章
- HDFS源码分析数据块校验之DataBlockScanner
DataBlockScanner是运行在数据节点DataNode上的一个后台线程.它为所有的块池管理块扫描.针对每个块池,一个BlockPoolSliceScanner对象将会被创建,其运行在一个单独 ...
- HDFS源码分析EditLog之读取操作符
在<HDFS源码分析EditLog之获取编辑日志输入流>一文中,我们详细了解了如何获取编辑日志输入流EditLogInputStream.在我们得到编辑日志输入流后,是不是就该从输入流中获 ...
- HDFS源码分析之数据块及副本状态BlockUCState、ReplicaState
关于数据块.副本的介绍,请参考文章<HDFS源码分析之数据块Block.副本Replica>. 一.数据块状态BlockUCState 数据块状态用枚举类BlockUCState来表示,代 ...
- HDFS源码分析EditLog之获取编辑日志输入流
在<HDFS源码分析之EditLogTailer>一文中,我们详细了解了编辑日志跟踪器EditLogTailer的实现,介绍了其内部编辑日志追踪线程EditLogTailerThread的 ...
- HDFS源码分析心跳汇报之数据块汇报
在<HDFS源码分析心跳汇报之数据块增量汇报>一文中,我们详细介绍了数据块增量汇报的内容,了解到它是时间间隔更长的正常数据块汇报周期内一个smaller的数据块汇报,它负责将DataNod ...
- Yii2.0源码分析之——控制器文件分析(Controller.php)创建动作、执行动作
在Yii中,当请求一个Url的时候,首先在application中获取request信息,然后由request通过urlManager解析出route,再在Module中根据route来创建contr ...
- HDFS源码分析之UnderReplicatedBlocks(一)
http://blog.csdn.net/lipeng_bigdata/article/details/51160359 UnderReplicatedBlocks是HDFS中关于块复制的一个重要数据 ...
- HDFS源码分析数据块复制监控线程ReplicationMonitor(二)
HDFS源码分析数据块复制监控线程ReplicationMonitor(二)
- HDFS源码分析数据块复制监控线程ReplicationMonitor(一)
ReplicationMonitor是HDFS中关于数据块复制的监控线程,它的主要作用就是计算DataNode工作,并将复制请求超时的块重新加入到待调度队列.其定义及作为线程核心的run()方法如下: ...
随机推荐
- Linux文本过滤命令grep、awk、sed
grep的使用: 一.grep一般格式: grep [选项] 基本正则表达式 [文件] 这里的正则表达式可以为字符串.在grep命令中输入字符串参数时,最好将其用双引号括起来.调用变量时也可以使用双引 ...
- 2. LVS/DR 配置
平台:RedHat Enterprise Linux centos6.3 ipvsadm ipvs 1.DR模型 DR模型:直接路由模型,每个Real Server ...
- Tomcat Deployment failure ,locked one or more files
在用Eclipse+Tomcat配置J2EE项目时,出现如下提示错误: Undeployment Failure could not be redeployed because it could no ...
- C#设计模式---观察者模式简单例子
在开发过程中经常遇到一个模块中的 一个方法调用了其他模块中相关的方法 比如说在一个系统中,如果出现了错误,就调用专门进行错误处理的模块中的方法进行错误处理 而因为错误处理的操作有很多,所以将这些具体的 ...
- windows 用wireshark抓本机的包
原文: http://bijian1013.iteye.com/blog/2299856 1.也可以用另外一个工具: RawCap 当然也不是说windows下就别想抓到本地回路的包了,肯定有别的方法 ...
- Android之短信验证码
我们今天所使用的方案仅仅是android手机设备集成短信验证码功能的方案之中的一个. 我们所採用的方案是使用聚合数据的短信验证sdk. 程序的界面例如以下所看到的: 实现步骤: 1.到聚合数据官网上申 ...
- Recycling Settings for an Application Pool <recycling>
Overview The <recycling> element contains configuration settings that control the conditions t ...
- JavaScript 数组去重 方法汇总
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...
- 非阻塞socket中read、write返回值
read返回值 >0 读取数据的长度 =0 接收到对端发送的FIN,表示对端的写端关闭. <0 如果errno=EINTR.收到信号并从信号处理函数返回时,慢系统调用会返回并设 ...
- TCP/IP协议分析(推荐)
一;前言 学习过TCP/IP协议的人多有一种感觉,这东西太抽象了,没有什么数据实例,看完不久就忘了.本文将介绍一种直观的学习方法,利用协议分析工具学习TCP/IP,在学习的过程中能直观的看到数据的具体 ...