Airflow 使用随笔(内含 TimeZone 和 Backfill 等的详解)
其实怎么部署 airflow 又哪些特性,然后功能又是如何全面都可以在 Reference 的文章里面找到,都不是重点这里就不赘述了。
这里重点谈一下我在部署完成仔细阅读文档之后觉得可以总结的一些东西,或者踩到的一些坑。
首选明确 airflow 中最重要的几个概念:
DAG
DAG 意为有向无循环图,在 Airflow 中则定义了整个完整的作业。同一个 DAG 中的所有 Task 拥有相同的调度时间。
Task
Task 为 DAG 中具体的作业任务,它必须存在于某一个 DAG 之中。Task 在 DAG 中配置依赖关系,跨 DAG 的依赖是可行的,但是并不推荐。跨 DAG 依赖会导致 DAG 图的直观性降低,并给依赖管理带来麻烦。
DAG Run
当一个 DAG 满足它的调度时间,或者被外部触发时,就会产生一个 DAG Run。可以理解为由 DAG 实例化的实例。
Task Instance
当一个 Task 被调度启动时,就会产生一个 Task Instance。可以理解为由 Task 实例化的实例。
在使用过程中,需要谨记这些概念不然很容易就被绕晕。
TimeZone:
时区问题可以说是 airflow 中根本绕不开的一个问题,官方文档也拿整整一章来阐述时区给 airflow 带来的影响。
官方默认使用 UTC+0 时间来作为 airflow 的时间,并且在 airflow 提供的 webserver 上,这个时区直到现在 1.10.1 版本上依旧是无法被配置的。所以我们在 server 上看到的时间默认都是 UTC+0 的时间。但是系统启动的时间,以及写入数据库元数据的时间确是可以配置时区的。他的格式遵循 IANA 格式。可以通过配置 airflow 的配置文件 airflow.cfg 改变这个时区。例如我在启动的时候就配置了中国时间:
# Default timezone in case supplied date times are naive
# can be utc (default), system, or any IANA timezone string (e.g. Europe/Amsterdam)
#default_timezone = utc
default_timezone = Asia/Chongqing
当配置了这个生效之后,我们给任务配置的启动时间,以及 runtime 之类的时间会遵循这个时间(虽然在 ui 界面上看还是 UTC+0 的非常分裂。。。)
例如我们来看一个执行任务:
可以看到 Last Run 字段指明了该字段是 2018-12-09 16:00 被执行了。按照我们刚才的配置 UTC+8 应该是 10号的 0点被执行了。我们来看下数据库里是否真的是这样呢?
可以通过下图发现完全按照我们预期:
mysql> select * from task_instance where task_id = 'sensors'\G;
*************************** . row ***************************
task_id: sensors
dag_id: sensors_dag
execution_date: -- ::00.000000
start_date: -- ::04.547631
end_date: -- ::50.686937
duration: 286.139
state: success
try_number:
hostname: zed-2
unixname: apache-airflow
job_id:
pool: NULL
queue: default
priority_weight:
operator: BashOperator
queued_dttm: -- ::03.302928
pid:
max_tries:
executor_config: }q .
所以说,证明了只要我们设置了刚才的参数之后,只有 ui 界面依然继续展示 UTC+0 的时间,其实我们真正的执行操作的时区已经被修改过来了。
另外还需要注意的一个地方在于任务的申明那里,在申明一个 dag 的时候同样需要指明自己的时区,否则一脸懵逼为啥到了时间为什么没有正常调度起来。
import pendulum
import datetime local_tz = pendulum.timezone("Asia/Chongqing") # start 接收一个 %Y-%m-%d %H:%M:%S 的字符串时间
def init_default_args(owner, start_time, retry=, email=None):
if not email:
email = []
if not isinstance(start_time, datetime.datetime):
raise TypeError('start_date must be datetime.') d = start_time.replace(tzinfo=local_tz) return {
'owner': owner,
'depends_on_past': False,
'start_date': d,
'email': email,
'email_on_failure': False,
'email_on_retry': False,
'retries': retry,
'retry_delay': datetime.timedelta(minutes=),
}
这里我在申明默认 default_args 参数的时候就显示的指明了时区并且赋值给 start_date 让他开始的时间复合我们的预期。
Backfill:
感觉这个也非常让人懵逼值得拿出来多多少少谈一下。
光从字面意思来理解作为一个中国人我真的是一下没有 get 到他的点,但是后来去查了一下文档结合他的行为看了一下可以把它简单的理解成当你错过了某一次执行时间之后,往回去补充执行的行为。我们可以使用手动方法来执行这个行为
airflow backfill sensors -s -- -e --
他会回补这个时间段开始的 和 -e 后面时间段结束期间所有的任务执行。回补的意思就是把没有执行的操作都执行一遍。
这个特性想法很好,但是自自动触发的时候不注意就会产生非常不可预期的问题。
比如刚才在上面我们谈到的在给 dag 配置的时候指定的 default_args 上面有一个参数 start_date。如果我们不给 dag 指定不回补,那么 airflow 会默认回补从系统当前时间到我们指定的 start_date 期间的任务。如果这个参数设置得不恰当会打来恐怖的回补,所以一般我都会禁用回补。
sensors_dag = DAG(
'sensors_dag',
default_args=default_args,
schedule_interval=u'0 0 * * *',
catchup=False)
指定 catchup=False 。让他从最新的任务时间点开始执行。这个在官方文档 Scheduling & Triggers 一章有详细提到。
Operator:
现在这一章我只有一个地方觉得有点坑。可能是我还没有深度使用吧
由于 airflow 支持模版 jinja 的模版功能,所以说在使用 BashOperator 的时候要注意自己的写法,官方文档对此有描述
Troubleshooting
Jinja template not found
Add a space after the script name when directly calling a Bash script with the bash_command argument. This is because Airflow tries to apply a Jinja template to it, which will fail. t2 = BashOperator(
task_id='bash_example', # This fails with `Jinja template not found` error
# bash_command="/home/batcher/test.sh", # This works (has a space after)
bash_command="/home/batcher/test.sh ",
dag=dag)
我觉得这个。。。还蛮坑的- -
这一章其实应该有蛮多的内容可以延伸,包括社区也给 airflow 贡献了非常多的 Operator。等我深度使用之后再来 backfill 吧哈哈哈哈。
--------------------------------------------分割线--------------------------------------------
最近遇到一个这样的问题,我们有一次 scheduler 死了,然后一下启起来发现全部任务被一块执行了。甚至每个 dag 多天没有跑完的任务直接起来,把我们的服务器直接按死了。
后面我们需要让该类情况得到控制,至少让一个 dag 同一时刻只有一天被调度起来,而不是单一 dag 多天同时被拉起来,这样直接把服务器打挂的概率非常高。
需要将全局文件 airflow.cfg 的参数 max_active_runs_per_dag 从默认的 16 改为 1
Reference:
https://airflow.apache.org airflow 官方文档
https://zhuanlan.zhihu.com/p/44768244 如何部署一个健壮的 apache-airflow 调度系统
http://www.shuhegroup.com/newsmedias/%E6%B5%85%E8%B0%88%E8%B0%83%E5%BA%A6%E5%B7%A5%E5%85%B7airflow/ 浅谈调度工具——Airflow
https://stackoverflow.com/questions/49231340/how-to-limit-airflow-to-run-only-1-dag-run-at-a-time How to limit Airflow to run only 1 DAG run at a time?
https://airflow.apache.org/faq.html
Airflow 使用随笔(内含 TimeZone 和 Backfill 等的详解)的更多相关文章
- Android随笔之——闹钟制作铺垫之AlarmManager详解
说实话,之前写的两篇博客Android广播机制Broadcast详解.Android时间.日期相关类和方法以及现在要写的,都算是为之后要写的闹钟应用做铺垫,有兴趣的话,大家可以去看看前两篇博客. 一. ...
- PyQt(Python+Qt)学习随笔:富文本编辑器QTextEdit功能详解
专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 一.概述 QTextEdit是一个高级的所见即所得的文档查看器和编辑器 ...
- Linux随笔 - Linux LVM逻辑卷配置过程详解[转载]
许多Linux使用者安装操作系统时都会遇到这样的困境:如何精确评估和分配各个硬盘分区的容量,如果当初评估不准确,一旦系统分区不够用时可能不得不备份.删除相关数据,甚至被迫重新规划分区并重装操作系统,以 ...
- PyQt(Python+Qt)学习随笔:纯文本编辑器QPlainTextEdit功能详解
专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 一.概述 QPlainTextEdit是用于纯文本的一个高级文档编辑器 ...
- 【图文+视频新手也友好】Java一维数组详细讲解(内含练习题答案+详解彩蛋喔~)
目录 视频讲解: 一.数组的概述 二.一维数组的使用 三.Arrays工具类中的sort方法(sort方法用的多,我们具体讲一下) 四.数组中的常见异常 五.一维数组练习题 六.彩蛋(本期视频使用的P ...
- Android随笔之——PackageManager详解
参考:http://www.cnblogs.com/xingfuzzhd/p/3374504.html 今天要讲的是PackageManager.Android系统为我们提供了很多服务管理的类,包括A ...
- [转载] Android随笔之——PackageManager详解
本文转载自: http://www.cnblogs.com/travellife/p/3932823.html 参考:http://www.cnblogs.com/xingfuzzhd/p/33745 ...
- php.ini中date.timezone设置详解
date.timezone设置php5默认date.timezone为utc,改为date.timezone = PRC即可解决时间相差八小时的问题,但我在php的官方文档中看了半天也没找到这个参数啊 ...
- 【转】Android随笔之——PackageManager详解
参考:http://www.cnblogs.com/xingfuzzhd/p/3374504.html 今天要讲的是PackageManager.Android系统为我们提供了很多服务管理的类,包括A ...
随机推荐
- 强化学习(三)—— 时序差分法(SARSA和Q-Learning)
1.时序差分法基本概念 虽然蒙特卡洛方法可以在不知道状态转移概率矩阵的前提下,灵活地求解强化学习问题,但是蒙特卡洛方法需要所有的采样序列都是完整的状态序列.如果我们没有完整的状态序列就无法用蒙特卡洛方 ...
- shzr要填的各种坑
shzr要填的各种坑 如果大家看到我学了什么算法没写总结,或者做了什么题没写题解,欢迎让我填坑. 计划要写的: [ ] 点分治 [ ] 整体二分 [ ] CDQ分治 [ ] Min-Max容斥 [√] ...
- 自然周与自然月的Hive统计SQL
按照周或者月统计活跃数: 周: SELECT week, COUNT(DISTINCT pin), business_type FROM ( SELECT DISTINCT user_log_acct ...
- ASCII,unicode, utf8 ,big5 ,gb2312,gbk,gb18030等几种常用编码区别(转载)
原文出处:http://www.blogjava.net/xcp/archive/2009/10/29/coding2.html 最近老为编码问题而烦燥,下定决心一定要将其弄明白!本文主要总结网上一些 ...
- VC++6.0 add files to project 造成Visual Studio崩溃的解决方法
1.下载filetool.exe,然后将文件解压在一个小文件夹内2.打开filetool.dsw 在release模式下编译程序,复制filetool.dll3.放在VC6.0安装目录AddIns的下 ...
- 1、Django系列之web应用与http协议
第1节:最简单的web应用程序 Web应用程序指供浏览器访问的程序,通常也简称为Web应用.应用程序有两种模式C/S.B/S.C/S是客户端/服务器端程序,也就是说这类程序一般独立运行.而B/S就是浏 ...
- Linux查看端口
1.lsof -i:端口号 用于查看某一端口的占用情况,比如查看8000端口使用情况,lsof -i:8000 2.netstat -tunlp |grep 端口号 用于查看指定的端口号的进程情况 ...
- codeforces#552 D. Vanya and Triangles(几何)
题意:给出n个不同的点,问能组成多少个不同的三角形 题解:对于每个点对,我们生成一个直线,用a*x+b=y表示,用map记录ab,这样就确定了一个直线,这样我们就能算出有多少点是共线的,这样复杂度就是 ...
- hdu 1730 Nim博弈
题目来源:http://acm.hdu.edu.cn/showproblem.php?pid=1730 Nim博弈为:n堆石子,每个人可以在任意一堆中取任意数量的石子 n个数异或值为0就后手赢,否则先 ...
- C++ string中的find()函数
1.string中find()返回值是字母在母串中的位置(下标记录),如果没有找到,那么会返回一个特别的标记npos.(返回值可以看成是一个int型的数) #include<cstring> ...