Skip to content

Commit

Permalink
add indirect execution context access
Browse files Browse the repository at this point in the history
  • Loading branch information
alangenfeld committed Jun 26, 2023
1 parent f3b70f9 commit 37be28c
Show file tree
Hide file tree
Showing 3 changed files with 76 additions and 40 deletions.
29 changes: 25 additions & 4 deletions python_modules/dagster/dagster/_core/execution/context/compute.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
from abc import ABC, abstractmethod
from contextlib import contextmanager
from contextvars import ContextVar
from typing import (
AbstractSet,
Any,
Expand Down Expand Up @@ -47,6 +49,8 @@

from .system import StepExecutionContext

8


class AbstractComputeExecutionContext(ABC):
"""Base class for op context implemented by OpExecutionContext and DagstermillExecutionContext.
Expand Down Expand Up @@ -734,9 +738,26 @@ def get_asset_provenance(self, asset_key: AssetKey) -> Optional[DataProvenance]:
return super().get_asset_provenance(asset_key)


def build_execution_context(
_current_context: ContextVar[Optional[OpExecutionContext]] = ContextVar(
"execution_context", default=None
)


@contextmanager
def enter_execution_context(
step_context: StepExecutionContext,
) -> Union[OpExecutionContext, AssetExecutionContext]:
) -> Iterator[Union[OpExecutionContext, AssetExecutionContext]]:
if step_context.is_sda_step:
return AssetExecutionContext(step_context)
return OpExecutionContext(step_context)
ctx = AssetExecutionContext(step_context)
else:
ctx = OpExecutionContext(step_context)

token = _current_context.set(ctx)
try:
yield ctx
finally:
_current_context.reset(token)


def get_execution_context():
return _current_context.get()
72 changes: 36 additions & 36 deletions python_modules/dagster/dagster/_core/execution/plan/compute.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@
from dagster._core.definitions.op_definition import OpComputeFunction
from dagster._core.errors import DagsterExecutionStepExecutionError, DagsterInvariantViolationError
from dagster._core.events import DagsterEvent
from dagster._core.execution.context.compute import build_execution_context
from dagster._core.execution.context.compute import enter_execution_context
from dagster._core.execution.context.system import StepExecutionContext
from dagster._core.system_config.objects import ResolvedRunConfig
from dagster._utils import iterate_with_context
Expand Down Expand Up @@ -135,46 +135,46 @@ def _yield_compute_results(
) -> Iterator[OpOutputUnion]:
check.inst_param(step_context, "step_context", StepExecutionContext)

context = build_execution_context(step_context)
user_event_generator = compute_fn(context, inputs)

if isinstance(user_event_generator, Output):
raise DagsterInvariantViolationError(
(
"Compute function for {described_node} returned an Output rather than "
"yielding it. The compute_fn of the {node_type} must yield "
"its results"
).format(
described_node=step_context.describe_op(),
node_type=step_context.op_def.node_type_str,
with enter_execution_context(step_context) as context:
user_event_generator = compute_fn(context, inputs)

if isinstance(user_event_generator, Output):
raise DagsterInvariantViolationError(
(
"Compute function for {described_node} returned an Output rather than "
"yielding it. The compute_fn of the {node_type} must yield "
"its results"
).format(
described_node=step_context.describe_op(),
node_type=step_context.op_def.node_type_str,
)
)
)

if user_event_generator is None:
return

if inspect.isasyncgen(user_event_generator):
user_event_generator = gen_from_async_gen(user_event_generator)

op_label = step_context.describe_op()
if user_event_generator is None:
return

if inspect.isasyncgen(user_event_generator):
user_event_generator = gen_from_async_gen(user_event_generator)

op_label = step_context.describe_op()

for event in iterate_with_context(
lambda: op_execution_error_boundary(
DagsterExecutionStepExecutionError,
msg_fn=lambda: f"Error occurred while executing {op_label}:",
step_context=step_context,
step_key=step_context.step.key,
op_def_name=step_context.op_def.name,
op_name=step_context.op.name,
),
user_event_generator,
):
if context.has_events():
yield from context.consume_events()
yield _validate_event(event, step_context)

for event in iterate_with_context(
lambda: op_execution_error_boundary(
DagsterExecutionStepExecutionError,
msg_fn=lambda: f"Error occurred while executing {op_label}:",
step_context=step_context,
step_key=step_context.step.key,
op_def_name=step_context.op_def.name,
op_name=step_context.op.name,
),
user_event_generator,
):
if context.has_events():
yield from context.consume_events()
yield _validate_event(event, step_context)

if context.has_events():
yield from context.consume_events()


def execute_core_compute(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
from dagster import OpExecutionContext, job, op
from dagster._core.definitions.job_definition import JobDefinition
from dagster._core.definitions.op_definition import OpDefinition
from dagster._core.execution.context.compute import get_execution_context
from dagster._core.storage.dagster_run import DagsterRun


Expand All @@ -20,3 +21,17 @@ def foo():
ctx_op()

assert foo.execute_in_process().success


def test_get_context():
assert get_execution_context() is None

@op
def o(context):
assert context == get_execution_context()

@job
def j():
o()

assert j.execute_in_process().success

0 comments on commit 37be28c

Please sign in to comment.