Scheduling

This lesson clarifies the working of schedule_interval and start_date, which can be confusing for complex crontab expressions.

When initially working with Airflow, it is common to get confused with how all the scheduling parameters work together. In this lesson, we’ll explore the differences between the various parameters.

When creating a DAG, we can specify the start_date and a schedule_interval as parameters to the constructor. Let’s see an example:

dag = DAG(
    'Example9',
    default_args=default_args,
    description='Example DAG 9',
    schedule_interval='@daily',
    start_date=datetime(2020, 9, 5))

In the Example9 DAG, we set start_date to 5th Sept 2020, and the schedule_interval is set to @daily. Note that @daily is an alias for the 0 0 * * * crontab expression. There are other aliases for commonly used schedules, such as @weekly, @monthly, and @yearly, which all translate to crontab expressions under the hood. You can provide a crontab expression for the schedule_interval parameter for complex schedules. A good resource to work with crontab expressions is crontab.guru. Remember that Airflow works with UTC by default but can be configured to work with your local time too. Airflow will also schedule DAG runs for the previous days even though we are running the DAG now. Each DAG run will be associated with an execution date and a start date. The execution date is the date that the DAG should have run, and the start date is when Airflow actually runs it. Please don’t confuse the start_date that we pass into the DAG constructor with the start date associated with a DAG run; both are distinct.

The combination of start_date and schedule_interval implies that the DAG Example9 should have run starting from 5th Sept 2020 up until now. Let’s further assume that now is 9th Sept 2020 i.e., you are running the DAG on 9th of Sept 2020. Airflow is smart enough to run the missing DAGs for 5th, 6th, 7th, and 8th of Sept for you too. All these four runs will have a start date of Sept. 9th (since all of them were kicked off on Sept. 9th) but will all have different timestamps on the 9th. Additionally, the execution dates for these DAGs will span from Sept. 5th to Sept. 8th.

As you can see from the screenshot above, Airflow created DAG runs for each of the prior days. The execution date for each DAG run is distinct, but the start dates for all of the runs is the 9th of Sept. Sure, the hours, minutes, etc., differ because they are run at different times on the 9th of Sept. The astute reader would observe that the run for the 9th of Sept. is missing from the list in the screenshot above. This is because Airflow has its roots in ETL, which involves running batch jobs at the end of the day for that day. For instance, the data for the 4th of July will be collected for the entire day, and the ETL job for the 4th of July will actually run at 12 a.m. on the 5th of July. This makes sense because the job for a day should wait to have all the data for that day before running. In our example, the DAG run for the 9th of Sept. isn’t executed until 12 a.m. on the 10th of Sept., which is why it doesn’t show up in the listing in the screenshot. Thinking on the same lines, if the schedule for a DAG is 3 p.m. every day, then the DAG run for the previous day will run immediately after 3 p.m. the next day (and not 12 a.m.). There will be a full 24 hour delay before the DAG run for the previous day executes.

It is interesting to consider what happens if you set the schedule_interval to run every Monday and Friday. Let’s say we set the start_date for the DAG as August 15th, 2020.

Day Date DAG Execution Date DAG Start Date
Sunday Aug. 15th, 2020 - -
Monday Aug. 16th, 2020 - -
Tuesday Aug. 17th, 2020 - -
Wednesday Aug. 18th, 2020 - -
Thursday Aug. 19th, 2020 - -
Friday Aug. 20th, 2020 Aug. 16th, 2020 Aug. 20th, 2020
Saturday Aug. 21st, 2020 - -
Sunday Aug. 22nd, 2020 - -
Monday Aug. 23rd, 2020 Aug. 20th, 2020 Aug. 23rd, 2020
Tuesday Aug. 24th, 2020 - -
Wednesday Aug. 25th, 2020 - -
Thursday Aug. 26th, 2020 - -
Friday Aug. 27th, 2020 Aug. 20th, 2020 Aug. 27th, 2020
Saturday Aug. 28th, 2020 - -

The DAG’s start date is set to Aug. 15th, 2020 (not to be confused with a DAG run’s start date, as shown in the table above). The first DAG run is for Aug. 16th, 2020, but it’ll start on Friday, Aug. 20th, 2020! This is an idiosyncrasy of Airflow that can confuse seasoned engineers. The DAG runs for Aug. 16th, 2020 actually runs on Aug. 20th, 2020; the execution date for the DAG is Aug. 16th, 2020, but the start date is Aug. 20th, 2020. The DAG run with the execution date of Aug. 20th, 2020 will have a start date of next Monday, i.e., Aug. 23rd, 2020, and so on and so forth.

Finally, you may note that a DAG doesn’t execute at exactly the time it is supposed to run, e.g., a DAG run may start at 10:01 p.m. when its schedule asks it to run at 10:00 p.m. There may be a delay of a few seconds or so. There’s a configuration parameter, scheduler_heartbeat_sec, defined in airflow.cfg that controls how often the Airflow scheduler runs. The scheduler runs and looks for tasks to trigger, and there may be a delay in when a task becomes due and when the scheduler is able to run it. Making the scheduler run at a higher frequency can put pressure on the database, so any tweaks should be done cautiously.

Get hands-on with 1300+ tech skills courses.