Skip to content

PipelineRunner

laktory.dispatcher.DLTPipelineRunner ¤

Bases: DispatcherRunner

DLT Pipeline runner.

METHOD DESCRIPTION
get_id

Get deployed pipeline id

run

Run remote pipeline and monitor failures.

Functions¤

get_id ¤

get_id()

Get deployed pipeline id

Source code in laktory/dispatcher/dltpipelinerunner.py
26
27
28
29
30
31
32
33
34
def get_id(self) -> str:
    """Get deployed pipeline id"""
    logger.info(f"Getting id for pipeline {self.name}")
    for _pl in self.wc.pipelines.list_pipelines(
        filter=f"name LIKE '{self.name}'", max_results=1
    ):
        if self.name == _pl.name:
            self.id = _pl.pipeline_id
            return self.id

run ¤

run(wait=True, timeout=20 * 60, full_refresh=False, raise_exception=False, current_run_action='WAIT')

Run remote pipeline and monitor failures.

PARAMETER DESCRIPTION
wait

If True, runs synchronously and wait for completion.

TYPE: bool DEFAULT: True

timeout

Maximum time allowed for the pipeline to run before an exception is raised.

TYPE: int DEFAULT: 20 * 60

full_refresh

If True, tables are fully refreshed (re-built). Otherwise, only increments are processed.

TYPE: bool DEFAULT: False

raise_exception

If True, exceptions are raised if the pipeline fails.

TYPE: bool DEFAULT: False

current_run_action

Action to take with respect to current run (if any). Possible options are: - WAIT: wait for the current run to complete - CANCEL: cancel the current run - FAIL: raise an exception

TYPE: Literal['WAIT', 'CANCEL', 'FAIL'] DEFAULT: 'WAIT'

RETURNS DESCRIPTION
output

None

Source code in laktory/dispatcher/dltpipelinerunner.py
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
def run(
    self,
    wait: bool = True,
    timeout: int = 20 * 60,
    full_refresh: bool = False,
    raise_exception: bool = False,
    current_run_action: Literal["WAIT", "CANCEL", "FAIL"] = "WAIT",
):
    """
    Run remote pipeline and monitor failures.

    Parameters
    ----------
    wait:
        If `True`, runs synchronously and wait for completion.
    timeout:
        Maximum time allowed for the pipeline to run before an exception is
        raised.
    full_refresh:
        If `True`, tables are fully refreshed (re-built). Otherwise, only
        increments are processed.
    raise_exception:
        If `True`, exceptions are raised if the pipeline fails.
    current_run_action:
        Action to take with respect to current run (if any). Possible
        options are:
            - WAIT: wait for the current run to complete
            - CANCEL: cancel the current run
            - FAIL: raise an exception

    Returns
    -------
    output:
        None
    """
    from databricks.sdk.core import DatabricksError
    from databricks.sdk.service.pipelines import EventLevel
    from databricks.sdk.service.pipelines import UpdateInfoState

    event_ids = []

    # Start update
    t0 = time.time()
    try:
        self._update_start = self.wc.pipelines.start_update(
            pipeline_id=self.id,
            full_refresh=full_refresh,
            # validate_only=False,
        )
        logger.info(f"Pipeline {self.name} update started...")

    except DatabricksError as e:
        # Exception is raised when update is currently in progress
        if current_run_action.upper() == "FAIL":
            logger.info(f"Pipeline {self.name} already running...")
            raise e

        elif current_run_action.upper() == "WAIT":
            logger.info(
                f"Pipeline {self.name} waiting for current update to be completed..."
            )
            self.wc.pipelines.wait_get_pipeline_idle(pipeline_id=self.id)

        elif current_run_action.upper() == "CANCEL":
            logger.info(f"Pipeline {self.name} cancelling current update...")
            self.wc.pipelines.stop_and_wait(pipeline_id=self.id)

        logger.info(f"Pipeline {self.name} update started...")
        self._update_start = self.wc.pipelines.start_update(
            pipeline_id=self.id,
            full_refresh=full_refresh,
            # validate_only=False,
        )

    pstate = None
    self.get_update()
    logger.info(f"Pipeline {self.name} run URL: {self.update_url}")
    if wait:
        while (
            time.time() - t0 < timeout
            or self.update_state == UpdateInfoState.COMPLETED
        ):
            self.get_update()

            if self.update_state != pstate:
                logger.info(
                    f"Pipeline {self.name} state: {self.update_state.value}"
                )
                pstate = self.update_state

            # filter don't seem to work
            for event in self.wc.pipelines.list_pipeline_events(
                pipeline_id=self.id,
                max_results=100,
                # filter=f"(level in ('ERROR', 'WARN')) AND (timestamp > '{utc_datetime(t0).isoformat()}Z')"
                # filter=f"level in ('ERROR', 'WARN')",
                # filter=f"timestamp > '{utc_datetime(t0).isoformat()}Z'"
                page_token=None,
            ):
                if (
                    event.id in event_ids
                    or unix_timestamp(event.timestamp) < t0
                    or event.origin.update_id != self.update_id
                ):
                    continue

                if event.level == EventLevel.WARN:
                    logger.warn(event.message)
                elif event.level == EventLevel.ERROR:
                    logger.error(event.message)
                event_ids += [event.id]

            if self.update_state in [
                UpdateInfoState.CANCELED,
                UpdateInfoState.COMPLETED,
                UpdateInfoState.FAILED,
            ]:
                break

            time.sleep(1.0)

        logger.info(
            f"Pipeline {self.name} update terminated after {time.time() - t0: 5.2f} sec with {self.update_state}"
        )
        if raise_exception and self.update_state != UpdateInfoState.COMPLETED:
            raise Exception(
                f"Pipeline {self.name} update not completed ({self.update_state})"
            )