1.迁移数据量评估
通过hdfs dfs -du -h /
命令查看各目录总数据量。按照业务划分,统计各业务数据的总量
2.指定迁移节奏
因为数据量大,而且带宽有限,所以在HDFS文件变化之前全部迁移是不可能的,因为HDFS中的文件会随着业务每天变化。所以应该按照业务、分目录、分批次迁移
3.迁移工具
Hadoop自带的数据迁移工具DistCp,可以通过简单命令完成数据迁移
hadoop distcp hdfs://nn1:8020/data hdfs://nn2:8020/
4.迁移时间
因为老集群仍然在使用,所以建议在老集群低负载运行的时间段进行迁移
5.新老集群带宽
询问运维新老集群之间的最大传输带宽,多少的带宽可以尽量少的影响业务
是否可以对新老集群之间的网络进行改造,例如通过接网线的方式提高网络带宽
6.数据迁移前状态评估
先尝试小数据量迁移,可以先进行100G-1T的数据迁移,以评估迁移时会遇到的问题
DistCp(分布式拷贝)适用于大规模集群内部和集群之间拷贝的工具,它使用Map/Reduce实现文件分发,错误处理和恢复,以及报告生成。 它把文件和目录的列表作为map任务的输入,每个任务会完成源列表中部分文件的拷贝。 由于使用了Map/Reduce方法,这个工具在语义和执行上都会有特殊的地方。 这篇文档会为常用DistCp操作提供指南并阐述它的工作模型。
Distcp的本质是一个MapReduce任务,只有Map阶段,没有Reduce阶段,具备分布式执行的特性。在Map任务中从老集群读取数据,然后写入新集群,以此来完成数据迁移。
// org.apache.hadoop.tools.DistCp#main
// org.apache.hadoop.tools.DistCp#run
// org.apache.hadoop.tools.DistCp#execute
/**
* Implements the core-execution. Creates the file-list for copy,
* and launches the Hadoop-job, to do the copy.
* @return Job handle
* @throws Exception
*/
public Job execute() throws Exception {
Job job = createAndSubmitJob();
if (inputOptions.shouldBlock()) {
waitForJobCompletion(job);
}
return job;
}
/**
* Create Job object for submitting it, with all the configuration
*
* @return Reference to job object.
* @throws IOException - Exception if any
*/
private Job createJob() throws IOException {
String jobName = "distcp";
String userChosenName = getConf().get(JobContext.JOB_NAME);
if (userChosenName != null)
jobName += ": " + userChosenName;
Job job = Job.getInstance(getConf());
job.setJobName(jobName);
job.setInputFormatClass(DistCpUtils.getStrategy(getConf(), inputOptions));
job.setJarByClass(CopyMapper.class);
configureOutputFormat(job);
job.setMapperClass(CopyMapper.class);
// 无reduce阶段
job.setNumReduceTasks(0);
job.setMapOutputKeyClass(Text.class);
job.setMapOutputValueClass(Text.class);
job.setOutputFormatClass(CopyOutputFormat.class);
job.getConfiguration().set(JobContext.MAP_SPECULATIVE, "false");
job.getConfiguration().set(JobContext.NUM_MAPS,
String.valueOf(inputOptions.getMaxMaps()));
if (inputOptions.getSslConfigurationFile() != null) {
setupSSLConfig(job);
}
inputOptions.appendToConf(job.getConfiguration());
return job;
}
基本使用:
hadoop distcp hdfs://nn1:8020/foo/bar hdfs://nn2:8020/bar/foo
这个命令会把nn1集群的/foo/bar目录下的所有文件展开并存储到一个临时文件中,这些文件内容的拷贝工作被分配给多个map任务,然后每个TaskTracker分别执行从nn1到nn2的拷贝操作。
注意:注意DisCp使用绝对路径进行操作
bash$ hadoop distcp hdfs://nn1:8020/foo/a hdfs://nn1:8020/foo/b hdfs://nn2:8020/bar/foo
bash$ hadoop distcp -f hdfs://nn1:8020/srclist hdfs://nn2:8020/bar/foo
当从多个源拷贝时,如果两个源冲突,DistCp会停止拷贝并提示报错信息,如果在目的位置发生冲突,会根据选项设置解决,默认会跳过已经存在的目标文件(比如不用源文件做替换操作)。每次操作结束时,都会报告跳过的文件数目,但是如果某些拷贝操作失败了,但在之后的尝试成功了,那么报告的信息可能不准确。
每个Tasktracker必须能够与源端和目的端文件系统进行访问和交互。对于HDFS来说,源和目的端要运行相同版本的协议或者使用向下兼容的协议。
拷贝完成后,建议生成源端和目的端文件的列表,并交叉检查,来确认拷贝的真正成功。因为DistCp使用Map/Reduce和文件系统API进行操作,所以这三者或它们之间有任何问题,都会影响拷贝操作。一些DistCp命令的成功执行可以通过再次执行带-update参数的该命令来完成,但用户在如此操作之前应该对该命令的语法很熟悉
值得注意的是,当另一个客户端同时在向源文件写入时,拷贝很有可能会失败。尝试覆盖HDFS上正在被写入的文件的操作也会失败。如果一个源文件在拷贝之前被移动或删除了,拷贝失败同时输出异常FileNotFoundException
选项:
标识···································· | 描述 | 备注 |
---|---|---|
-p[rbugp] |
Preserve: r:replication /b:blocksize/ u:user /g:group /p:permission/ | 修改次数不会被保留。并且当指定-update时,更新的状态不会被同步,除非文件大小不同(比如文件被重新创建) |
-i |
忽略失败 | 这个选项会比默认情况提供关于拷贝的更精确的统计,同时它还将保留失败拷贝操作的日志,这些日志还可以用于调试。最后,如果一个map失败了,但并没有完成所有分块任务的尝试,这不会导致整个作业的失败 |
-log[logdir] |
记录日志到[logdir] | DistCp为每个文件的每次尝试拷贝操作都记录日志,并把日志作为map的输出。如果一个map失败了,当重新执行时这个日志不会被保留 |
-m |
同时拷贝的最大数目 | 指定了拷贝数据时map的数目,但注意,并不是map数越多吞吐量越大 |
-overwrite |
覆盖目标 | 如果一个map失败并且没有使用-i选项,不仅仅那些拷贝失败的文件,这个分块任务中的所有文件都会被重新拷贝,它会改变生成目标路径的语义,所以要小心使用这个选项 |
-f |
使用 作为源文件列表 |
这等价于把所有文件名列在命令行中。urilist_uri列表应该是完整合法的uri |
更新和覆盖
这里给出一些-update和-overwrite的例子。考虑一个从/foo/a 和 /foo/b 到 /bar/foo的拷贝,源路径包括:
hdfs://nn1:8020/foo/a
hdfs://nn1:8020/foo/a/aa
hdfs://nn1:8020/foo/a/ab
hdfs://nn1:8020/foo/b
hdfs://nn1:8020/foo/b/ba
hdfs://nn1:8020/foo/b/ab
如果没设置-update或-overwrite选项,那么两个源都会映射到目标端的/bar/foo/ab。如果设置了这两个选项,每个源目录的内容都会和目标目录的内容做比较。DistCp碰到这类冲突的情况会终止操作并退出
默认情况下,/bar/foo/a 和 /bar/foo/b目录都会被创建,所以并不会有冲突
现在考虑一个使用-update合法的操作:
distcp -update hdfs://nn1:8020/foo/a hdfs://nn1:8020/foo/b hdfs://nn2:8020/bar
其中源路径大小:
hdfs://nn1:8020/foo/a
hdfs://nn1:8020/foo/a/aa 32
hdfs://nn1:8020/foo/a/ab 32
hdfs://nn1:8020/foo/b
hdfs://nn1:8020/foo/b/ba 64
hdfs://nn1:8020/foo/b/bb 32
和目的路径/大小:
hdfs://nn2:8020/bar
hdfs://nn2:8020/bar/aa 32
hdfs://nn2:8020/bar/ba 32
hdfs://nn2:8020/bar/bb 64
会产生:
hdfs://nn2:8020/bar
hdfs://nn2:8020/bar/aa 32
hdfs://nn2:8020/bar/ab 32
hdfs://nn2:8020/bar/ba 64
hdfs://nn2:8020/bar/bb 32
只有nn2的aa文件没有被覆盖,如果指定了-overwrite选项,所有文件都会被覆盖
DistCp会尝试着均分需要拷贝的内容,这样每个map拷贝差不多相等大小的内容,但因为文件是最小的拷贝粒度,所以配置增加同时拷贝(如map)的数目不一定会增加实际同时拷贝的数目以及总吞吐量
如果没使用-m选项,DistCp会尝试在调度工作时指定map的数目为min(total_bytes / bytes.per.map, 20*num_task_trackers),其中bytes.per.map默认是256MB
建议对于长时间运行或定期运行的作业,根据源和目标集群大小、拷贝数量大小以及带宽调整map数目
对于不同Hadoop版本间的拷贝,用户应该使用HftpFileSystem。这是一个只读文件系统,所以DistCp必须运行在目标端集群上(更确切的说是能够写入目标集群的TaskTracker上)。源的格式是hftp://
(默认情况dfs.http.address是
)。
map拷贝输入文件失败时,会带来一些副效应
DistCp是一个MapReduce任务,如果在新集群上执行就向新集群的Yarn申请资源,老集群只有数据读取和网络传输的消耗
DistCp提供了-m
参数来设置map任务的最大数量(默认20),以提高并发性。注意这里要结合最大网络传输速率来设置
DistCp提供了-bandwidth
参数来控制单个Map任务的最大带宽,单位是MB
限速原理:
// org.apache.hadoop.tools.util.ThrottledInputStream#read()
public int read() throws IOException {
// 每次从源HDFS读取数据的时候,会进行限速
throttle();
int data = rawStream.read();
if (data != -1) {
bytesRead++;
}
return data;
}
private void throttle() throws IOException {
// getBytesPerSec()获取上次读取和此次读取这段时间内的速率(byte/s)与限速值作比较
while (getBytesPerSec() > maxBytesPerSec) {
try {
// 如果超速了则sleep一段时间
Thread.sleep(SLEEP_DURATION_MS);
totalSleepTime += SLEEP_DURATION_MS;
} catch (InterruptedException e) {
throw new IOException("Thread aborted", e);
}
}
}
DistCp负责进行CRC校验,可以通过-skipcrccheck参数来跳过校验来提高性能
DistCp提供了-p
参数来在新集群里保留状态(rbugpcaxt)(复制,块大小,用户,组,权限,校验和类型,ACL,XATTR,时间戳)。如果没有指定-p
参数,则文件权限是执行MapReduce任务的用户权限,迁移完成后需要手动执行chown命令变更
应该尽量避免这种情况出现,DistCp在任务启动的时候就会将需要copy的文件列表从源HDFS读取出来,如果迁移期间新增了文件,新增的文件就会被漏掉。删除文件则会导致改文件copy失败,可以通过-i参数忽略失败
Distcp提供了-overwrite 参数来覆盖已存在的文件。
删除掉新集群中的脏数据,重新执行迁移命令。不加-overwrite参数,来跳过已存在的文件。
Distcp提供-append参数将源HDFS文件的数据新增进去而不是覆盖它。