Oozie英文翻译为:驯象人。一个基于工作流引擎的开源框架,由Cloudera公司贡献给Apache,提供对Hadoop MapReduce、Pig Jobs的任务调度与协调。Oozie需要部署到Java Servlet容器中运行。主要用于定时调度任务,多任务可以按照执行的逻辑顺序调度。
第2章 Oozie的功能模块介绍
2.1模块 ==》实际上对应的3个 xml的文件
Workflow : 工作单位的协调依赖
顺序执行流程节点,支持fork(分支多个节点),join(合并多个节点为一个)
Coordinator : 协调调度器
定时触发workflow
Bundle : 多个workfole的调度问题, 少用
绑定多个Coordinator
2.2 Workflow常用节点
控制流节点(Control Flow Nodes)
控制流节点一般都是定义在工作流开始或者结束的位置,比如start,end,kill等。以及提供工作流的执行路径机制,如decision,fork,join等。
动作节点(Action Nodes)
负责执行具体动作的节点,比如:拷贝文件,执行某个Shell脚本等等。
基本的流程文件:
第3章 Oozie的部署
3.1 部署Hadoop(CDH版本的)
版本一:cloudera
CDH是Cloudera的Hadoop发行版,完全开源,比Apache Hadoop在兼容性,安全 性,稳定性上有所增强。
版本二:Hortonworks
Hortonworks 的主打产品是Hortonworks Data Platform (HDP),也同样是100%开 源的产品
3.1.2 修改Hadoop配置
core-site.xml
Hdfs.xml的配置 ==》second name 或者副本数
mapred-site.xml +=》指定历史服务地址
mapreduce.jobhistory.address hadoop102:10020 mapreduce.jobhistory.webapp.address hadoop102:19888yarn-site.xml
yarn.log.server.url http://hadoop102:19888/jobhistory/logs/完成后:记得scp同步到其他机器节点
格式化,一定要注意bin目录下的:bin/hdfs fs namenode format;
3.1.3 启动Hadoop集群
[atguigu@hadoop102 hadoop-2.5.0-cdh5.3.6]$ sbin/start-dfs.sh
[atguigu@hadoop103 hadoop-2.5.0-cdh5.3.6]$ sbin/start-yarn.sh
[atguigu@hadoop102 hadoop-2.5.0-cdh5.3.6]$ sbin/mr-jobhistory-daemon.sh start historyserver
注意:需要开启JobHistoryServer, 最好执行一个MR任务进行测试。
3.2 部署Oozie
3.2.1 解压Oozie==>根据视频安装
[atguigu@hadoop102 software]$ tar -zxvf /opt/software/cdh/oozie-4.0.0-cdh5.3.6.tar.gz -C /opt/module
3.2.2 在oozie根目录下解压自己的文件oozie-hadooplibs-4.0.0-cdh5.3.6.tar.gz
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ tar -zxvf oozie-hadooplibs-4.0.0-cdh5.3.6.tar.gz -C …/
完成后Oozie目录下会出现hadooplibs目录。
3.2.3 在Oozie目录下创建libext目录
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ mkdir libext/
3.2.4 拷贝依赖的Jar包到libext
1)将hadooplibs里面的jar包,拷贝到libext目录下:==》lib下面不要有路径
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ cp -ra hadooplibs/hadooplib-2.5.0-cdh5.3.6.oozie-4.0.0-cdh5.3.6/* libext/
2)拷贝Mysql驱动包到libext目录下:==》/mysql-connector-java-5.1.27
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ cp -a /opt/software/mysql-connector-java-5.1.27/mysql-connector-java-5.1.27-bin.jar ./libext/
3)将ext-2.2.zip拷贝到libext/目录下
ext是一个js框架,用于展示oozie前端页面:
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ cp -a /opt/software/cdh/ext-2.2.zip libext/
3.2.6 修改Oozie配置文件
oozie-site.xml
属性:oozie.service.JPAService.jdbc.driver
属性值:com.mysql.jdbc.Driver
解释:JDBC的驱动
属性:oozie.service.JPAService.jdbc.url
属性值:jdbc:mysql://hadoop102:3306/oozie ==》mysql中的数据库
解释:oozie所需的数据库地址
属性:oozie.service.JPAService.jdbc.username
属性值:root
解释:数据库用户名
属性:oozie.service.JPAService.jdbc.password
属性值:000000
解释:数据库密码
属性:oozie.service.HadoopAccessorService.hadoop.configurations
属性值:*=/opt/module/CDH/hadoop-2.5.0-cdh5.3.6/etc/hadoop
解释:让Oozie引用Hadoop的配置文件
3.2.7 在Mysql中创建Oozie的数据库
进入Mysql并创建oozie数据库:
$ mysql -uroot -p000000
mysql> create database oozie; ===》手动创建
3.2.8 初始化Oozie
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie-setup.sh sharelib create -fs hdfs://hadoop102:8020 -locallib oozie-sharelib-4.0.0-cdh5.3.6-yarn.tar.gz ==》目录文件
==.sharelib : 共享的lib包
执行成功之后,去50070检查对应目录有没有文件生成。
2) 执行oozie.sql文件建表
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/ooziedb.sh create -sqlfile oozie.sql -run
==》oozie.sql都是建表语句 脚本的执行语句
3)打包项目,生成war包 ==>web项目需要有war包才能启动web端
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie-setup.sh prepare-war
3.2.9 Oozie的启动与关闭
启动命令如下:
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozied.sh start
关闭命令如下:
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozied.sh stop
3.2.10 访问Oozie的Web页面
http://hadoop102:11000/oozie
第4章 Oozie的使用
4.1 案例一:Oozie调度shell脚本
目标:使用Oozie调度Shell脚本
分步实现:
1)创建工作目录
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ mkdir -p oozie-apps/shell
2)在oozie-apps/shell目录下创建两个文件——job.properties和workflow.xml文件
[atguigu@hadoop102 shell]$ touch workflow.xml
[atguigu@hadoop102 shell]$ touch job.properties
3)编辑job.properties和workflow.xml文件
job.properties
#HDFS地址
nameNode=hdfs://hadoop102:8020
#ResourceManager地址
jobTracker=hadoop103:8032 ===>yarn的角色:jobTRacker=resourcemanger
#队列名称
queueName=default ===》yarn容量调度器,队列的名称
examplesRoot=oozie-apps ==》keyvalue键值对,在路径中引用
oozie.wf.application.path=
n
a
m
e
N
o
d
e
/
u
s
e
r
/
{nameNode}/user/
nameNode/user/{user.name}/${examplesRoot}/shell
==》非常重要,是job/workflow的路径,这个要和HDFS对应,是上传的路径
执行顺序: 读取oozie-apps/shell/job.propertiess的配置文件
获取job在HDFS的地址,然后执行相应命令
workflow.xml
4)上传任务配置
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ /opt/module/cdh/hadoop-2.5.0-cdh5.3.6/bin/hadoop fs -put oozie-apps/ /user/atguigu
5)执行任务
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie job -oozie http://hadoop102:11000/oozie -config oozie-apps/shell/job.properties -run
==》config :本地路径下的job地址
==》-oozie http://hadoop102:11000/oozie 指定oozie的服务地址
6)杀死某个任务
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie job -oozie http://hadoop102:11000/oozie -kill 0000004-170425105153692-oozie-z-W
用mr的形式执行shell
4.2 案例二:Oozie逻辑调度执行多个Job
目标:使用Oozie执行多个Job调度
分步执行:
1)编辑job.properties和workflow.xml文件
job.properties
nameNode=hdfs://hadoop102:8020
jobTracker=hadoop103:8032
queueName=default
examplesRoot=oozie-apps
oozie.wf.application.path=
n
a
m
e
N
o
d
e
/
u
s
e
r
/
{nameNode}/user/
nameNode/user/{user.name}/
e
x
a
m
p
l
e
s
R
o
o
t
/
s
h
e
l
l
s
w
o
r
k
f
l
o
w
.
x
m
l
<
w
o
r
k
f
l
o
w
−
a
p
p
x
m
l
n
s
=
"
u
r
i
:
o
o
z
i
e
:
w
o
r
k
f
l
o
w
:
0.4
"
n
a
m
e
=
"
s
h
e
l
l
−
w
f
"
>
<
s
t
a
r
t
t
o
=
"
p
1
−
s
h
e
l
l
−
n
o
d
e
"
/
>
<
a
c
t
i
o
n
n
a
m
e
=
"
p
1
−
s
h
e
l
l
−
n
o
d
e
"
>
<
s
h
e
l
l
x
m
l
n
s
=
"
u
r
i
:
o
o
z
i
e
:
s
h
e
l
l
−
a
c
t
i
o
n
:
0.2
"
>
<
j
o
b
−
t
r
a
c
k
e
r
>
{examplesRoot}/shells workflow.xml <workflow-app xmlns="uri:oozie:workflow:0.4" name="shell-wf"> <start to="p1-shell-node"/> <action name="p1-shell-node"> <shell xmlns="uri:oozie:shell-action:0.2"> <job-tracker>
examplesRoot/shellsworkflow.xml<workflow−appxmlns="uri:oozie:workflow:0.4"name="shell−wf"><startto="p1−shell−node"/><actionname="p1−shell−node"><shellxmlns="uri:oozie:shell−action:0.2"><job−tracker>{jobTracker}
n
a
m
e
N
o
d
e
<
/
n
a
m
e
−
n
o
d
e
>
<
c
o
n
f
i
g
u
r
a
t
i
o
n
>
<
p
r
o
p
e
r
t
y
>
<
n
a
m
e
>
m
a
p
r
e
d
.
j
o
b
.
q
u
e
u
e
.
n
a
m
e
<
/
n
a
m
e
>
<
v
a
l
u
e
>
{nameNode}</name-node> <configuration> <property> <name>mapred.job.queue.name</name> <value>
nameNode</name−node><configuration><property><name>mapred.job.queue.name</name><value>{queueName}
mkdir
/opt/module/d1
<action name="p2-shell-node">
<shell xmlns="uri:oozie:shell-action:0.2">
<job-tracker>${jobTracker}</job-tracker>
<name-node>${nameNode}</name-node>
<configuration>
<property>
<name>mapred.job.queue.name</name>
<value>${queueName}</value>
</property>
</configuration>
<exec>mkdir</exec>
<argument>/opt/module/d2</argument>
<capture-output/>
</shell>
<ok to="end"/>
<error to="fail"/>
==》这种情况下,map端就不用配置了
,
通过yarn获取mapreduce中的所有属性,然后配置相应的位置
5)拷贝待执行的jar包到map-reduce的lib目录下
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ cp -a /opt /module/cdh/hadoop-2.5.0-cdh5.3.6/share/hadoop/mapreduce/hadoop-mapreduce-examples-2.5.0-cdh5.3.6.jar oozie-apps/map-reduce/lib
6)上传配置好的app文件夹到HDFS
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ /opt/module/cdh/hadoop-2.5.0-cdh5.3.6/bin/hdfs dfs -put oozie-apps/map-reduce/ /user/admin/oozie-apps
7)执行任务
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie job -oozie http://hadoop102:11000/oozie -config oozie-apps/map-reduce/job.properties -run
4.4 案例四:Oozie定时任务/循环任务
目标:Coordinator周期性调度任务
分步实现:==》在历史服务器中有时间戳,同步时间,需要先关闭历史服务器
1)配置Linux时区以及时间服务器
2)检查系统当前时区:
注意:如果显示的时区不是+0800,删除localtime文件夹后,再关联一个正确时区的链接过去,命令如下:
同步时间:
修改NTP配置文件:
去掉下面这行前面的# ,并把网段修改成自己的网段:
restrict 192.168.122.0 mask 255.255.255.0 nomodify notrap
注释掉以下几行:
#server 0.centos.pool.ntp.org
#server 1.centos.pool.ntp.org
#server 2.centos.pool.ntp.org
把下面两行前面的#号去掉,如果没有这两行内容,需要手动添加
server 127.127.1.0 # local clock
fudge 127.127.1.0 stratum 10
重启NTP服务:
注意,如果是centOS7以下的版本,使用命令:service ntpd start
注意,如果是centOS7以下的版本,使用命令:chkconfig ntpd on
集群其他节点去同步这台时间服务器时间:
首先需要关闭这两台计算机的ntp服务
centOS7以下,则:service ntpd stop
centOS7以下,则:chkconfig ntpd off
同步第一台服务器linux01的时间:
使用root用户制定计划任务,周期性同步时间:
*/10 * * * * /usr/sbin/ntpdate hadoop102
重启定时任务:
centOS7以下使用:service crond restart,
其他台机器的配置同理。
==》时间同步完成
3)配置oozie-site.xml文件
属性:oozie.processing.timezone
属性值:GMT+0800
解释:修改时区为东八区区时
注:该属性去oozie-default.xml中找到即可
4)修改js框架中的关于时间设置的代码
$ vi /opt/module/oozie-4.0.0-cdh5.3.6/oozie-server/webapps/oozie/oozie-console.js
修改如下:
function getTimeZone() {
Ext.state.Manager.setProvider(new Ext.state.CookieProvider());
return Ext.state.Manager.get(“TimezoneId”,“GMT”);
}
5)重启oozie服务,并重启浏览器(一定要注意清除缓存)
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozied.sh stop
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozied.sh start
6)拷贝官方模板配置定时任务
$ cp -r examples/apps/cron/ oozie-apps/
7)修改模板job.properties和coordinator.xml以及workflow.xml
job.properties
nameNode=hdfs://hadoop102:8020
jobTracker=hadoop103:8032
queueName=default
examplesRoot=oozie-apps
oozie.coord.application.path=
n
a
m
e
N
o
d
e
/
u
s
e
r
/
{nameNode}/user/
nameNode/user/{user.name}/KaTeX parse error: Expected 'EOF', got '#' at position 21: …plesRoot}/cron #̲start:必须设置为未来时间…{nameNode}/user/
u
s
e
r
.
n
a
m
e
/
{user.name}/
user.name/{examplesRoot}/cron
EXEC=p1.sh
coordinator.xml ==》定时调度
Oozie 这里最小5min; azkaban 最小1min
w
o
r
k
f
l
o
w
A
p
p
U
r
i
<
/
a
p
p
−
p
a
t
h
>
<
c
o
n
f
i
g
u
r
a
t
i
o
n
>
<
p
r
o
p
e
r
t
y
>
<
n
a
m
e
>
j
o
b
T
r
a
c
k
e
r
<
/
n
a
m
e
>
<
v
a
l
u
e
>
{workflowAppUri}</app-path> <configuration> <property> <name>jobTracker</name> <value>
workflowAppUri</app−path><configuration><property><name>jobTracker</name><value>{jobTracker}
nameNode
n
a
m
e
N
o
d
e
<
/
v
a
l
u
e
>
<
/
p
r
o
p
e
r
t
y
>
<
p
r
o
p
e
r
t
y
>
<
n
a
m
e
>
q
u
e
u
e
N
a
m
e
<
/
n
a
m
e
>
<
v
a
l
u
e
>
{nameNode}</value> </property> <property> <name>queueName</name> <value>
nameNode</value></property><property><name>queueName</name><value>{queueName}
workflow.xml ==》写的是自定义的脚本
j
o
b
T
r
a
c
k
e
r
<
/
j
o
b
−
t
r
a
c
k
e
r
>
<
n
a
m
e
−
n
o
d
e
>
{jobTracker}</job-tracker> <name-node>
jobTracker</job−tracker><name−node>{nameNode}
mapred.job.queue.name
q
u
e
u
e
N
a
m
e
<
/
v
a
l
u
e
>
<
/
p
r
o
p
e
r
t
y
>
<
/
c
o
n
f
i
g
u
r
a
t
i
o
n
>
<
e
x
e
c
>
{queueName}</value> </property> </configuration> <exec>
queueName</value></property></configuration><exec>{EXEC}
/user/atguigu/oozie-apps/cron/KaTeX parse error: Expected 'EOF', got '#' at position 7: {EXEC}#̲{EXEC}
Shell action failed, error message[${wf:errorMessage(wf:lastErrorNode())}]
8)上传配置
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ /opt/module/cdh/hadoop-2.5.0-cdh5.3.6/bin/hdfs dfs -put oozie-apps/cron/ /user/atguigu/oozie-apps
9)启动任务
[atguigu@hadoop102 oozie-4.0.0-cdh5.3.6]$ bin/oozie job -oozie http://hadoop102:11000/oozie -config oozie-apps/cron/job.properties -run
注意:oozie允许的最小执行任务的频率是5分钟
2
==》HIVE模板
第5章 常见问题总结
1)Mysql权限配置
授权所有主机可以使用root用户操作所有数据库和数据表
mysql> grant all on . to root@’%’ identified by ‘000000’;
mysql> flush privileges;
mysql> exit;
2)workflow.xml配置的时候不要忽略file属性
3)jps查看进程时,注意有没有bootstrap
4)关闭oozie
如果bin/oozied.sh stop无法关闭,则可以使用kill -9 [pid],之后oozie-server/temp/xxx.pid文件一定要删除。
5)Oozie重新打包时,一定要注意先关闭进程,删除对应文件夹下面的pid文件。(可以参考第4条目)
6)配置文件一定要生效
起始标签和结束标签无对应则不生效,配置文件的属性写错了,那么则执行默认的属性。
7)libext下边的jar存放于某个文件夹中,导致share/lib创建不成功。
8)调度任务时,找不到指定的脚本,可能是oozie-site.xml里面的Hadoop配置文件没有关联上。
9)修改Hadoop配置文件,需要重启集群。一定要记得scp到其他节点。
10)JobHistoryServer必须开启,集群要重启的。
11)Mysql配置如果没有生效的话,默认使用derby数据库。
12)在本地修改完成的job配置,必须重新上传到HDFS。
13)将HDFS中上传的oozie配置文件下载下来查看是否有错误。
14)Linux用户名和Hadoop的用户名不一致。