## Licensed to the Apache Software Foundation (ASF) under one# or more contributor license agreements. See the NOTICE file# distributed with this work for additional information# regarding copyright ownership. The ASF licenses this file# to you under the Apache License, Version 2.0 (the# "License"); you may not use this file except in compliance# with the License. You may obtain a copy of the License at## http://www.apache.org/licenses/LICENSE-2.0## Unless required by applicable law or agreed to in writing,# software distributed under the License is distributed on an# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY# KIND, either express or implied. See the License for the# specific language governing permissions and limitations# under the License.from__future__importannotationsfromtypingimportTYPE_CHECKINGfromairflow.exceptionsimportAirflowSkipExceptionfromairflow.sensors.baseimportBaseSensorOperatorfromairflow.triggers.temporalimportDateTimeTriggerfromairflow.utilsimporttimezoneifTYPE_CHECKING:fromairflow.utils.contextimportContext
[docs]classTimeDeltaSensor(BaseSensorOperator):""" Waits for a timedelta after the run's data interval. :param delta: time length to wait after the data interval before succeeding. .. seealso:: For more information on how to use this sensor, take a look at the guide: :ref:`howto/operator:TimeDeltaSensor` """def__init__(self,*,delta,**kwargs):super().__init__(**kwargs)self.delta=delta
[docs]defpoke(self,context:Context):target_dttm=context["data_interval_end"]target_dttm+=self.deltaself.log.info("Checking if the time (%s) has come",target_dttm)returntimezone.utcnow()>target_dttm
[docs]classTimeDeltaSensorAsync(TimeDeltaSensor):""" A deferrable drop-in replacement for TimeDeltaSensor. Will defers itself to avoid taking up a worker slot while it is waiting. :param delta: time length to wait after the data interval before succeeding. .. seealso:: For more information on how to use this sensor, take a look at the guide: :ref:`howto/operator:TimeDeltaSensorAsync` """
[docs]defexecute(self,context:Context):target_dttm=context["data_interval_end"]target_dttm+=self.deltatry:trigger=DateTimeTrigger(moment=target_dttm)except(TypeError,ValueError)ase:ifself.soft_fail:raiseAirflowSkipException("Skipping due to soft_fail is set to True.")fromeraiseself.defer(trigger=trigger,method_name="execute_complete")
[docs]defexecute_complete(self,context,event=None):"""Execute for when the trigger fires - return immediately."""returnNone